1 | /* |
2 | * Copyright 2006-2007 the original author or authors. |
3 | * |
4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
5 | * you may not use this file except in compliance with the License. |
6 | * You may obtain a copy of the License at |
7 | * |
8 | * http://www.apache.org/licenses/LICENSE-2.0 |
9 | * |
10 | * Unless required by applicable law or agreed to in writing, software |
11 | * distributed under the License is distributed on an "AS IS" BASIS, |
12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
13 | * See the License for the specific language governing permissions and |
14 | * limitations under the License. |
15 | */ |
16 | |
17 | package org.springframework.batch.repeat.context; |
18 | |
19 | import java.util.ArrayList; |
20 | import java.util.HashMap; |
21 | import java.util.HashSet; |
22 | import java.util.List; |
23 | import java.util.Map; |
24 | import java.util.Set; |
25 | |
26 | import org.springframework.batch.repeat.RepeatContext; |
27 | |
28 | public class RepeatContextSupport extends SynchronizedAttributeAccessor implements RepeatContext { |
29 | |
30 | private RepeatContext parent; |
31 | |
32 | private int count; |
33 | |
34 | private volatile boolean completeOnly; |
35 | |
36 | private volatile boolean terminateOnly; |
37 | |
38 | private Map<String, Set<Runnable>> callbacks = new HashMap<String, Set<Runnable>>(); |
39 | |
40 | /** |
41 | * Constructor for {@link RepeatContextSupport}. The parent can be null, but |
42 | * should be set to the enclosing repeat context if there is one, e.g. if |
43 | * this context is an inner loop. |
44 | * @param parent |
45 | */ |
46 | public RepeatContextSupport(RepeatContext parent) { |
47 | super(); |
48 | this.parent = parent; |
49 | } |
50 | |
51 | /* |
52 | * (non-Javadoc) |
53 | * |
54 | * @see org.springframework.batch.repeat.RepeatContext#isCompleteOnly() |
55 | */ |
56 | public boolean isCompleteOnly() { |
57 | return completeOnly; |
58 | } |
59 | |
60 | /* |
61 | * (non-Javadoc) |
62 | * |
63 | * @see org.springframework.batch.repeat.RepeatContext#setCompleteOnly() |
64 | */ |
65 | public void setCompleteOnly() { |
66 | completeOnly = true; |
67 | } |
68 | |
69 | /* |
70 | * (non-Javadoc) |
71 | * |
72 | * @see org.springframework.batch.repeat.RepeatContext#isTerminateOnly() |
73 | */ |
74 | public boolean isTerminateOnly() { |
75 | return terminateOnly; |
76 | } |
77 | |
78 | /* |
79 | * (non-Javadoc) |
80 | * |
81 | * @see org.springframework.batch.repeat.RepeatContext#setTerminateOnly() |
82 | */ |
83 | public void setTerminateOnly() { |
84 | terminateOnly = true; |
85 | setCompleteOnly(); |
86 | } |
87 | |
88 | /* |
89 | * (non-Javadoc) |
90 | * |
91 | * @see org.springframework.batch.repeat.RepeatContext#getParent() |
92 | */ |
93 | public RepeatContext getParent() { |
94 | return parent; |
95 | } |
96 | |
97 | /** |
98 | * Used by clients to increment the started count. |
99 | */ |
100 | public synchronized void increment() { |
101 | count++; |
102 | } |
103 | |
104 | /* |
105 | * (non-Javadoc) |
106 | * |
107 | * @see org.springframework.batch.repeat.RepeatContext#getStartedCount() |
108 | */ |
109 | public synchronized int getStartedCount() { |
110 | return count; |
111 | } |
112 | |
113 | /* |
114 | * (non-Javadoc) |
115 | * |
116 | * @see |
117 | * org.springframework.batch.repeat.RepeatContext#registerDestructionCallback |
118 | * (java.lang.String, java.lang.Runnable) |
119 | */ |
120 | public void registerDestructionCallback(String name, Runnable callback) { |
121 | synchronized (callbacks) { |
122 | Set<Runnable> set = callbacks.get(name); |
123 | if (set == null) { |
124 | set = new HashSet<Runnable>(); |
125 | callbacks.put(name, set); |
126 | } |
127 | set.add(callback); |
128 | } |
129 | } |
130 | |
131 | /* |
132 | * (non-Javadoc) |
133 | * |
134 | * @see org.springframework.batch.repeat.RepeatContext#close() |
135 | */ |
136 | public void close() { |
137 | |
138 | List<RuntimeException> errors = new ArrayList<RuntimeException>(); |
139 | |
140 | Set<Map.Entry<String, Set<Runnable>>> copy; |
141 | |
142 | synchronized (callbacks) { |
143 | copy = new HashSet<Map.Entry<String, Set<Runnable>>>(callbacks.entrySet()); |
144 | } |
145 | |
146 | for (Map.Entry<String, Set<Runnable>> entry : copy) { |
147 | |
148 | for (Runnable callback : entry.getValue()) { |
149 | /* |
150 | * Potentially we could check here if there is an attribute with |
151 | * the given name - if it has been removed, maybe the callback |
152 | * is invalid. On the other hand it is less surprising for the |
153 | * callback register if it is always executed. |
154 | */ |
155 | if (callback != null) { |
156 | /* |
157 | * The documentation of the interface says that these |
158 | * callbacks must not throw exceptions, but we don't trust |
159 | * them necessarily... |
160 | */ |
161 | try { |
162 | callback.run(); |
163 | } |
164 | catch (RuntimeException t) { |
165 | errors.add(t); |
166 | } |
167 | } |
168 | } |
169 | } |
170 | |
171 | if (errors.isEmpty()) { |
172 | return; |
173 | } |
174 | |
175 | throw (RuntimeException) errors.get(0); |
176 | } |
177 | |
178 | } |