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

com.ibm.batch.container.impl.RetryHandler Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2012 International Business Machines Corp.
 * 
 * See the NOTICE file distributed with this work for additional information
 * regarding copyright ownership. 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 com.ibm.batch.container.impl;

import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;

import jsr352.batch.jsl.Chunk;
import jsr352.batch.jsl.ExceptionClassFilter;

import com.ibm.batch.container.artifact.proxy.RetryProcessListenerProxy;
import com.ibm.batch.container.artifact.proxy.RetryReadListenerProxy;
import com.ibm.batch.container.artifact.proxy.RetryWriteListenerProxy;
import com.ibm.batch.container.exception.BatchContainerRuntimeException;

public class RetryHandler {

	/**
	 *
	 * Logic for handling retryable records.
	 *
	 * A RetryHandler object is attached to every BDS that inherits from AbstractBatchDataStream.
	 *
	 */

	private static final String className = RetryHandler.class.getName();
	private static Logger logger = Logger.getLogger(RetryHandler.class.getPackage().getName());

	public static final String RETRY_COUNT      = "retry-limit";
	public static final String RETRY_INCLUDE_EX = "include class";
	public static final String RETRY_EXCLUDE_EX = "exclude class";

	private static final int RETRY_NONE = 0;
	private static final int RETRY_READ = 1;
	private static final int RETRY_PROCESS = 2;
	private static final int RETRY_WRITE = 3;

	private int retryType = RETRY_NONE;

	/*private RetryProcessListenerProxy _retryProcessListener = null;
	  private RetryReadListenerProxy _retryReadListener = null;
	  private RetryWriteListenerProxy _retryWriteListener = null;*/

	List _retryProcessListeners = null;
	List _retryReadListeners = null;
	List _retryWriteListeners = null;

	private long _jobId = 0;
	private String _stepId = null;
	private Set _retryNoRBIncludeExceptions = null;
	private Set _retryNoRBExcludeExceptions = null;
	private Set _retryIncludeExceptions = null;
	private Set _retryExcludeExceptions = null;
	private int _retryLimit = 0;
	private long _retryCount = 0;
	private Exception _retryException = null;

	public RetryHandler(Chunk chunk, long l, String stepId)
	{
		_jobId = l;
		_stepId = stepId;

		initialize(chunk);
	}


	/**
	 * Add the user-defined RetryProcessListener.
	 *
	 */
	public void addRetryProcessListener(List retryProcessListeners)
	{
		_retryProcessListeners =  retryProcessListeners;
	}

	/**
	 * Add the user-defined RetryReadListener.
	 *
	 */
	public void addRetryReadListener(List retryReadListeners)
	{
		_retryReadListeners = retryReadListeners;
	}

	/**
	 * Add the user-defined RetryWriteListener.
	 *
	 */
	public void addRetryWriteListener(List retryWriteListeners)
	{
		_retryWriteListeners = retryWriteListeners;
	}


