001 /** 002 * Copyright 2004-2013 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.kuali.hr.time.batch; 017 018 import org.apache.commons.lang.StringUtils; 019 import org.apache.log4j.Logger; 020 import org.kuali.hr.time.service.base.TkServiceLocator; 021 import org.kuali.hr.time.util.TkConstants; 022 import org.springframework.transaction.TransactionStatus; 023 import org.springframework.transaction.support.TransactionCallbackWithoutResult; 024 025 public abstract class BatchJobEntryRunnable implements Runnable{ 026 027 private Logger LOG = Logger.getLogger(BatchJobEntryRunnable.class); 028 029 long startTime; 030 long endTime; 031 Long tkBatchJobEntryId; 032 Long tkBatchJobId; 033 BatchJobEntry batchJobEntry; 034 035 public BatchJobEntryRunnable(BatchJobEntry entry) { 036 this.batchJobEntry = entry; 037 this.tkBatchJobEntryId = entry.getTkBatchJobEntryId(); 038 this.tkBatchJobId = entry.getTkBatchJobId(); 039 } 040 041 /** 042 * Method that is called before the user function doWork() is called. Any 043 * pre-work maintenance can be done in this method. 044 */ 045 void doBeforeRun() { 046 startTime = System.currentTimeMillis(); 047 LOG.debug("Before run."); 048 batchJobEntry.setBatchJobEntryStatus(TkConstants.BATCH_JOB_ENTRY_STATUS.RUNNING); 049 TkServiceLocator.getBatchJobEntryService().saveBatchJobEntry(batchJobEntry); 050 } 051 052 @Override 053 public final void run() { 054 TkServiceLocator.getTransactionTemplate().execute(new TransactionCallbackWithoutResult() { 055 @Override 056 protected void doInTransactionWithoutResult(TransactionStatus status) { 057 try { 058 doBeforeRun(); 059 doWork(); 060 doAfterRun(); 061 } catch (Throwable t) { 062 LOG.warn("BatchJobEntry: " + batchJobEntry.getTkBatchJobEntryId() + " in Exception status."); 063 batchJobEntry.setBatchJobException(t.getStackTrace().toString()); 064 } 065 }; 066 }); 067 } 068 069 /** 070 * Implement this method in your subclass. Place business logic here to handle 071 * whatever needs to be done for this unit of work. 072 * 073 * @throws Exception when problem encountered during work processing, exception 074 * stored in BatchJobEntry, and transaction rolled back. 075 */ 076 public abstract void doWork() throws Exception; 077 078 /** 079 * Method that is called after the user function doWork() is called. Any 080 * post-work maintenance can be done in this method. 081 */ 082 void doAfterRun() { 083 endTime = System.currentTimeMillis(); 084 long runtime = endTime - startTime; 085 runtime = (runtime > 0) ? runtime : 1; // hack around 0 length job... just in case. 086 LOG.debug("Job finished in " + runtime / 1000 + " seconds."); 087 088 if (StringUtils.isEmpty(batchJobEntry.getBatchJobException())) { 089 batchJobEntry.setBatchJobEntryStatus(TkConstants.BATCH_JOB_ENTRY_STATUS.FINISHED); 090 } else { 091 batchJobEntry.setBatchJobEntryStatus(TkConstants.BATCH_JOB_ENTRY_STATUS.EXCEPTION); 092 } 093 TkServiceLocator.getBatchJobEntryService().saveBatchJobEntry(batchJobEntry); 094 } 095 096 public Long getTkBatchJobEntryId() { 097 return tkBatchJobEntryId; 098 } 099 100 public Long getTkBatchJobId() { 101 return tkBatchJobId; 102 } 103 104 public BatchJobEntry getBatchJobEntry() { 105 return batchJobEntry; 106 } 107 }