1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package org.kuali.rice.krad.datadictionary.uif;
17
18 import java.util.ArrayList;
19 import java.util.HashMap;
20 import java.util.List;
21 import java.util.Map;
22 import java.util.Map.Entry;
23 import java.util.concurrent.ExecutorService;
24 import java.util.concurrent.Executors;
25
26 import org.apache.commons.lang.StringUtils;
27 import org.apache.commons.logging.Log;
28 import org.apache.commons.logging.LogFactory;
29 import org.kuali.rice.core.api.config.property.ConfigContext;
30 import org.kuali.rice.krad.datadictionary.DataDictionaryException;
31 import org.kuali.rice.krad.datadictionary.DefaultListableBeanFactory;
32 import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
33 import org.kuali.rice.krad.uif.UifConstants;
34 import org.kuali.rice.krad.uif.UifConstants.ViewType;
35 import org.kuali.rice.krad.uif.lifecycle.ViewLifecycle;
36 import org.kuali.rice.krad.uif.service.ViewTypeService;
37 import org.kuali.rice.krad.uif.util.CopyUtils;
38 import org.kuali.rice.krad.uif.util.ViewModelUtils;
39 import org.kuali.rice.krad.uif.view.View;
40 import org.kuali.rice.krad.util.KRADConstants;
41 import org.springframework.beans.PropertyValues;
42 import org.springframework.beans.factory.config.BeanDefinition;
43
44
45
46
47
48
49
50
51
52
53
54
55 public class UifDictionaryIndex implements Runnable {
56 private static final Log LOG = LogFactory.getLog(UifDictionaryIndex.class);
57
58 private static final int VIEW_CACHE_SIZE = 1000;
59
60 private DefaultListableBeanFactory ddBeans;
61
62
63 private Map<String, String> viewBeanEntriesById = new HashMap<String, String>();
64
65
66 private Map<String, ViewTypeDictionaryIndex> viewEntriesByType = new HashMap<String, ViewTypeDictionaryIndex>();
67
68
69 private Map<String, UifViewPool> viewPools;
70
71
72 private int threadPoolSize = 4;
73
74 public UifDictionaryIndex(DefaultListableBeanFactory ddBeans) {
75 this.ddBeans = ddBeans;
76 }
77
78 @Override
79 public void run() {
80 try {
81 Integer size = new Integer(ConfigContext.getCurrentContextConfig().getProperty(
82 KRADConstants.KRAD_DICTIONARY_INDEX_POOL_SIZE));
83 threadPoolSize = size.intValue();
84 } catch (NumberFormatException nfe) {
85
86 }
87
88 buildViewIndicies();
89 }
90
91
92
93
94
95
96
97
98
99
100
101 public View getViewById(final String viewId) {
102
103 if (viewPools.containsKey(viewId)) {
104 final UifViewPool viewPool = viewPools.get(viewId);
105 synchronized (viewPool) {
106 if (!viewPool.isEmpty()) {
107 View view = viewPool.getViewInstance();
108
109
110 Runnable createView = new Runnable() {
111 public void run() {
112 View newViewInstance = CopyUtils.copy(getImmutableViewById(viewId));
113 viewPool.addViewInstance(newViewInstance);
114 }
115 };
116
117 Thread t = new Thread(createView);
118 t.start();
119
120 return view;
121 } else {
122 LOG.info("Pool size for view with id: " + viewId
123 + " is empty. Considering increasing max pool size.");
124 }
125 }
126 }
127
128 View view = getImmutableViewById(viewId);
129
130 return CopyUtils.copy(view);
131 }
132
133
134
135
136
137
138
139 public View getImmutableViewById(String viewId) {
140 String beanName = viewBeanEntriesById.get(viewId);
141 if (StringUtils.isBlank(beanName)) {
142 throw new DataDictionaryException("Unable to find View with id: " + viewId);
143 }
144
145 View view = ddBeans.getBean(beanName, View.class);
146
147 if (UifConstants.ViewStatus.CREATED.equals(view.getViewStatus())) {
148 try {
149 ViewLifecycle.preProcess(view);
150 } catch (IllegalStateException ex) {
151 if (LOG.isDebugEnabled()) {
152 LOG.debug("preProcess not run due to an IllegalStateException. Exception message: "
153 + ex.getMessage());
154 }
155 }
156 }
157
158 return view;
159 }
160
161
162
163
164
165
166
167
168
169
170
171
172 public View getViewByTypeIndex(ViewType viewTypeName, Map<String, String> indexKey) {
173 String viewId = getViewIdByTypeIndex(viewTypeName, indexKey);
174 if (StringUtils.isNotBlank(viewId)) {
175 return getViewById(viewId);
176 }
177
178 return null;
179 }
180
181
182
183
184
185
186
187
188
189 public String getViewIdByTypeIndex(ViewType viewTypeName, Map<String, String> indexKey) {
190 String index = buildTypeIndex(indexKey);
191
192 ViewTypeDictionaryIndex typeIndex = getTypeIndex(viewTypeName);
193
194 return typeIndex.get(index);
195 }
196
197
198
199
200
201
202
203
204
205 public boolean viewByTypeExist(ViewType viewTypeName, Map<String, String> indexKey) {
206 boolean viewExist = false;
207
208 String index = buildTypeIndex(indexKey);
209 ViewTypeDictionaryIndex typeIndex = getTypeIndex(viewTypeName);
210
211 String viewId = typeIndex.get(index);
212 if (StringUtils.isNotBlank(viewId)) {
213 viewExist = true;
214 }
215
216 return viewExist;
217 }
218
219
220
221
222
223
224
225
226
227
228
229
230 public PropertyValues getViewPropertiesById(String viewId) {
231 String beanName = viewBeanEntriesById.get(viewId);
232 if (StringUtils.isNotBlank(beanName)) {
233 BeanDefinition beanDefinition = ddBeans.getMergedBeanDefinition(beanName);
234
235 return beanDefinition.getPropertyValues();
236 }
237
238 return null;
239 }
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255 public PropertyValues getViewPropertiesByType(ViewType viewTypeName, Map<String, String> indexKey) {
256 String index = buildTypeIndex(indexKey);
257
258 ViewTypeDictionaryIndex typeIndex = getTypeIndex(viewTypeName);
259
260 String beanName = typeIndex.get(index);
261 if (StringUtils.isNotBlank(beanName)) {
262 BeanDefinition beanDefinition = ddBeans.getMergedBeanDefinition(beanName);
263
264 return beanDefinition.getPropertyValues();
265 }
266
267 return null;
268 }
269
270
271
272
273
274
275
276
277
278 public List<View> getViewsForType(ViewType viewTypeName) {
279 List<View> typeViews = new ArrayList<View>();
280
281
282 if (viewEntriesByType.containsKey(viewTypeName.name())) {
283 ViewTypeDictionaryIndex typeIndex = viewEntriesByType.get(viewTypeName.name());
284 for (Entry<String, String> typeEntry : typeIndex.getViewIndex().entrySet()) {
285 View typeView = ddBeans.getBean(typeEntry.getValue(), View.class);
286 typeViews.add(typeView);
287 }
288 } else {
289 throw new DataDictionaryException("Unable to find view index for type: " + viewTypeName);
290 }
291
292 return typeViews;
293 }
294
295
296
297
298
299
300 protected void buildViewIndicies() {
301 LOG.info("Starting View Index Building");
302
303 viewBeanEntriesById = new HashMap<String, String>();
304 viewEntriesByType = new HashMap<String, ViewTypeDictionaryIndex>();
305 viewPools = new HashMap<String, UifViewPool>();
306
307 boolean inDevMode = Boolean.parseBoolean(ConfigContext.getCurrentContextConfig().getProperty(
308 KRADConstants.ConfigParameters.KRAD_DEV_MODE));
309
310 ExecutorService executor = Executors.newFixedThreadPool(threadPoolSize);
311
312 String[] beanNames = ddBeans.getBeanNamesForType(View.class);
313 for (final String beanName : beanNames) {
314 BeanDefinition beanDefinition = ddBeans.getMergedBeanDefinition(beanName);
315 PropertyValues propertyValues = beanDefinition.getPropertyValues();
316
317 String id = ViewModelUtils.getStringValFromPVs(propertyValues, "id");
318 if (StringUtils.isBlank(id)) {
319 id = beanName;
320 }
321
322 if (viewBeanEntriesById.containsKey(id)) {
323 throw new DataDictionaryException("Two views must not share the same id. Found duplicate id: " + id);
324 }
325
326 viewBeanEntriesById.put(id, beanName);
327
328 indexViewForType(propertyValues, id);
329
330
331 if (!inDevMode) {
332 String poolSizeStr = ViewModelUtils.getStringValFromPVs(propertyValues, "preloadPoolSize");
333 if (StringUtils.isNotBlank(poolSizeStr)) {
334 int poolSize = Integer.parseInt(poolSizeStr);
335 if (poolSize < 1) {
336 continue;
337 }
338
339 final View view = (View) ddBeans.getBean(beanName);
340 final UifViewPool viewPool = new UifViewPool();
341 viewPool.setMaxSize(poolSize);
342 for (int j = 0; j < poolSize; j++) {
343 Runnable createView = new Runnable() {
344 @Override
345 public void run() {
346 viewPool.addViewInstance((View) CopyUtils.copy(view));
347 }
348 };
349
350 executor.execute(createView);
351 }
352 viewPools.put(id, viewPool);
353 }
354 }
355 }
356
357 executor.shutdown();
358
359 LOG.info("Completed View Index Building");
360 }
361
362
363
364
365
366
367
368
369
370
371 protected void indexViewForType(PropertyValues propertyValues, String id) {
372 String viewTypeName = ViewModelUtils.getStringValFromPVs(propertyValues, "viewTypeName");
373 if (StringUtils.isBlank(viewTypeName)) {
374 return;
375 }
376
377 UifConstants.ViewType viewType = ViewType.valueOf(viewTypeName);
378
379 ViewTypeService typeService = KRADServiceLocatorWeb.getViewService().getViewTypeService(viewType);
380 if (typeService == null) {
381
382 return;
383 }
384
385
386 Map<String, String> typeParameters = typeService.getParametersFromViewConfiguration(propertyValues);
387
388
389 String index = buildTypeIndex(typeParameters);
390
391
392 ViewTypeDictionaryIndex typeIndex = getTypeIndex(viewType);
393
394 typeIndex.put(index, id);
395 }
396
397
398
399
400
401
402
403
404
405 protected ViewTypeDictionaryIndex getTypeIndex(UifConstants.ViewType viewType) {
406 ViewTypeDictionaryIndex typeIndex = null;
407
408 if (viewEntriesByType.containsKey(viewType.name())) {
409 typeIndex = viewEntriesByType.get(viewType.name());
410 } else {
411 typeIndex = new ViewTypeDictionaryIndex();
412 viewEntriesByType.put(viewType.name(), typeIndex);
413 }
414
415 return typeIndex;
416 }
417
418
419
420
421
422
423
424 protected String buildTypeIndex(Map<String, String> typeParameters) {
425 String index = "";
426
427 for (String parameterName : typeParameters.keySet()) {
428 if (StringUtils.isNotBlank(index)) {
429 index += "|||";
430 }
431 index += parameterName + "^^" + typeParameters.get(parameterName);
432 }
433
434 return index;
435 }
436
437 }