	/**
	 * Read the retry exception lists from the BDS props.
	 */
	private void initialize(Chunk chunk)
	{
		final String mName = "initialize";

		if(logger.isLoggable(Level.FINER)) 
			logger.entering(className, mName);

		try
		{
			if (chunk.getRetryLimit() != null){
				_retryLimit = Integer.parseInt(chunk.getRetryLimit());
			}
		}
		catch (NumberFormatException nfe)
		{
			throw new RuntimeException("NumberFormatException reading " + RETRY_COUNT, nfe);
		}

		if (_retryLimit > 0)
		{
			// Read the include/exclude exceptions.
			_retryIncludeExceptions = new HashSet();
			_retryExcludeExceptions = new HashSet();
			_retryNoRBIncludeExceptions = new HashSet();
			_retryNoRBExcludeExceptions = new HashSet();

			String includeEx = null;
			String excludeEx = null;
			String includeExNoRB = null;
			String excludeExNoRB = null;

			if (chunk.getRetryableExceptionClasses() != null) {
				if (chunk.getRetryableExceptionClasses().getIncludeList() != null) {
					List includes = chunk.getRetryableExceptionClasses().getIncludeList();
					if (includes.size() > 1) {
						String msg = "TODO: Do not currently support >1  element, even though spec allows this.";
						logger.severe(msg);
						throw new IllegalArgumentException(msg);
					} else if (includes.size() == 1) {
						includeEx = includes.get(0).getClazz();
						logger.finer("RETRYHANDLE: include: " + includeEx);
					}  else {
						logger.finer("RETRYHANDLE: include element not present");
					}
				}
				if (chunk.getRetryableExceptionClasses().getExcludeList() != null) {
					List excludes = chunk.getRetryableExceptionClasses().getExcludeList();
					if (excludes.size() > 1) {
						String msg = "TODO: Do not currently support >1  element, even though spec allows this.";
						logger.severe(msg);
						throw new IllegalArgumentException(msg);
					} else if (excludes.size() == 1) {
						excludeEx = excludes.get(0).getClazz();
						logger.finer("RETRYHANDLE: exclude: " + excludeEx);
					}  else {
						logger.finer("RETRYHANDLE: exclude element not present");
					}
				}
			}

			if (chunk.getNoRollbackExceptionClasses() != null) {
				if (chunk.getNoRollbackExceptionClasses().getIncludeList() != null) {
					List includes = chunk.getNoRollbackExceptionClasses().getIncludeList();
					if (includes.size() > 1) {
						String msg = "TODO: Do not currently support >1  element, even though spec allows this.";
						logger.severe(msg);
						throw new IllegalArgumentException(msg);
					} else if (includes.size() == 1) {
						includeExNoRB = includes.get(0).getClazz();
						logger.finer("RETRYHANDLE: include no rollback: " + includeExNoRB);
					}  else {
						logger.finer("RETRYHANDLE: include element not present");
					}
				}
				if (chunk.getNoRollbackExceptionClasses().getExcludeList() != null) {
					List excludes = chunk.getNoRollbackExceptionClasses().getExcludeList();
					if (excludes.size() > 1) {
						String msg = "TODO: Do not currently support >1  element, even though spec allows this.";
						logger.severe(msg);
						throw new IllegalArgumentException(msg);
					} else if (excludes.size() == 1) {
						excludeExNoRB = excludes.get(0).getClazz();
						logger.finer("RETRYHANDLE: exclude no rollback: " + excludeExNoRB);
					}  else {
						logger.finer("RETRYHANDLE: include element not present");
					}
				}
			}

			if (includeEx != null) {
				_retryIncludeExceptions.add(includeEx.trim());
			}
			if (excludeEx != null) {
				_retryExcludeExceptions.add(excludeEx.trim());
			}
			if (includeExNoRB != null) {
				_retryNoRBIncludeExceptions.add(includeExNoRB.trim());
			}
			if (excludeExNoRB != null) {
				_retryNoRBExcludeExceptions.add(excludeExNoRB.trim());
			}

			if (logger.isLoggable(Level.FINE)) {
				logger.logp(Level.FINE, className, mName,
						"added include exception " + includeEx
								+ "; added exclude exception " + excludeEx);
				logger.logp(Level.FINE, className, mName,
						"added include no rollback exception " + includeExNoRB
								+ "; added exclude no rollback exception " + excludeExNoRB);
			}
		}
	        
	    if(logger.isLoggable(Level.FINER)) {
	      logger.exiting(className, mName, this.toString());
	    }
	  }
	  
	  public boolean isRollbackException(Exception e)
	  {
		  return !isNoRollbackException(e);
	  }
	  /**
	   * Handle exception from a read failure.
	   */
	  public void handleExceptionRead(Exception e)
	  {
	    final String mName = "handleExceptionRead";
	    
	    logger.finer("RETRYHANDLE: in retryhandler handle exception on a read:" + e.toString());

	    if(logger.isLoggable(Level.FINER)) 
	      logger.logp(Level.FINE, className, mName, e.getClass().getName() + "; " + this.toString());
	    
	    if (!isRetryLimitReached() && isRetryable(e))
	    {
	       retryType = RETRY_READ;
	       _retryException = e;
	      // Retry it.  Log it.  Call the RetryListener.
	      ++_retryCount;
	      logRetry(e);

	      if (_retryReadListeners != null) {
	    	  for (RetryReadListenerProxy retryReadListenerProxy : _retryReadListeners) {
	    		  retryReadListenerProxy.onRetryReadException(e);
				}
	      }
	    }
	    else
	    {
	      // No retry.  Throw it back.
	      if(logger.isLoggable(Level.FINER)) 
	        logger.logp(Level.FINE, className, mName, "No retry.  Rethrow", e);
	      	throw new BatchContainerRuntimeException(e);
	    }

	    if(logger.isLoggable(Level.FINER)) 
	      logger.exiting(className, mName, e);
	  }

	  /** 
	   * Handle exception from a process failure.
	   */
	  public void handleExceptionProcess(Exception e, Object w)
	  {
	    final String mName = "handleExceptionProcess";
	    
	    if(logger.isLoggable(Level.FINER)) 
	      logger.logp(Level.FINE, className, mName, e.getClass().getName() + "; " + this.toString());
	    
	    if (!isRetryLimitReached() && isRetryable(e))
	    {
	      retryType = RETRY_PROCESS;
	      _retryException = e;
	      // Retry it.  Log it.  Call the RetryListener.
	      ++_retryCount;
	      logRetry(e);

	      if (_retryProcessListeners != null) {
	    	  for (RetryProcessListenerProxy retryProcessListenerProxy : _retryProcessListeners) {
	    		  retryProcessListenerProxy.onRetryProcessException(w, e);
				}
	      }
	    }
	    else
	    {
	      // No retry.  Throw it back.
	      if(logger.isLoggable(Level.FINER)) 
	        logger.logp(Level.FINE, className, mName, "No retry.  Rethrow ", e);
	      throw new BatchContainerRuntimeException(e);
	    }
	  }
	  
