1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package org.kuali.rice.krad.datadictionary.validator;
17
18 import org.apache.commons.lang.StringUtils;
19 import org.apache.commons.logging.Log;
20 import org.apache.commons.logging.LogFactory;
21 import org.kuali.rice.krad.datadictionary.DataDictionary;
22 import org.kuali.rice.krad.datadictionary.DataDictionaryEntry;
23 import org.kuali.rice.krad.datadictionary.DataDictionaryException;
24 import org.kuali.rice.krad.uif.component.Component;
25 import org.kuali.rice.krad.uif.util.ExpressionUtils;
26 import org.kuali.rice.krad.uif.util.UifBeanFactoryPostProcessor;
27 import org.kuali.rice.krad.uif.view.View;
28 import org.springframework.beans.factory.support.KualiDefaultListableBeanFactory;
29 import org.springframework.beans.factory.xml.XmlBeanDefinitionReader;
30 import org.springframework.core.io.FileSystemResource;
31 import org.springframework.core.io.Resource;
32 import org.springframework.core.io.ResourceLoader;
33
34 import java.util.ArrayList;
35 import java.util.Map;
36
37
38
39
40
41
42 public class Validator {
43 private static final Log LOG = LogFactory.getLog(Validator.class);
44
45 private static ArrayList<ErrorReport> errorReports = new ArrayList<ErrorReport>();
46 ;
47 private ValidationTrace tracerTemp;
48 private int numberOfErrors;
49 private int numberOfWarnings;
50
51
52
53
54 public Validator() {
55 tracerTemp = new ValidationTrace();
56 numberOfErrors = 0;
57 numberOfWarnings = 0;
58 }
59
60 public static void addErrorReport(ErrorReport report) {
61 errorReports.add(report);
62 }
63
64 public static void resetErrorReport() {
65 errorReports = new ArrayList<ErrorReport>();
66 }
67
68
69
70
71
72
73
74
75 private boolean runValidations(KualiDefaultListableBeanFactory beans, boolean failOnWarning) {
76 LOG.info("Starting Dictionary Validation");
77 resetErrorReport();
78 Map<String, View> uifBeans;
79
80 try {
81 uifBeans = beans.getBeansOfType(View.class);
82 for (View views : uifBeans.values()) {
83 try {
84 ValidationTrace tracer = tracerTemp.getCopy();
85 if (doValidationOnUIFBean(views)) {
86 tracer.setValidationStage(ValidationTrace.START_UP);
87 runValidationsOnComponents(views, tracer);
88 }
89 } catch (Exception e) {
90 String value[] = {views.getId(), "Exception = " + e.getMessage()};
91 tracerTemp.createError("Error Validating Bean View", value);
92 }
93 }
94 } catch (Exception e) {
95 String value[] = {"Validation set = views", "Exception = " + e.getMessage()};
96 tracerTemp.createError("Error in Loading Spring Beans", value);
97 }
98
99 Map<String, DataDictionaryEntry> ddBeans;
100
101 try {
102 ddBeans = beans.getBeansOfType(DataDictionaryEntry.class);
103 for (DataDictionaryEntry entry : ddBeans.values()) {
104 try {
105
106 ValidationTrace tracer = tracerTemp.getCopy();
107 tracer.setValidationStage(ValidationTrace.BUILD);
108 entry.completeValidation(tracer);
109
110 } catch (Exception e) {
111 String value[] = {"Validation set = Data Dictionary Entries", "Exception = " + e.getMessage()};
112 tracerTemp.createError("Error in Loading Spring Beans", value);
113 }
114 }
115 } catch (Exception e) {
116 String value[] = {"Validation set = Data Dictionary Entries", "Exception = " + e.getMessage()};
117 tracerTemp.createError("Error in Loading Spring Beans", value);
118 }
119
120 compileFinalReport();
121
122 LOG.info("Completed Dictionary Validation");
123
124 if (numberOfErrors > 0) {
125 return false;
126 }
127 if (failOnWarning) {
128 if (numberOfWarnings > 0) {
129 return false;
130 }
131 }
132
133 return true;
134 }
135
136
137
138
139
140
141
142
143 public boolean validate(Component object, boolean failOnWarning) {
144 LOG.info("Starting Dictionary Validation");
145
146 if (doValidationOnUIFBean(object)) {
147 ValidationTrace tracer = tracerTemp.getCopy();
148 resetErrorReport();
149
150 tracer.setValidationStage(ValidationTrace.BUILD);
151
152 LOG.debug("Validating Component: " + object.getId());
153 object.completeValidation(tracer.getCopy());
154
155 runValidationsOnLifecycle(object, tracer.getCopy());
156
157 runValidationsOnPrototype(object, tracer.getCopy());
158 }
159
160 compileFinalReport();
161
162 LOG.info("Completed Dictionary Validation");
163
164 if (numberOfErrors > 0) {
165 return false;
166 }
167 if (failOnWarning) {
168 if (numberOfWarnings > 0) {
169 return false;
170 }
171 }
172
173 return true;
174 }
175
176
177
178
179
180
181
182 public boolean validate(String[] xmlFiles, boolean failOnWarning) {
183 KualiDefaultListableBeanFactory beans = loadBeans(xmlFiles);
184
185 return runValidations(beans, failOnWarning);
186 }
187
188
189
190
191
192
193
194
195
196
197 public boolean validate(String xmlFiles[], ResourceLoader loader, KualiDefaultListableBeanFactory beans,
198 boolean failOnWarning) {
199 tracerTemp = new ValidationTrace(xmlFiles, loader);
200 return runValidations(beans, failOnWarning);
201 }
202
203
204
205
206
207
208
209 private void runValidationsOnComponents(Component component, ValidationTrace tracer) {
210
211 try {
212 ExpressionUtils.populatePropertyExpressionsFromGraph(component, false);
213 } catch (Exception e) {
214 String value[] = {"view = " + component.getId()};
215 tracerTemp.createError("Error Validating Bean View while loading expressions", value);
216 }
217
218 LOG.debug("Validating View: " + component.getId());
219
220 try {
221 component.completeValidation(tracer.getCopy());
222 } catch (Exception e) {
223 String value[] = {component.getId()};
224 tracerTemp.createError("Error Validating Bean View", value);
225 }
226
227 try {
228 runValidationsOnLifecycle(component, tracer.getCopy());
229 } catch (Exception e) {
230 String value[] = {component.getId(), component.getComponentsForLifecycle().size() + "",
231 "Exception " + e.getMessage()};
232 tracerTemp.createError("Error Validating Bean Lifecycle", value);
233 }
234
235 try {
236 runValidationsOnPrototype(component, tracer.getCopy());
237 } catch (Exception e) {
238 String value[] = {component.getId(), component.getComponentPrototypes().size() + "",
239 "Exceptions : " + e.getLocalizedMessage()};
240 tracerTemp.createError("Error Validating Bean Prototypes", value);
241 }
242 }
243
244
245
246
247
248
249
250 private void runValidationsOnLifecycle(Component component, ValidationTrace tracer) {
251 if (component.getComponentsForLifecycle() == null) {
252 return;
253 }
254 if (!doValidationOnUIFBean(component)) {
255 return;
256 }
257 tracer.addBean(component);
258 for (int j = 0; j < component.getComponentsForLifecycle().size(); j++) {
259 Component temp = component.getComponentsForLifecycle().get(j);
260 if (temp == null) {
261 continue;
262 }
263 if (tracer.getValidationStage() == ValidationTrace.START_UP) {
264 ExpressionUtils.populatePropertyExpressionsFromGraph(temp, false);
265 }
266 if (temp.isRender()) {
267 temp.completeValidation(tracer.getCopy());
268 runValidationsOnLifecycle(temp, tracer.getCopy());
269 }
270 }
271 }
272
273
274
275
276
277
278
279 private void runValidationsOnPrototype(Component component, ValidationTrace tracer) {
280 if (component.getComponentPrototypes() == null) {
281 return;
282 }
283 if (!doValidationOnUIFBean(component)) {
284 return;
285 }
286 tracer.addBean(component);
287 for (int j = 0; j < component.getComponentPrototypes().size(); j++) {
288 Component temp = component.getComponentPrototypes().get(j);
289 if (temp == null) {
290 continue;
291 }
292 if (tracer.getValidationStage() == ValidationTrace.START_UP) {
293 ExpressionUtils.populatePropertyExpressionsFromGraph(temp, false);
294 }
295 if (temp.isRender()) {
296 temp.completeValidation(tracer.getCopy());
297 runValidationsOnPrototype(temp, tracer.getCopy());
298 }
299 }
300 }
301
302
303
304
305
306
307
308 private boolean doValidationOnUIFBean(Component component) {
309 if (component.getId() == null) {
310 return true;
311 }
312 if (component.getId().length() < 3) {
313 return true;
314 }
315 String temp = component.getId().substring(0, 3).toLowerCase();
316 if (temp.contains("uif")) {
317 return false;
318 }
319 return true;
320 }
321
322
323
324
325
326
327
328 public static boolean validateSpringEL(String expression) {
329 if (expression == null) {
330 return true;
331 }
332 if (expression.compareTo("") == 0) {
333 return true;
334 }
335 if (expression.length() <= 3) {
336 return false;
337 }
338
339 if (!expression.substring(0, 1).contains("@") || !expression.substring(1, 2).contains("{") ||
340 !expression.substring(expression.length() - 1, expression.length()).contains("}")) {
341 return false;
342 }
343
344 expression = expression.substring(2, expression.length() - 2);
345
346 ArrayList<String> values = getExpressionValues(expression);
347
348 for (int i = 0; i < values.size(); i++) {
349 checkPropertyName(values.get(i));
350 }
351
352 return true;
353 }
354
355
356
357
358
359
360
361 private static ArrayList<String> getExpressionValues(String expression) {
362 expression = StringUtils.replace(expression, "!=", " != ");
363 expression = StringUtils.replace(expression, "==", " == ");
364 expression = StringUtils.replace(expression, ">", " > ");
365 expression = StringUtils.replace(expression, "<", " < ");
366 expression = StringUtils.replace(expression, "<=", " <= ");
367 expression = StringUtils.replace(expression, ">=", " >= ");
368
369 String stack = "";
370 ArrayList<String> controlNames = new ArrayList<String>();
371
372 boolean expectingSingleQuote = false;
373 boolean ignoreNext = false;
374 for (int i = 0; i < expression.length(); i++) {
375 char c = expression.charAt(i);
376 if (!expectingSingleQuote && !ignoreNext && (c == '(' || c == ' ' || c == ')')) {
377 ExpressionUtils.evaluateCurrentStack(stack.trim(), controlNames);
378
379 stack = "";
380 continue;
381 } else if (!ignoreNext && c == '\'') {
382 stack = stack + c;
383 expectingSingleQuote = !expectingSingleQuote;
384 } else if (c == '\\') {
385 stack = stack + c;
386 ignoreNext = !ignoreNext;
387 } else {
388 stack = stack + c;
389 ignoreNext = false;
390 }
391 }
392
393 if (StringUtils.isNotEmpty(stack)) {
394 ExpressionUtils.evaluateCurrentStack(stack.trim(), controlNames);
395 }
396
397 return controlNames;
398 }
399
400
401
402
403
404
405
406 private static boolean checkPropertyName(String name) {
407 if (!Character.isLetter(name.charAt(0))) {
408 return false;
409 }
410
411 return true;
412 }
413
414
415
416
417
418
419
420
421 public static boolean checkExpressions(Component object, String property) {
422 if (object.getPropertyExpressions().containsKey(property)) {
423 return true;
424 }
425 return false;
426 }
427
428
429
430
431 private void compileFinalReport() {
432 ArrayList<ErrorReport> reports = Validator.errorReports;
433 for (int i = 0; i < reports.size(); i++) {
434 if (reports.get(i).getErrorStatus() == ErrorReport.ERROR) {
435 numberOfErrors++;
436 } else if (reports.get(i).getErrorStatus() == ErrorReport.WARNING) {
437 numberOfWarnings++;
438 }
439 }
440 }
441
442
443
444
445
446
447
448 public KualiDefaultListableBeanFactory loadBeans(String[] xmlFiles) {
449
450 LOG.info("Starting XML File Load");
451 KualiDefaultListableBeanFactory beans = new KualiDefaultListableBeanFactory();
452 XmlBeanDefinitionReader xmlReader = new XmlBeanDefinitionReader(beans);
453
454 DataDictionary.setupProcessor(beans);
455
456 ArrayList<String> coreFiles = new ArrayList<String>();
457 ArrayList<String> testFiles = new ArrayList<String>();
458
459 for (int i = 0; i < xmlFiles.length; i++) {
460 if (xmlFiles[i].contains("classpath")) {
461 coreFiles.add(xmlFiles[i]);
462 } else {
463 testFiles.add(xmlFiles[i]);
464 }
465 }
466 String core[] = new String[coreFiles.size()];
467 coreFiles.toArray(core);
468
469 String test[] = new String[testFiles.size()];
470 testFiles.toArray(test);
471
472 try {
473 xmlReader.loadBeanDefinitions(core);
474 } catch (Exception e) {
475 LOG.error("Error loading bean definitions", e);
476 throw new DataDictionaryException("Error loading bean definitions: " + e.getLocalizedMessage());
477 }
478
479 try {
480 xmlReader.loadBeanDefinitions(getResources(test));
481 } catch (Exception e) {
482 LOG.error("Error loading bean definitions", e);
483 throw new DataDictionaryException("Error loading bean definitions: " + e.getLocalizedMessage());
484 }
485
486 UifBeanFactoryPostProcessor factoryPostProcessor = new UifBeanFactoryPostProcessor();
487 factoryPostProcessor.postProcessBeanFactory(beans);
488
489 tracerTemp = new ValidationTrace(xmlFiles, xmlReader.getResourceLoader());
490
491 LOG.info("Completed XML File Load");
492
493 return beans;
494 }
495
496
497
498
499
500
501
502 private Resource[] getResources(String files[]) {
503 Resource resources[] = new Resource[files.length];
504 for (int i = 0; i < files.length; i++) {
505 resources[0] = new FileSystemResource(files[i]);
506 }
507
508 return resources;
509 }
510
511
512
513
514
515
516 public int getNumberOfErrors() {
517 return numberOfErrors;
518 }
519
520
521
522
523
524
525 public int getNumberOfWarnings() {
526 return numberOfWarnings;
527 }
528
529
530
531
532
533
534
535 public ErrorReport getErrorReport(int index) {
536 return errorReports.get(index);
537 }
538
539
540
541
542
543
544 public int getErrorReportSize() {
545 return errorReports.size();
546 }
547 }