1 | |
|
2 | |
|
3 | |
|
4 | |
|
5 | |
|
6 | |
|
7 | |
|
8 | |
|
9 | |
|
10 | |
|
11 | |
|
12 | |
|
13 | |
|
14 | |
|
15 | |
|
16 | |
package org.kuali.rice.ksb.messaging.exceptionhandling; |
17 | |
|
18 | |
import java.sql.Timestamp; |
19 | |
|
20 | |
import org.apache.commons.lang.StringUtils; |
21 | |
import org.apache.log4j.Logger; |
22 | |
import org.kuali.rice.core.api.config.property.ConfigContext; |
23 | |
import org.kuali.rice.ksb.api.bus.ServiceConfiguration; |
24 | |
import org.kuali.rice.ksb.messaging.PersistedMessageBO; |
25 | |
import org.kuali.rice.ksb.service.KSBServiceLocator; |
26 | |
import org.kuali.rice.ksb.util.KSBConstants; |
27 | |
|
28 | |
|
29 | |
|
30 | |
|
31 | |
|
32 | |
|
33 | |
|
34 | 0 | public class DefaultMessageExceptionHandler implements MessageExceptionHandler { |
35 | |
|
36 | 0 | private static final Logger LOG = Logger.getLogger(DefaultMessageExceptionHandler.class); |
37 | |
|
38 | |
private static final long DEFAULT_TIME_INCREMENT = 60 * 60 * 1000; |
39 | |
|
40 | |
private static final int DEFAULT_MAX_RETRIES = 7; |
41 | |
|
42 | |
public void handleException(Throwable throwable, PersistedMessageBO message, Object service) throws Exception { |
43 | 0 | if (isInException(message)) { |
44 | 0 | placeInException(throwable, message); |
45 | |
} else { |
46 | 0 | requeue(throwable, message); |
47 | |
} |
48 | 0 | } |
49 | |
|
50 | |
public void handleExceptionLastDitchEffort(Throwable throwable, PersistedMessageBO message, Object service) throws Exception { |
51 | 0 | LOG.error("Complete failure when attempting to put message into exception routing! Message was: " + message, throwable); |
52 | 0 | } |
53 | |
|
54 | |
public boolean isInException(PersistedMessageBO message) { |
55 | 0 | ServiceConfiguration serviceConfiguration = message.getMethodCall().getServiceConfiguration(); |
56 | |
|
57 | 0 | if (getImmediateExceptionRouting()) { |
58 | 0 | return true; |
59 | |
} |
60 | |
|
61 | 0 | Integer globalMaxRetryAttempts = getGlobalMaxRetryAttempts(); |
62 | 0 | if (globalMaxRetryAttempts != null) { |
63 | 0 | LOG.info("Global Max Retry has been set, so is overriding other max retry attempts."); |
64 | 0 | LOG.info("Global Max Retry count = " + globalMaxRetryAttempts + "."); |
65 | 0 | return (message.getRetryCount().intValue() >= globalMaxRetryAttempts.intValue()); |
66 | |
} |
67 | |
|
68 | 0 | if (serviceConfiguration.getRetryAttempts() > 0) { |
69 | 0 | LOG.info("Message set for retry exception handling. Message retry count = " + message.getRetryCount()); |
70 | 0 | if (message.getRetryCount() >= serviceConfiguration.getRetryAttempts()) { |
71 | 0 | return true; |
72 | |
} |
73 | 0 | } else if (serviceConfiguration.getMillisToLive() > 0) { |
74 | 0 | LOG.info("Message set for time to live exception handling. Message expiration date = " + message.getExpirationDate().getTime()); |
75 | 0 | if (System.currentTimeMillis() > message.getExpirationDate().getTime()) { |
76 | 0 | return true; |
77 | |
} |
78 | 0 | } else if (message.getRetryCount() >= this.getMaxRetryAttempts()) { |
79 | 0 | LOG.info("Message set for default exception handling. Comparing retry count = " + message.getRetryCount() + " against default max count."); |
80 | 0 | return true; |
81 | |
} |
82 | 0 | return false; |
83 | |
} |
84 | |
|
85 | |
protected void requeue(Throwable throwable, PersistedMessageBO message) throws Exception { |
86 | 0 | Integer retryCount = message.getRetryCount(); |
87 | 0 | message.setQueueStatus(KSBConstants.ROUTE_QUEUE_QUEUED); |
88 | 0 | long addMilliseconds = Math.round(getTimeIncrement() * Math.pow(2, retryCount)); |
89 | 0 | Timestamp currentTime = message.getQueueDate(); |
90 | 0 | Timestamp newTime = new Timestamp(currentTime.getTime() + addMilliseconds); |
91 | 0 | message.setQueueStatus(KSBConstants.ROUTE_QUEUE_QUEUED); |
92 | 0 | message.setRetryCount(new Integer(retryCount + 1)); |
93 | 0 | message.setQueueDate(newTime); |
94 | 0 | scheduleExecution(throwable, message); |
95 | 0 | } |
96 | |
|
97 | |
protected void placeInException(Throwable throwable, PersistedMessageBO message) throws Exception { |
98 | 0 | message.setQueueStatus(KSBConstants.ROUTE_QUEUE_EXCEPTION); |
99 | 0 | message.setQueueDate(new Timestamp(System.currentTimeMillis())); |
100 | 0 | KSBServiceLocator.getMessageQueueService().save(message); |
101 | 0 | } |
102 | |
|
103 | |
protected void scheduleExecution(Throwable throwable, PersistedMessageBO message) throws Exception { |
104 | 0 | KSBServiceLocator.getExceptionRoutingService().scheduleExecution(throwable, message, null); |
105 | 0 | } |
106 | |
|
107 | |
public Integer getMaxRetryAttempts() { |
108 | |
try { |
109 | 0 | return new Integer(ConfigContext.getCurrentContextConfig().getProperty(KSBConstants.Config.ROUTE_QUEUE_MAX_RETRY_ATTEMPTS_KEY)); |
110 | 0 | } catch (NumberFormatException e) { |
111 | 0 | LOG.error("Constant '" + KSBConstants.Config.ROUTE_QUEUE_MAX_RETRY_ATTEMPTS_KEY + "' is not a number and is being " + "used as a default for exception messages. " + DEFAULT_MAX_RETRIES + " will be used as a retry limit until this number is fixed"); |
112 | 0 | return DEFAULT_MAX_RETRIES; |
113 | |
} |
114 | |
} |
115 | |
|
116 | |
public Integer getGlobalMaxRetryAttempts() { |
117 | 0 | String globalMax = ConfigContext.getCurrentContextConfig().getProperty(KSBConstants.Config.ROUTE_QUEUE_MAX_RETRY_ATTEMPTS_OVERRIDE_KEY); |
118 | 0 | if (StringUtils.isBlank(globalMax)) { |
119 | 0 | return null; |
120 | |
} |
121 | |
try { |
122 | 0 | Integer globalMaxRetries = new Integer(globalMax); |
123 | 0 | if (globalMaxRetries >= 0) { |
124 | 0 | return globalMaxRetries; |
125 | |
} |
126 | 0 | } catch (NumberFormatException e) { |
127 | 0 | LOG.error("Constant '" + KSBConstants.Config.ROUTE_QUEUE_MAX_RETRY_ATTEMPTS_OVERRIDE_KEY + "' is not a number and is being " + "used as a default for exception messages. " + DEFAULT_MAX_RETRIES + " will be used as a retry limit until this number is fixed"); |
128 | 0 | } |
129 | 0 | return null; |
130 | |
} |
131 | |
|
132 | |
public Long getTimeIncrement() { |
133 | |
try { |
134 | 0 | return new Long(ConfigContext.getCurrentContextConfig().getProperty(KSBConstants.Config.ROUTE_QUEUE_TIME_INCREMENT_KEY)); |
135 | 0 | } catch (NumberFormatException e) { |
136 | 0 | LOG.error("Constant '" + KSBConstants.Config.ROUTE_QUEUE_TIME_INCREMENT_KEY + "' is not a number and will not be used " + "as the default time increment for exception routing. Default of " + DEFAULT_TIME_INCREMENT + " will be used."); |
137 | 0 | return DEFAULT_TIME_INCREMENT; |
138 | |
} |
139 | |
} |
140 | |
|
141 | |
public Boolean getImmediateExceptionRouting() { |
142 | 0 | return new Boolean(ConfigContext.getCurrentContextConfig().getProperty(KSBConstants.Config.IMMEDIATE_EXCEPTION_ROUTING)); |
143 | |
} |
144 | |
} |