	  /** 
	   * Handle exception from a write failure.
	   */
	  public void handleExceptionWrite(Exception e, List w)
	  {
	    final String mName = "handleExceptionWrite";
	    
	    if(logger.isLoggable(Level.FINER)) 
	      logger.logp(Level.FINE, className, mName, e.getClass().getName() + "; " + this.toString());

	    if (!isRetryLimitReached() && isRetryable(e))
	    {
	      // Retry it.  Log it.  Call the RetryListener.
	      retryType = RETRY_WRITE;
	      _retryException = e;
	      ++_retryCount;
	      logRetry(e);

	      if (_retryWriteListeners != null) {
	    	  for (RetryWriteListenerProxy retryWriteListenerProxy : _retryWriteListeners) {
	    		  retryWriteListenerProxy.onRetryWriteException(w, e);
				}
	      }
	    }
	    else
	    {
	      // No retry.  Throw it back.
	      if(logger.isLoggable(Level.FINER)) 
	        logger.logp(Level.FINE, className, mName, "No retry.  Rethrow ", e);
	      throw new BatchContainerRuntimeException(e);
	    }
	  }


	  /**
	   * Check the retryable exception lists to determine whether
	   * the given Exception is retryable.
	   */
	  private boolean isRetryable(Exception e)
	  {
	    final String mName = "isRetryable";

	    String exClassName = e.getClass().getName();

	    boolean retVal = ((_retryIncludeExceptions.isEmpty() || containsException(_retryIncludeExceptions, e)) &&
                !containsException(_retryExcludeExceptions, e));
	    
	    if(logger.isLoggable(Level.FINE)) 
	      logger.logp(Level.FINE, className, mName, mName + ": " + retVal + ": " + exClassName);

	    return retVal;
	  }
	
	  private boolean isNoRollbackException(Exception e)
	  {
		  final String mName = "isNoRollbackException";

		  String exClassName = e.getClass().getName();
		  
		  boolean retVal = false;
		  
		  if(!_retryNoRBIncludeExceptions.isEmpty())
		  {
			  retVal = (containsException(_retryNoRBIncludeExceptions, e) && !containsException(_retryNoRBExcludeExceptions, e));
		  }
			  
		  if(logger.isLoggable(Level.FINE)) 
		    logger.logp(Level.FINE, className, mName, mName + ": " + retVal + ": " + exClassName);

		  return retVal;
	  }
	  
	  /**
	   * Check whether given exception is in the specified exception list 
	   */
	  private boolean containsException(Set retryList, Exception e)
	  {
	    final String mName = "containsException";
	    boolean retVal = false;

	    for ( Iterator it = retryList.iterator(); it.hasNext(); ) {
	        String exClassName = (String) it.next();
	       
	        try {
	        	if (retVal = Thread.currentThread().getContextClassLoader().loadClass(exClassName).isInstance(e))
	        		break;
	        } catch (ClassNotFoundException cnf) {
	        	logger.logp(Level.FINE, className, mName, cnf.getLocalizedMessage());
	        }
	    }

	    if(logger.isLoggable(Level.FINE)) 
	      logger.logp(Level.FINE, className, mName, mName + ": " + retVal );

	    return retVal;
	  }

	  /**
	   * Check whether given exception is in retyable exception list 
	   */
	  private boolean containsRetryableNoRB(Set retryList, Exception e)
	  {
	    final String mName = "containsRetryableNoRB";
	    boolean retVal = false;

	    for ( Iterator it = retryList.iterator(); it.hasNext(); ) {
	        String exClassName = (String) it.next();   
	        try {
	        	if (retVal = Thread.currentThread().getContextClassLoader().loadClass(exClassName).isInstance(e))
	        		break;
	        } catch (ClassNotFoundException cnf) {
	        	logger.logp(Level.FINE, className, mName, cnf.getLocalizedMessage());
	        }
	    }
	    

	    if(logger.isLoggable(Level.FINE)) 
	      logger.logp(Level.FINE, className, mName, mName + ": " + retVal );

	    return retVal;
	  }
	  

	  /**
	   * Check if the retry limit has been reached.
	   *
	   * Note: if retry handling isn't enabled (i.e. not configured in xJCL), then this method 
	   *       will always return TRUE.
	   */
	  private boolean isRetryLimitReached()
	  {
	    return (_retryCount >= _retryLimit);
	  }

	  
	  private void logRetry(Exception e)
	  {
	    String key = "record.retried.norollback.by.batch.container";
	    Object[] details = { _jobId, _stepId, e.getClass().getName() + ": " + e.getMessage() };
	    //String message = LoggerUtil.getFormattedMessage(key, details, true);
	    //logger.info(message);	
		}

	  public Exception getException()
	  {
		  return _retryException;
	  }
	  
	  public long getRetryCount()
	  {
	    return _retryCount;
	  }

	  public void setRetryCount(long retryCount)
	  {
	    final String mName = "setRetryCount";

	    _retryCount = retryCount;

	    if(logger.isLoggable(Level.FINE)) 
	      logger.logp(Level.FINE, className, mName, "setRetryCount: " + _retryCount);
	  }

	  public String toString()
	  {
	    return "RetryHandler{" + super.toString() + "}count:limit=" + _retryCount + ":" + _retryLimit;
	  }

}