001/** 002 * Copyright 2005-2015 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 */ 016package org.kuali.rice.krad.uif.util; 017 018import static org.junit.Assert.assertEquals; 019import static org.junit.Assert.assertFalse; 020import static org.junit.Assert.assertNotNull; 021import static org.junit.Assert.assertNull; 022import static org.junit.Assert.assertSame; 023import static org.junit.Assert.assertTrue; 024import static org.junit.Assert.fail; 025 026import java.beans.BeanInfo; 027import java.beans.IntrospectionException; 028import java.beans.Introspector; 029import java.beans.PropertyDescriptor; 030import java.beans.PropertyEditorSupport; 031import java.io.Serializable; 032import java.lang.annotation.Retention; 033import java.lang.annotation.RetentionPolicy; 034import java.lang.reflect.Method; 035import java.math.BigDecimal; 036import java.sql.Timestamp; 037import java.text.SimpleDateFormat; 038import java.util.Arrays; 039import java.util.Date; 040import java.util.List; 041import java.util.Map; 042 043import org.junit.AfterClass; 044import org.junit.BeforeClass; 045import org.junit.Test; 046import org.kuali.rice.core.api.CoreConstants; 047import org.kuali.rice.core.api.config.property.ConfigContext; 048import org.kuali.rice.core.api.resourceloader.GlobalResourceLoader; 049import org.kuali.rice.core.api.util.type.KualiDecimal; 050import org.kuali.rice.core.api.util.type.KualiPercent; 051import org.kuali.rice.krad.service.KRADServiceLocatorWeb; 052import org.kuali.rice.krad.uif.UifConstants; 053import org.kuali.rice.krad.uif.UifParameters; 054import org.kuali.rice.krad.uif.component.BindingInfo; 055import org.kuali.rice.krad.uif.container.CollectionGroup; 056import org.kuali.rice.krad.uif.container.CollectionGroupBase; 057import org.kuali.rice.krad.uif.container.CollectionGroupBuilder; 058import org.kuali.rice.krad.uif.container.Group; 059import org.kuali.rice.krad.uif.container.GroupBase; 060import org.kuali.rice.krad.uif.element.Action; 061import org.kuali.rice.krad.uif.element.Message; 062import org.kuali.rice.krad.uif.element.ViewHeader; 063import org.kuali.rice.krad.uif.layout.StackedLayoutManager; 064import org.kuali.rice.krad.uif.layout.StackedLayoutManagerBase; 065import org.kuali.rice.krad.uif.lifecycle.ViewLifecycle; 066import org.kuali.rice.krad.uif.service.impl.ViewHelperServiceImpl; 067import org.kuali.rice.krad.uif.view.FormView; 068import org.kuali.rice.krad.uif.view.ViewPresentationControllerBase; 069import org.kuali.rice.krad.util.GlobalVariables; 070import org.kuali.rice.krad.util.KRADConstants; 071import org.kuali.rice.krad.web.bind.RequestAccessible; 072import org.kuali.rice.krad.web.bind.UifConfigurableWebBindingInitializer; 073import org.kuali.rice.krad.web.bind.UifServletRequestDataBinder; 074import org.kuali.rice.krad.web.form.UifFormBase; 075import org.slf4j.Logger; 076import org.slf4j.LoggerFactory; 077import org.springframework.mock.web.MockHttpServletRequest; 078import org.springframework.web.context.request.RequestContextHolder; 079import org.springframework.web.context.request.ServletRequestAttributes; 080import org.springframework.web.context.request.ServletWebRequest; 081import org.springframework.web.servlet.ModelAndView; 082 083public class ObjectPropertyUtilsTest extends ProcessLoggingUnitTest { 084 085 final Logger LOG = LoggerFactory.getLogger(ObjectPropertyUtilsTest.class); 086 087 @Retention(RetentionPolicy.RUNTIME) 088 public @interface TestAnnotation { 089 String afoo(); 090 } 091 092 @BeforeClass 093 public static void setup() throws Exception { 094 UifUnitTestUtils.establishMockConfig("ObjectPropertyUtilsTest"); 095 } 096 097 @AfterClass 098 public static void teardown() throws Exception { 099 UifUnitTestUtils.tearDownMockConfig(); 100 } 101 102 public static class TestBean implements Serializable { 103 104 private static final long serialVersionUID = 1L; 105 106 public TestBean() {} 107 108 @RequestAccessible 109 private String rwProp; 110 111 private TestBeanTwo complexProp; 112 113 public String getRwProp() { 114 return this.rwProp; 115 } 116 117 public void setRwProp(String r) { 118 this.rwProp = r; 119 } 120 121 private String woProp; 122 123 public void setWoProp(String w) { 124 this.woProp = w; 125 } 126 127 private String roProp; 128 129 @TestAnnotation(afoo = "abar") 130 public String getRoProp() { 131 return this.roProp; 132 } 133 134 private Boolean bitProp; 135 136 public boolean isBitProp() { 137 return bitProp != null && bitProp; 138 } 139 140 public Boolean getBitProp() { 141 return bitProp; 142 } 143 144 public void setBitProp(Boolean bitProp) { 145 this.bitProp = bitProp; 146 } 147 148 private boolean booleanProp; 149 150 public boolean isBooleanProp() { 151 return booleanProp; 152 } 153 154 public void setBooleanProp(boolean booleanProp) { 155 this.booleanProp = booleanProp; 156 } 157 158 private Timestamp timestampProp; 159 160 public Timestamp getTimestampProp() { 161 return timestampProp; 162 } 163 164 public void setTimestampProp(Timestamp timestampProp) { 165 this.timestampProp = timestampProp; 166 } 167 168 private Date dateProp; 169 170 public Date getDateProp() { 171 return dateProp; 172 } 173 174 public void setDateProp(Date dateProp) { 175 this.dateProp = dateProp; 176 } 177 178 private int intProp; 179 180 public int getIntProp() { 181 return intProp; 182 } 183 184 private BigDecimal bigDecimalProp; 185 186 public BigDecimal getBigDecimalProp() { 187 return bigDecimalProp; 188 } 189 190 public void setBigDecimalProp(BigDecimal bigDecimalProp) { 191 this.bigDecimalProp = bigDecimalProp; 192 } 193 194 public void setIntProp(int intProp) { 195 this.intProp = intProp; 196 } 197 198 private Integer integerProp; 199 200 public Integer getIntegerProp() { 201 return integerProp; 202 } 203 204 public void setIntegerProp(Integer integerProp) { 205 this.integerProp = integerProp; 206 } 207 208 private TestBean next; 209 210 public TestBean getNext() { 211 return next; 212 } 213 214 public void setNext(TestBean next) { 215 this.next = next; 216 } 217 218 private List<String> stuffs; 219 220 public List<String> getStuffs() { 221 return stuffs; 222 } 223 224 public void setStuffs(List<String> stuffs) { 225 this.stuffs = stuffs; 226 } 227 228 private Object[] arrayProp; 229 230 public Object[] getArrayProp() { 231 return arrayProp; 232 } 233 234 public void setArrayProp(Object[] arrayProp) { 235 this.arrayProp = arrayProp; 236 } 237 238 private Map<String, Object> mapProp; 239 240 public Map<String, Object> getMapProp() { 241 return this.mapProp; 242 } 243 244 public void setMapProp(Map<String, Object> mapProp) { 245 this.mapProp = mapProp; 246 } 247 248 /** 249 * @return the complexProp 250 */ 251 public TestBeanTwo getComplexProp() { 252 return this.complexProp; 253 } 254 255 /** 256 * @param complexProp the complexProp to set 257 */ 258 public void setComplexProp(TestBeanTwo complexProp) { 259 this.complexProp = complexProp; 260 } 261 } 262 263 public static class TestBeanTwo { 264 265 private String fooProp; 266 267 /** 268 * @return the fooProp 269 */ 270 public String getFooProp() { 271 return this.fooProp; 272 } 273 274 /** 275 * @param fooProp the fooProp to set 276 */ 277 public void setFooProp(String fooProp) { 278 this.fooProp = fooProp; 279 } 280 } 281 282 @Test 283 public void testSetBoolean() { 284 TestBean tb = new TestBean(); 285 ObjectPropertyUtils.setPropertyValue(tb, "booleanProp", "true"); 286 assertTrue(tb.isBooleanProp()); 287 } 288 289 @Test 290 public void testGetPropertyDescriptor() { 291 Map<String, PropertyDescriptor> pds = ObjectPropertyUtils.getPropertyDescriptors(TestBean.class); 292 assertNotNull(pds.get("rwProp")); 293 assertNotNull(pds.get("roProp")); 294 assertNotNull(pds.get("woProp")); 295 assertNull(pds.get("foobar")); 296 } 297 298 @Test 299 public void testGet() { 300 TestBean tb = new TestBean(); 301 tb.setRwProp("foobar"); 302 assertEquals("foobar", ObjectPropertyUtils.getPropertyValue(tb, "rwProp")); 303 304 tb.roProp = "barbaz"; 305 assertEquals("barbaz", ObjectPropertyUtils.getPropertyValue(tb, "roProp")); 306 307 try { 308 ObjectPropertyUtils.getPropertyValue(tb, "woProp"); 309 // KULRICE-10677 - should return null - fail("expected exception"); 310 } catch (RuntimeException e) { 311 // KULRICE-10677 - should return null 312 throw e; 313 } 314 } 315 316 @Test 317 public void testLookup() { 318 TestBean tb = new TestBean(); 319 tb.roProp = "barbaz"; 320 assertEquals("barbaz", ObjectPropertyUtils.getPropertyValue(tb, "roProp")); 321 322 Map<String, Object> tm = new java.util.HashMap<String, Object>(); 323 tb.setMapProp(tm); 324 tm.put("barbaz", "hooray!"); 325 tm.put("bar.baz", "hoorah!"); 326 tm.put("bar.[baz]", "foobah!"); 327 tm.put("b'('r.[\"ain)\"s]", "zombie!"); 328 assertEquals("hooray!", ObjectPropertyUtils.getPropertyValue(tb, "mapProp[barbaz]")); 329 assertEquals("hooray!", ObjectPropertyUtils.getPropertyValue(tb, "mapProp['barbaz']")); 330 assertEquals("hooray!", ObjectPropertyUtils.getPropertyValue(tb, "mapProp[\"barbaz\"]")); 331 assertEquals("hoorah!", ObjectPropertyUtils.getPropertyValue(tb, "mapProp[bar.baz]")); 332 assertEquals("foobah!", ObjectPropertyUtils.getPropertyValue(tb, "mapProp[bar.[baz]]")); 333 assertEquals("zombie!", ObjectPropertyUtils.getPropertyValue(tb, "mapProp['b'('r.[\"ain)\"s]']")); 334 assertEquals("zombie!", ObjectPropertyUtils.getPropertyValue(tb, "mapProp[b'('r.[\"ain)\"s]]")); 335 336 TestBean tb2 = new TestBean(); 337 tb2.setRwProp("foodbar"); 338 tb.setNext(tb2); 339 tm.put("blah", new Object[]{"next", "rwProp"}); 340 tm.put("baz", tb2); 341 assertTrue(ObjectPropertyUtils.isReadableProperty(tb, "mapProp[\"baz\"].rwProp")); 342 assertEquals("barbaz", ObjectPropertyUtils.getPropertyValue(tb, "roProp")); 343 assertEquals("foodbar", ObjectPropertyUtils.getPropertyValue(tb, "next.rwProp")); 344 345 tb.setStuffs(Arrays.asList(new String[]{"foo", "bar", "baz",})); 346 assertEquals("bar", ObjectPropertyUtils.getPropertyValue(tb, "stuffs[1]")); 347 348 TestBean rb = new TestBean(); 349 TestBean nb = new TestBean(); 350 TestBean lb = new TestBean(); 351 rb.setNext(nb); 352 nb.setNext(lb); 353 assertEquals(String.class, ObjectPropertyUtils.getPropertyType(rb, "next.next.rwProp")); 354 rb.setRwProp("baz"); 355 nb.setRwProp("bar"); 356 lb.setRwProp("foo"); 357 assertEquals("foo", ObjectPropertyUtils.getPropertyValue(rb, "next.next.rwProp")); 358 } 359 360 @Test 361 public void testSet() throws Throwable { 362 TestBean tb = new TestBean(); 363 ObjectPropertyUtils.setPropertyValue(tb, "rwProp", "foobar"); 364 assertEquals("foobar", tb.getRwProp()); 365 366 ObjectPropertyUtils.setPropertyValue(tb, "woProp", "barbaz"); 367 assertEquals("barbaz", tb.woProp); 368 369 try { 370 ObjectPropertyUtils.setPropertyValue(tb, "roProp", "bazfoo"); 371 fail("expected exception"); 372 } catch (Exception E) { 373 // OK! 374 } 375 376 long now = System.currentTimeMillis(); 377 ObjectPropertyUtils.setPropertyValue(tb, "dateProp", new java.sql.Date(now)); 378 assertEquals(now, tb.getDateProp().getTime()); 379 380 String dateStr = "01/03/2013"; 381 ObjectPropertyUtils.setPropertyValue(tb, "dateProp", dateStr); 382 Date expectedDate = new SimpleDateFormat("MM/dd/yy").parse(dateStr); 383 assertEquals(expectedDate, tb.getDateProp()); 384 } 385 386 @Test 387 public void testGetAsText() throws Throwable { 388 String dateStr = "01/03/2013"; 389 Date expectedDate = new SimpleDateFormat("MM/dd/yy").parse(dateStr); 390 TestBean tb = new TestBean(); 391 tb.setDateProp(expectedDate); 392 assertEquals("01/03/13", ObjectPropertyUtils.getPropertyValueAsText(tb, "dateProp")); 393 } 394 395 public static class TestForm extends UifFormBase { 396 397 private static final long serialVersionUID = 6597388705374534394L; 398 private TestBean bean; 399 400 /** 401 * @return the bean 402 */ 403 public TestBean getBean() { 404 return this.bean; 405 } 406 407 /** 408 * @param bean the bean to set 409 */ 410 public void setBean(TestBean bean) { 411 this.bean = bean; 412 } 413 414 } 415 416 public static class FooEditor extends PropertyEditorSupport { 417 418 @Override 419 public String getAsText() { 420 return "foobar"; 421 } 422 423 } 424 425 // TODO: Move to integration test 426// @Test 427// public void testCustomEditor() throws Throwable { 428// TestForm form = new TestForm(); 429// MockHttpServletRequest request = new MockHttpServletRequest(); 430// request.setParameter(UifParameters.VIEW_ID, "TestViewWithPropertyEditor"); 431// request.setParameter("bean.next.rwProp", "not foobar"); 432// RequestContextHolder.setRequestAttributes(new ServletRequestAttributes(request)); 433// UifServletRequestDataBinder binder = new UifServletRequestDataBinder(form); 434// new UifConfigurableWebBindingInitializer().initBinder(binder, new ServletWebRequest(request)); 435// binder.bind(request); 436// 437// ModelAndView modelAndView = new ModelAndView(); 438// modelAndView.addObject(UifConstants.DEFAULT_MODEL_NAME, form); 439// 440// KRADServiceLocatorWeb.getModelAndViewService().prepareView(request, modelAndView); 441// assertEquals("foobar", ObjectPropertyUtils.getPropertyValueAsText(form, "bean.next.rwProp")); 442// } 443 444 @Test 445 public void testPathSet() { 446 TestBean tb = new TestBean(); 447 ObjectPropertyUtils.setPropertyValue(tb, "rwProp", "bar"); 448 assertEquals("bar", tb.getRwProp()); 449 ObjectPropertyUtils.setPropertyValue(tb, "next", new TestBean()); 450 ObjectPropertyUtils.setPropertyValue(tb, "next.next", new TestBean()); 451 ObjectPropertyUtils.setPropertyValue(tb, "next.next.woProp", "baz"); 452 assertEquals("baz", tb.getNext().getNext().woProp); 453 } 454 455 @Test 456 public void testBulk() { 457 Map<String, String> pd = new java.util.HashMap<String, String>(); 458 pd.put("rwProp", "foobar"); 459 pd.put("intProp", "3"); 460 pd.put("booleanProp", "true"); 461 pd.put("stuffs", "foo,bar,baz"); 462 for (int i = 0; i < 10000; i++) { 463 TestBean tb = new TestBean(); 464 ObjectPropertyUtils.copyPropertiesToObject(pd, tb); 465 assertEquals("foobar", tb.getRwProp()); 466 assertEquals(3, tb.getIntProp()); 467 assertEquals(true, tb.isBooleanProp()); 468 assertEquals(3, tb.getStuffs().size()); 469 assertEquals("foo", tb.getStuffs().get(0)); 470 assertEquals("bar", tb.getStuffs().get(1)); 471 assertEquals("baz", tb.getStuffs().get(2)); 472 } 473 } 474 475 @Test 476 public void testReadWriteCheck() { 477 TestBean tb = new TestBean(); 478 assertTrue(ObjectPropertyUtils.isReadableProperty(tb, "rwProp")); 479 assertTrue(ObjectPropertyUtils.isWritableProperty(tb, "rwProp")); 480 assertTrue(ObjectPropertyUtils.isReadableProperty(tb, "roProp")); 481 assertFalse(ObjectPropertyUtils.isWritableProperty(tb, "roProp")); 482 assertFalse(ObjectPropertyUtils.isReadableProperty(tb, "woProp")); 483 assertTrue(ObjectPropertyUtils.isWritableProperty(tb, "woProp")); 484 } 485 486 @Test 487 public void testKradUifTemplateHeaderMetadata() { 488 FormView formView = new FormView(); 489 ViewHeader viewHeader = new ViewHeader(); 490 formView.setHeader(viewHeader); 491 Message headerMetadataMessage = new Message(); 492 viewHeader.setMetadataMessage(headerMetadataMessage); 493 assertSame(headerMetadataMessage, ObjectPropertyUtils.getPropertyValue(formView, "header.metadataMessage")); 494 } 495 496 /** 497 * Collection list item type, for testing UIF interaction with ObjectPropertyUtils. 498 */ 499 public static class CollectionTestItem { 500 501 /** 502 * A string property, called foobar. 503 */ 504 String foobar; 505 506 /** 507 * @return the foobar 508 */ 509 public String getFoobar() { 510 return this.foobar; 511 } 512 513 /** 514 * @param foobar the foobar to set 515 */ 516 public void setFoobar(String foobar) { 517 this.foobar = foobar; 518 } 519 520 } 521 522 /** 523 * Reference to a collection, for testing UIF interaction with ObjectPropertyUtils. 524 */ 525 public static class CollectionTestListRef { 526 527 /** 528 * The collection. 529 */ 530 List<CollectionTestItem> bar; 531 532 /** 533 * Mapping of new line items. 534 */ 535 Map<String, CollectionTestItem> baz; 536 537 /** 538 * @return the bar 539 */ 540 public List<CollectionTestItem> getBar() { 541 return this.bar; 542 } 543 544 /** 545 * @param bar the bar to set 546 */ 547 public void setBar(List<CollectionTestItem> bar) { 548 this.bar = bar; 549 } 550 551 /** 552 * @return the baz 553 */ 554 public Map<String, CollectionTestItem> getBaz() { 555 return this.baz; 556 } 557 558 /** 559 * @param baz the baz to set 560 */ 561 public void setBaz(Map<String, CollectionTestItem> baz) { 562 this.baz = baz; 563 } 564 565 } 566 567 /** 568 * Mock collection form for UIF interaction with ObjectPropertyUtils. 569 */ 570 public static class CollectionTestForm extends UifFormBase { 571 572 private static final long serialVersionUID = 1798800132492441253L; 573 574 /** 575 * Reference to a data object that has a collection. 576 */ 577 CollectionTestListRef foo; 578 579 /** 580 * @return the foo 581 */ 582 public CollectionTestListRef getFoo() { 583 return this.foo; 584 } 585 586 /** 587 * @param foo the foo to set 588 */ 589 public void setFoo(CollectionTestListRef foo) { 590 this.foo = foo; 591 } 592 593 } 594 595 @Test 596 public void testKradUifCollectionGroupBuilder() throws Throwable { 597 UifUnitTestUtils.establishMockConfig(ObjectPropertyUtilsTest.class.getSimpleName()); 598 UifUnitTestUtils.establishMockUserSession("testuser"); 599 try { 600 // Performance medium generates this property path: 601 // newCollectionLines['newCollectionLines_'mediumCollection1'_.subList'] 602 603 // Below recreates the stack trace that ensued due to poorly escaped quotes, 604 // and proves that the parser works around bad quoting in a manner similar to BeanWrapper 605 606 final CollectionGroupBuilder collectionGroupBuilder = new CollectionGroupBuilder(); 607 final CollectionTestForm form = new CollectionTestForm(); 608 CollectionTestItem item = new CollectionTestItem(); 609 item.setFoobar("barfoo"); 610 ObjectPropertyUtils.setPropertyValue(form, "foo.baz['foo_bar_'badquotes'_.foobar']", item); 611 assertEquals("barfoo", form.foo.baz.get("foo_bar_'badquotes'_.foobar").foobar); 612 613 final FormView view = new FormView(); 614 view.setFormClass(CollectionTestForm.class); 615 view.setViewHelperService(new ViewHelperServiceImpl()); 616 view.setPresentationController(new ViewPresentationControllerBase()); 617 view.setAuthorizer(UifUnitTestUtils.getAllowMostViewAuthorizer()); 618 619 final CollectionGroup collectionGroup = new CollectionGroupBase(); 620 collectionGroup.setCollectionObjectClass(CollectionTestItem.class); 621 collectionGroup.setAddLinePropertyName("addLineFoo"); 622 623 StackedLayoutManager layoutManager = new StackedLayoutManagerBase(); 624 Group lineGroupPrototype = new GroupBase(); 625 layoutManager.setLineGroupPrototype(lineGroupPrototype); 626 collectionGroup.setLayoutManager(layoutManager); 627 628 BindingInfo addLineBindingInfo = new BindingInfo(); 629 addLineBindingInfo.setBindingPath("foo.baz['foo_bar_'badquotes'_.foobar']"); 630 collectionGroup.setAddLineBindingInfo(addLineBindingInfo); 631 632 BindingInfo collectionBindingInfo = new BindingInfo(); 633 collectionBindingInfo.setBindingPath("foo.bar"); 634 collectionGroup.setBindingInfo(collectionBindingInfo); 635 636 ViewLifecycle.encapsulateLifecycle(view, form, null, new Runnable() { 637 @Override 638 public void run() { 639 collectionGroupBuilder.build(view, form, (CollectionGroup) CopyUtils.copy(collectionGroup)); 640 } 641 }); 642 } finally { 643 GlobalVariables.setUserSession(null); 644 GlobalVariables.clear(); 645 GlobalResourceLoader.stop(); 646 } 647 } 648 649 @Test 650 public void testSetStringMapFromInt() { 651 Action action = new Action(); 652 ObjectPropertyUtils.setPropertyValue(action, "actionParameters['lineIndex']", 34); 653 assertEquals("34", action.getActionParameter("lineIndex")); 654 } 655 656 @Test 657 public void testClassNavigation() { 658 assertEquals(String.class, ObjectPropertyUtils.getPropertyType(TestBean.class, "complexProp.fooProp")); 659 660 try { 661 // valid first reference, invalid second reference 662 assertEquals(null, ObjectPropertyUtils.getPropertyType(TestBean.class, "complexProp.foobar")); 663 // NULL is ok - fail("KULRICE-10677 - is this ok?"); 664 } catch (IllegalArgumentException e) { 665 // IAE is not ok - KULRICE-10677 is this ok? 666 throw e; 667 } 668 669 try { 670 // invalid single reference 671 assertEquals(null, ObjectPropertyUtils.getPropertyType(TestBean.class, "foo")); 672 // NULL is ok - fail("KULRICE-10677 - is this ok?"); 673 } catch (IllegalArgumentException e) { 674 // IAE is not ok - KULRICE-10677 is this ok? 675 throw e; 676 } 677 678 try { 679 // invalid first reference 680 assertEquals(null, ObjectPropertyUtils.getPropertyType(TestBean.class, "foo.bar")); 681 // NULL is ok - fail("KULRICE-10677 - is this ok?"); 682 } catch (IllegalArgumentException e) { 683 // IAE is not ok - KULRICE-10677 is this ok? 684 throw e; 685 } 686 } 687 688 @Test 689 public void testPropertySplitPath() { 690 String path = "foo.foo1.foo2"; 691 String[] splitPaths = ObjectPropertyUtils.splitPropertyPath(path); 692 693 assertEquals(3, splitPaths.length); 694 assertEquals("foo", splitPaths[0]); 695 assertEquals("foo1", splitPaths[1]); 696 assertEquals("foo2", splitPaths[2]); 697 698 path = "foo[1]"; 699 splitPaths = ObjectPropertyUtils.splitPropertyPath(path); 700 701 assertEquals(1, splitPaths.length); 702 assertEquals("foo[1]", splitPaths[0]); 703 704 path = "foo.foo1['key.nested'].foo2"; 705 splitPaths = ObjectPropertyUtils.splitPropertyPath(path); 706 707 assertEquals(3, splitPaths.length); 708 assertEquals("foo", splitPaths[0]); 709 assertEquals("foo1['key.nested']", splitPaths[1]); 710 assertEquals("foo2", splitPaths[2]); 711 712 path = "foo.foo1['key.nested'].foo2.foo3['key.nest.nest'].foo4"; 713 splitPaths = ObjectPropertyUtils.splitPropertyPath(path); 714 715 assertEquals(5, splitPaths.length); 716 assertEquals("foo", splitPaths[0]); 717 assertEquals("foo1['key.nested']", splitPaths[1]); 718 assertEquals("foo2", splitPaths[2]); 719 assertEquals("foo3['key.nest.nest']", splitPaths[3]); 720 assertEquals("foo4", splitPaths[4]); 721 } 722 723 @Test 724 public void testCanonicalPath() { 725 String path = "foo.foo1.foo2"; 726 assertEquals(path, ObjectPropertyUtils.getCanonicalPath(path)); 727 728 path = "foo[1]"; 729 assertEquals("foo", ObjectPropertyUtils.getCanonicalPath(path)); 730 731 path = "foo.foo1['key.nested'].foo2"; 732 assertEquals("foo.foo1.foo2", ObjectPropertyUtils.getCanonicalPath(path)); 733 734 path = "foo.foo1['key.nested'].foo2.foo3['key.nest.nest'].foo4"; 735 assertEquals("foo.foo1.foo2.foo3.foo4", ObjectPropertyUtils.getCanonicalPath(path)); 736 } 737 738 // Classes used by testGetterInInterfaceOrSuperHasWiderType to check covariant return types on JDK6 739 740 // Holds an abstract class that is extended by Integer 741 public interface NumberHolder { 742 Number getNumber(); 743 } 744 745 public class NumberedImplOne implements NumberHolder { 746 @Override 747 public Integer getNumber() { 748 return 1; 749 } 750 } 751 752 public abstract class AbstractNumberHolder implements NumberHolder { 753 @Override 754 public abstract Number getNumber(); 755 } 756 757 public class ConcreteNumberHolder extends AbstractNumberHolder { 758 @Override 759 public Number getNumber() { 760 return 1; 761 } 762 } 763 764 public class ConcreteNarrowedNumberHolder extends ConcreteNumberHolder { 765 @Override 766 public Integer getNumber() { 767 return 1; 768 } 769 } 770 771 public class ConcreteNarrowedNumberHolderSub extends ConcreteNarrowedNumberHolder { 772 773 } 774 775 // Holds an interface that is implemented by Integer 776 public interface ComparableHolder { 777 Comparable<?> getComparable(); 778 } 779 780 public class ComparableHolderImpl implements ComparableHolder { 781 @Override 782 public Integer getComparable() { 783 return 1; 784 } 785 } 786 787 // Holds a concrete superclass of KualiPercent 788 public interface KualiDecimalHolder { 789 KualiDecimal getDecimal(); 790 } 791 792 public class KualiPercentHolder implements KualiDecimalHolder { 793 @Override 794 public KualiPercent getDecimal() { 795 return new KualiPercent(1d); 796 } 797 } 798 799 public class Base { 800 public Number getValue() { return null; } 801 } 802 803 public class Int extends Base { 804 public Integer getValue() { return Integer.valueOf(1); } 805 } 806 807 /** 808 * Verifies (at least when run on Linux in JDK6) our fix for the JDK6 Introspector 809 * bug/shortcoming WRT covariant return types that results in a wider getter method being 810 * preferred over a more specific implementation getter method. 811 * 812 * <p> 813 * This makes the type reported by Introspector for read methods depending on the order of 814 * Methods depend on the order that they are returned by reflection on a class, which has been 815 * demonstrated to vary between platforms. 816 * </p> 817 */ 818 @Test 819 public void testGetterInInterfaceOrSuperHasWiderType() { 820 Method readMethod = null; 821 822 readMethod = ObjectPropertyUtils.getReadMethod(ComparableHolderImpl.class, "comparable"); 823 //System.out.println(readMethod.getReturnType()); 824 assertEquals(Integer.class, readMethod.getReturnType()); 825 826 readMethod = ObjectPropertyUtils.getReadMethod(NumberedImplOne.class, "number"); 827 //System.out.println(readMethod.getReturnType()); 828 assertEquals(Integer.class, readMethod.getReturnType()); 829 830 readMethod = ObjectPropertyUtils.getReadMethod(ConcreteNarrowedNumberHolder.class, "number"); 831 //System.out.println(readMethod.getReturnType()); 832 assertEquals(Integer.class, readMethod.getReturnType()); 833 834 readMethod = ObjectPropertyUtils.getReadMethod(ConcreteNarrowedNumberHolderSub.class, "number"); 835 //System.out.println(readMethod.getReturnType()); 836 assertEquals(Integer.class, readMethod.getReturnType()); 837 838 // This case is *not* covered by our workaround, and would fail w/ JDK 6 on Linux if enabled. 839 // The interface has a concrete superclass, which will be returned in JDK6 on Linux where the 840 // Method order returned by reflection on a class is different, and the Introspector isn't smart 841 // enough to ask which Method return type is more specific. 842 for (int i = 0; i < 1; ++i) { 843 readMethod = ObjectPropertyUtils.getReadMethod(KualiPercentHolder.class, "decimal"); 844 //System.out.println(readMethod.getReturnType()); 845 //System.out.println(readMethod.getReturnType() == KualiDecimal.class); 846 } 847 848 if (readMethod.getReturnType() == KualiDecimal.class) { 849 LOG.info("I bet you're using JDK6 on Linux"); 850 } 851 852 //System.out.println("=============================================="); 853 //System.out.println("ObjectPropertyUtils.getReadMethod(Base.class, \"value\")"); 854 readMethod = ObjectPropertyUtils.getReadMethod(Base.class, "value"); 855 //System.out.println(readMethod.getReturnType()); 856 assertEquals(Number.class, readMethod.getReturnType()); 857 858 //System.out.println("=============================================="); 859 //System.out.println("ObjectPropertyUtils.getReadMethod(Int.class, \"value\")"); 860 readMethod = ObjectPropertyUtils.getReadMethod(Int.class, "value"); 861 //System.out.println(readMethod.getReturnType()); 862 assertEquals(Integer.class, readMethod.getReturnType()); 863 864 // System.out.println("=============================================="); 865 // try { 866 // // lookup bean info for given class 867 // BeanInfo info = Introspector.getBeanInfo(Int.class); 868 // if (info != null) { 869 // // get list of descriptors and iterate to validate each property 870 // PropertyDescriptor[] descriptors = info.getPropertyDescriptors(); 871 // for (int i = 0; i < descriptors.length; i++) { 872 // System.out.println("--------------------------"); 873 // PropertyDescriptor descriptor = descriptors [i]; 874 // System.out.println(descriptor.getName()); 875 // 876 // // check if read method is bridge and lookup real method 877 // readMethod = descriptor.getReadMethod(); 878 // System.out.println(readMethod.getReturnType()); 879 // System.out.println("readMethod.isBridge()=>" + readMethod.isBridge()); 880 // if (readMethod != null && readMethod.isBridge()) { 881 // Method method = Int.class.getMethod(readMethod.getName(), readMethod.getParameterTypes()); 882 // System.out.println(method.getReturnType()); 883 // 884 // // if method found, update read method 885 // // which also updates property type 886 // if (method != null) { 887 // descriptor.setReadMethod(method); 888 // 889 // // TODO: lookup write method with same type in case the write 890 // // method is the bridge method 891 // } 892 // } 893 // } 894 // } 895 // } catch (IntrospectionException e) { 896 // e.printStackTrace(); //To change body of catch statement use File | Settings | File Templates. 897 // } catch (NoSuchMethodException e) { 898 // e.printStackTrace(); //To change body of catch statement use File | Settings | File Templates. 899 // } 900 901 // Other cases to test if we have to refine this functionality: 902 // * similar to the ConcreteNarrowedNumberHolder, 903 // but creating an abstract impl of AbstractKualiDecimalHolder as the intermediate class 904 // * similar to ConcreteNarrowedNumberHolderSub, but ConcreteNarrowedKualiDecimalHolderSub 905 } 906 907}