[commons-beanutils] 09/56: [svn-inject] Forking commons-beanutils source to Trunk
Emmanuel Bourg
ebourg-guest at moszumanska.debian.org
Fri May 30 12:01:04 UTC 2014
This is an automated email from the git hooks/post-receive script.
ebourg-guest pushed a commit to branch master
in repository commons-beanutils.
commit c1cf3dca3afe4fd82de6869f060be83f472523ea
Author: Arnaud Vandyck <avdyk at debian.org>
Date: Thu Mar 9 22:05:16 2006 +0000
[svn-inject] Forking commons-beanutils source to Trunk
---
trunk/.cvsignore | 8 +
trunk/LICENSE.txt | 202 ++
trunk/NOTICE.txt | 2 +
trunk/PROPOSAL.html | 128 +
trunk/README.txt | 26 +
trunk/RELEASE-NOTES.txt | 90 +
trunk/STATUS.html | 131 +
trunk/build.properties.sample | 38 +
trunk/build.xml | 575 +++
trunk/maven.xml | 19 +
trunk/optional/bean-collections/.cvsignore | 4 +
trunk/optional/bean-collections/LICENSE.txt | 202 ++
.../bean-collections/build.properties.sample | 39 +
trunk/optional/bean-collections/build.xml | 301 ++
trunk/optional/bean-collections/project.xml | 50 +
.../optional/bean-collections/src/conf/MANIFEST.MF | 7 +
.../apache/commons/beanutils/BeanComparator.java | 191 +
.../java/org/apache/commons/beanutils/BeanMap.java | 803 +++++
.../apache/commons/beanutils/BeanPredicate.java | 121 +
.../beanutils/BeanPropertyValueChangeClosure.java | 224 ++
.../BeanPropertyValueEqualsPredicate.java | 275 ++
.../beanutils/BeanToPropertyValueTransformer.java | 205 ++
.../bean-collections/src/java/overview.html | 19 +
.../apache/commons/beanutils/AbstractParent.java | 53 +
.../org/apache/commons/beanutils/AlphaBean.java | 41 +
.../beanutils/BeanCollectionsTestSuite.java | 57 +
.../commons/beanutils/BeanComparatorTestCase.java | 261 ++
.../commons/beanutils/BeanPredicateTestCase.java | 61 +
.../BeanPropertyValueChangeClosureTest.java | 321 ++
.../BeanPropertyValueEqualsPredicateTest.java | 261 ++
.../BeanToPropertyValueTransformerTest.java | 246 ++
.../test/org/apache/commons/beanutils/Child.java | 23 +
.../org/apache/commons/beanutils/TestBean.java | 572 +++
.../org/apache/commons/beanutils/TestBeanMap.java | 344 ++
trunk/optional/bean-collections/xdocs/index.xml | 85 +
.../optional/bean-collections/xdocs/navigation.xml | 40 +
trunk/project.properties | 23 +
trunk/project.xml | 194 ++
trunk/src/conf/MANIFEST.MF | 7 +
.../apache/commons/beanutils/BasicDynaBean.java | 402 +++
.../apache/commons/beanutils/BasicDynaClass.java | 290 ++
.../beanutils/BeanAccessLanguageException.java | 47 +
.../org/apache/commons/beanutils/BeanUtils.java | 315 ++
.../apache/commons/beanutils/BeanUtilsBean.java | 1075 ++++++
.../apache/commons/beanutils/ConstructorUtils.java | 408 +++
.../commons/beanutils/ContextClassLoaderLocal.java | 142 +
.../commons/beanutils/ConversionException.java | 89 +
.../org/apache/commons/beanutils/ConvertUtils.java | 298 ++
.../apache/commons/beanutils/ConvertUtilsBean.java | 581 ++++
.../org/apache/commons/beanutils/Converter.java | 48 +
.../commons/beanutils/ConvertingWrapDynaBean.java | 76 +
.../org/apache/commons/beanutils/DynaBean.java | 166 +
.../org/apache/commons/beanutils/DynaClass.java | 87 +
.../org/apache/commons/beanutils/DynaProperty.java | 315 ++
.../apache/commons/beanutils/JDBCDynaClass.java | 214 ++
.../org/apache/commons/beanutils/LazyDynaBean.java | 836 +++++
.../apache/commons/beanutils/LazyDynaClass.java | 337 ++
.../org/apache/commons/beanutils/LazyDynaMap.java | 428 +++
.../beanutils/MappedPropertyDescriptor.java | 562 +++
.../org/apache/commons/beanutils/MethodUtils.java | 851 +++++
.../apache/commons/beanutils/MutableDynaClass.java | 120 +
.../commons/beanutils/NestedNullException.java | 46 +
.../apache/commons/beanutils/PropertyUtils.java | 581 ++++
.../commons/beanutils/PropertyUtilsBean.java | 1784 ++++++++++
.../commons/beanutils/ResultSetDynaClass.java | 200 ++
.../commons/beanutils/ResultSetIterator.java | 344 ++
.../apache/commons/beanutils/RowSetDynaClass.java | 247 ++
.../org/apache/commons/beanutils/WrapDynaBean.java | 333 ++
.../apache/commons/beanutils/WrapDynaClass.java | 298 ++
.../converters/AbstractArrayConverter.java | 168 +
.../beanutils/converters/BigDecimalConverter.java | 124 +
.../beanutils/converters/BigIntegerConverter.java | 124 +
.../converters/BooleanArrayConverter.java | 183 +
.../beanutils/converters/BooleanConverter.java | 140 +
.../beanutils/converters/ByteArrayConverter.java | 145 +
.../beanutils/converters/ByteConverter.java | 127 +
.../converters/CharacterArrayConverter.java | 145 +
.../beanutils/converters/CharacterConverter.java | 123 +
.../beanutils/converters/ClassConverter.java | 130 +
.../beanutils/converters/DoubleArrayConverter.java | 145 +
.../beanutils/converters/DoubleConverter.java | 126 +
.../beanutils/converters/FileConverter.java | 110 +
.../beanutils/converters/FloatArrayConverter.java | 145 +
.../beanutils/converters/FloatConverter.java | 125 +
.../converters/IntegerArrayConverter.java | 145 +
.../beanutils/converters/IntegerConverter.java | 125 +
.../beanutils/converters/LongArrayConverter.java | 145 +
.../beanutils/converters/LongConverter.java | 125 +
.../beanutils/converters/ShortArrayConverter.java | 145 +
.../beanutils/converters/ShortConverter.java | 125 +
.../beanutils/converters/SqlDateConverter.java | 124 +
.../beanutils/converters/SqlTimeConverter.java | 124 +
.../converters/SqlTimestampConverter.java | 124 +
.../beanutils/converters/StringArrayConverter.java | 146 +
.../beanutils/converters/StringConverter.java | 63 +
.../commons/beanutils/converters/URLConverter.java | 127 +
.../commons/beanutils/converters/package.html | 12 +
.../beanutils/locale/BaseLocaleConverter.java | 235 ++
.../commons/beanutils/locale/LocaleBeanUtils.java | 533 +++
.../beanutils/locale/LocaleBeanUtilsBean.java | 957 +++++
.../beanutils/locale/LocaleConvertUtils.java | 291 ++
.../beanutils/locale/LocaleConvertUtilsBean.java | 456 +++
.../commons/beanutils/locale/LocaleConverter.java | 45 +
.../converters/BigDecimalLocaleConverter.java | 198 ++
.../converters/BigIntegerLocaleConverter.java | 198 ++
.../locale/converters/ByteLocaleConverter.java | 221 ++
.../locale/converters/DateLocaleConverter.java | 279 ++
.../locale/converters/DecimalLocaleConverter.java | 242 ++
.../locale/converters/DoubleLocaleConverter.java | 214 ++
.../locale/converters/FloatLocaleConverter.java | 221 ++
.../locale/converters/IntegerLocaleConverter.java | 221 ++
.../locale/converters/LongLocaleConverter.java | 199 ++
.../locale/converters/ShortLocaleConverter.java | 199 ++
.../locale/converters/SqlDateLocaleConverter.java | 218 ++
.../locale/converters/SqlTimeLocaleConverter.java | 217 ++
.../converters/SqlTimestampLocaleConverter.java | 216 ++
.../locale/converters/StringLocaleConverter.java | 286 ++
.../beanutils/locale/converters/package.html | 12 +
.../apache/commons/beanutils/locale/package.html | 10 +
.../java/org/apache/commons/beanutils/package.html | 990 ++++++
.../org/apache/commons/collections/ArrayStack.java | 199 ++
.../org/apache/commons/collections/Buffer.java | 68 +
.../collections/BufferUnderflowException.java | 76 +
.../apache/commons/collections/FastHashMap.java | 714 ++++
.../org/apache/commons/collections/package.html | 13 +
trunk/src/java/overview.html | 17 +
trunk/src/media/logo.xcf | Bin 0 -> 22422 bytes
trunk/src/test/org/apache/commons/beanutils/A.java | 34 +
.../apache/commons/beanutils/AbstractChild.java | 33 +
.../apache/commons/beanutils/AbstractParent.java | 53 +
.../org/apache/commons/beanutils/AlphaBean.java | 41 +
.../commons/beanutils/BasicDynaBeanTestCase.java | 1030 ++++++
.../commons/beanutils/BeanUtilsBenchCase.java | 373 ++
.../commons/beanutils/BeanUtilsTestCase.java | 1341 +++++++
.../commons/beanutils/BeanWithInnerBean.java | 45 +
.../commons/beanutils/BeanificationTestCase.java | 541 +++
.../org/apache/commons/beanutils/BenchBean.java | 146 +
.../org/apache/commons/beanutils/BetaBean.java | 38 +
.../test/org/apache/commons/beanutils/Child.java | 23 +
.../beanutils/ConstructorUtilsTestCase.java | 257 ++
.../commons/beanutils/ConvertUtilsTestCase.java | 654 ++++
.../commons/beanutils/DynaBeanUtilsTestCase.java | 1266 +++++++
.../beanutils/DynaPropertyUtilsTestCase.java | 2648 ++++++++++++++
.../commons/beanutils/DynaResultSetTestCase.java | 249 ++
.../commons/beanutils/DynaRowSetTestCase.java | 247 ++
.../apache/commons/beanutils/ExtendMapBean.java | 44 +
.../commons/beanutils/LazyDynaBeanTestCase.java | 483 +++
.../commons/beanutils/LazyDynaClassTestCase.java | 256 ++
.../commons/beanutils/LazyDynaMapTestCase.java | 486 +++
.../commons/beanutils/MappedPropertyTestBean.java | 43 +
.../commons/beanutils/MethodUtilsTestCase.java | 560 +++
.../apache/commons/beanutils/NestedTestBean.java | 107 +
.../commons/beanutils/PassTestException.java | 27 +
.../apache/commons/beanutils/PrimitiveBean.java | 69 +
.../commons/beanutils/PropertyUtilsBenchCase.java | 266 ++
.../commons/beanutils/PropertyUtilsTestCase.java | 3665 ++++++++++++++++++++
.../apache/commons/beanutils/SonOfAlphaBean.java | 25 +
.../org/apache/commons/beanutils/TestBean.java | 572 +++
.../commons/beanutils/TestBeanPackageSubclass.java | 32 +
.../commons/beanutils/TestBeanPublicSubclass.java | 32 +
.../apache/commons/beanutils/TestResultSet.java | 864 +++++
.../commons/beanutils/TestResultSetMetaData.java | 170 +
.../commons/beanutils/ThrowExceptionConverter.java | 35 +
.../commons/beanutils/WrapDynaBeanTestCase.java | 182 +
.../converters/ByteConverterTestCase.java | 124 +
.../beanutils/converters/ConverterTestSuite.java | 50 +
.../converters/DoubleConverterTestCase.java | 136 +
.../converters/FileConverterTestCase.java | 96 +
.../converters/FloatConverterTestCase.java | 136 +
.../converters/IntegerConverterTestCase.java | 124 +
.../converters/LongConverterTestCase.java | 124 +
.../converters/NumberConverterTestBase.java | 91 +
.../converters/ShortConverterTestCase.java | 124 +
.../converters/StringArrayConverterTestCase.java | 48 +
.../beanutils/converters/URLConverterTestCase.java | 111 +
.../locale/LocaleBeanificationTestCase.java | 533 +++
.../beanutils/locale/LocaleConvertTestSuite.java | 48 +
.../locale/LocaleConvertUtilsTestCase.java | 639 ++++
.../converters/DateLocaleConverterTestCase.java | 143 +
.../apache/commons/beanutils/priv/PackageBean.java | 69 +
.../apache/commons/beanutils/priv/PrivateBean.java | 110 +
.../commons/beanutils/priv/PrivateBeanFactory.java | 52 +
.../beanutils/priv/PrivateBeanSubclass.java | 58 +
.../commons/beanutils/priv/PrivateDirect.java | 45 +
.../commons/beanutils/priv/PrivateIndirect.java | 46 +
.../commons/beanutils/priv/PublicSubBean.java | 69 +
.../apache/commons/collections/ReferenceMap.java | 957 +++++
.../collections/keyvalue/AbstractKeyValue.java | 81 +
.../collections/keyvalue/AbstractMapEntry.java | 92 +
.../collections/keyvalue/DefaultMapEntry.java | 66 +
.../commons/collections/keyvalue/KeyValue.java | 46 +
trunk/xdocs/.cvsignore | 1 +
trunk/xdocs/images/logo.png | Bin 0 -> 12805 bytes
trunk/xdocs/index.xml | 153 +
trunk/xdocs/navigation.xml | 48 +
trunk/xdocs/proposal.xml | 136 +
trunk/xdocs/style/project.css | 5 +
197 files changed, 50602 insertions(+)
diff --git a/trunk/.cvsignore b/trunk/.cvsignore
new file mode 100644
index 0000000..fc3f324
--- /dev/null
+++ b/trunk/.cvsignore
@@ -0,0 +1,8 @@
+build.properties
+dist
+target
+beanutils.ipr
+maven.log
+velocity.log
+.classpath
+.project
diff --git a/trunk/LICENSE.txt b/trunk/LICENSE.txt
new file mode 100644
index 0000000..d645695
--- /dev/null
+++ b/trunk/LICENSE.txt
@@ -0,0 +1,202 @@
+
+ Apache License
+ Version 2.0, January 2004
+ http://www.apache.org/licenses/
+
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+ 1. Definitions.
+
+ "License" shall mean the terms and conditions for use, reproduction,
+ and distribution as defined by Sections 1 through 9 of this document.
+
+ "Licensor" shall mean the copyright owner or entity authorized by
+ the copyright owner that is granting the License.
+
+ "Legal Entity" shall mean the union of the acting entity and all
+ other entities that control, are controlled by, or are under common
+ control with that entity. For the purposes of this definition,
+ "control" means (i) the power, direct or indirect, to cause the
+ direction or management of such entity, whether by contract or
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
+ outstanding shares, or (iii) beneficial ownership of such entity.
+
+ "You" (or "Your") shall mean an individual or Legal Entity
+ exercising permissions granted by this License.
+
+ "Source" form shall mean the preferred form for making modifications,
+ including but not limited to software source code, documentation
+ source, and configuration files.
+
+ "Object" form shall mean any form resulting from mechanical
+ transformation or translation of a Source form, including but
+ not limited to compiled object code, generated documentation,
+ and conversions to other media types.
+
+ "Work" shall mean the work of authorship, whether in Source or
+ Object form, made available under the License, as indicated by a
+ copyright notice that is included in or attached to the work
+ (an example is provided in the Appendix below).
+
+ "Derivative Works" shall mean any work, whether in Source or Object
+ form, that is based on (or derived from) the Work and for which the
+ editorial revisions, annotations, elaborations, or other modifications
+ represent, as a whole, an original work of authorship. For the purposes
+ of this License, Derivative Works shall not include works that remain
+ separable from, or merely link (or bind by name) to the interfaces of,
+ the Work and Derivative Works thereof.
+
+ "Contribution" shall mean any work of authorship, including
+ the original version of the Work and any modifications or additions
+ to that Work or Derivative Works thereof, that is intentionally
+ submitted to Licensor for inclusion in the Work by the copyright owner
+ or by an individual or Legal Entity authorized to submit on behalf of
+ the copyright owner. For the purposes of this definition, "submitted"
+ means any form of electronic, verbal, or written communication sent
+ to the Licensor or its representatives, including but not limited to
+ communication on electronic mailing lists, source code control systems,
+ and issue tracking systems that are managed by, or on behalf of, the
+ Licensor for the purpose of discussing and improving the Work, but
+ excluding communication that is conspicuously marked or otherwise
+ designated in writing by the copyright owner as "Not a Contribution."
+
+ "Contributor" shall mean Licensor and any individual or Legal Entity
+ on behalf of whom a Contribution has been received by Licensor and
+ subsequently incorporated within the Work.
+
+ 2. Grant of Copyright License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ copyright license to reproduce, prepare Derivative Works of,
+ publicly display, publicly perform, sublicense, and distribute the
+ Work and such Derivative Works in Source or Object form.
+
+ 3. Grant of Patent License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ (except as stated in this section) patent license to make, have made,
+ use, offer to sell, sell, import, and otherwise transfer the Work,
+ where such license applies only to those patent claims licensable
+ by such Contributor that are necessarily infringed by their
+ Contribution(s) alone or by combination of their Contribution(s)
+ with the Work to which such Contribution(s) was submitted. If You
+ institute patent litigation against any entity (including a
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
+ or a Contribution incorporated within the Work constitutes direct
+ or contributory patent infringement, then any patent licenses
+ granted to You under this License for that Work shall terminate
+ as of the date such litigation is filed.
+
+ 4. Redistribution. You may reproduce and distribute copies of the
+ Work or Derivative Works thereof in any medium, with or without
+ modifications, and in Source or Object form, provided that You
+ meet the following conditions:
+
+ (a) You must give any other recipients of the Work or
+ Derivative Works a copy of this License; and
+
+ (b) You must cause any modified files to carry prominent notices
+ stating that You changed the files; and
+
+ (c) You must retain, in the Source form of any Derivative Works
+ that You distribute, all copyright, patent, trademark, and
+ attribution notices from the Source form of the Work,
+ excluding those notices that do not pertain to any part of
+ the Derivative Works; and
+
+ (d) If the Work includes a "NOTICE" text file as part of its
+ distribution, then any Derivative Works that You distribute must
+ include a readable copy of the attribution notices contained
+ within such NOTICE file, excluding those notices that do not
+ pertain to any part of the Derivative Works, in at least one
+ of the following places: within a NOTICE text file distributed
+ as part of the Derivative Works; within the Source form or
+ documentation, if provided along with the Derivative Works; or,
+ within a display generated by the Derivative Works, if and
+ wherever such third-party notices normally appear. The contents
+ of the NOTICE file are for informational purposes only and
+ do not modify the License. You may add Your own attribution
+ notices within Derivative Works that You distribute, alongside
+ or as an addendum to the NOTICE text from the Work, provided
+ that such additional attribution notices cannot be construed
+ as modifying the License.
+
+ You may add Your own copyright statement to Your modifications and
+ may provide additional or different license terms and conditions
+ for use, reproduction, or distribution of Your modifications, or
+ for any such Derivative Works as a whole, provided Your use,
+ reproduction, and distribution of the Work otherwise complies with
+ the conditions stated in this License.
+
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
+ any Contribution intentionally submitted for inclusion in the Work
+ by You to the Licensor shall be under the terms and conditions of
+ this License, without any additional terms or conditions.
+ Notwithstanding the above, nothing herein shall supersede or modify
+ the terms of any separate license agreement you may have executed
+ with Licensor regarding such Contributions.
+
+ 6. Trademarks. This License does not grant permission to use the trade
+ names, trademarks, service marks, or product names of the Licensor,
+ except as required for reasonable and customary use in describing the
+ origin of the Work and reproducing the content of the NOTICE file.
+
+ 7. Disclaimer of Warranty. Unless required by applicable law or
+ agreed to in writing, Licensor provides the Work (and each
+ Contributor provides its Contributions) on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+ implied, including, without limitation, any warranties or conditions
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+ PARTICULAR PURPOSE. You are solely responsible for determining the
+ appropriateness of using or redistributing the Work and assume any
+ risks associated with Your exercise of permissions under this License.
+
+ 8. Limitation of Liability. In no event and under no legal theory,
+ whether in tort (including negligence), contract, or otherwise,
+ unless required by applicable law (such as deliberate and grossly
+ negligent acts) or agreed to in writing, shall any Contributor be
+ liable to You for damages, including any direct, indirect, special,
+ incidental, or consequential damages of any character arising as a
+ result of this License or out of the use or inability to use the
+ Work (including but not limited to damages for loss of goodwill,
+ work stoppage, computer failure or malfunction, or any and all
+ other commercial damages or losses), even if such Contributor
+ has been advised of the possibility of such damages.
+
+ 9. Accepting Warranty or Additional Liability. While redistributing
+ the Work or Derivative Works thereof, You may choose to offer,
+ and charge a fee for, acceptance of support, warranty, indemnity,
+ or other liability obligations and/or rights consistent with this
+ License. However, in accepting such obligations, You may act only
+ on Your own behalf and on Your sole responsibility, not on behalf
+ of any other Contributor, and only if You agree to indemnify,
+ defend, and hold each Contributor harmless for any liability
+ incurred by, or claims asserted against, such Contributor by reason
+ of your accepting any such warranty or additional liability.
+
+ END OF TERMS AND CONDITIONS
+
+ APPENDIX: How to apply the Apache License to your work.
+
+ To apply the Apache License to your work, attach the following
+ boilerplate notice, with the fields enclosed by brackets "[]"
+ replaced with your own identifying information. (Don't include
+ the brackets!) The text should be enclosed in the appropriate
+ comment syntax for the file format. We also recommend that a
+ file or class name and description of purpose be included on the
+ same "printed page" as the copyright notice for easier
+ identification within third-party archives.
+
+ Copyright [yyyy] [name of copyright owner]
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
diff --git a/trunk/NOTICE.txt b/trunk/NOTICE.txt
new file mode 100644
index 0000000..3f59805
--- /dev/null
+++ b/trunk/NOTICE.txt
@@ -0,0 +1,2 @@
+This product includes software developed by
+The Apache Software Foundation (http://www.apache.org/).
diff --git a/trunk/PROPOSAL.html b/trunk/PROPOSAL.html
new file mode 100755
index 0000000..3ecf654
--- /dev/null
+++ b/trunk/PROPOSAL.html
@@ -0,0 +1,128 @@
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<html>
+<head>
+<title>Proposal for BeanUtils Package</title>
+</head>
+<body bgcolor="white">
+
+<div align="center">
+<h1>Proposal for <em>BeanUtils</em> Package</h1>
+</div>
+
+<h3>(0) Rationale</h3>
+
+<p>Most Java developers are used to creating Java classes that conform to the
+JavaBeans naming patterns for property getters and setters. It is natural to
+then access these methods directly, using calls to the corresponding
+<code>getXxx</code> and <code>setXxx</code> methods. However, there are some
+occasions where dynamic access to Java object properties (without compiled-in
+knowledge of the property getter and setter methods to be called) is needed.
+Example use cases include:</p>
+<ul>
+<li>Building scripting languages that interact with the Java object model
+ (such as the Bean Scripting Framework).</li>
+<li>Building template language processors for web presentation and similar
+ uses (such as JSP or Velocity).</li>
+<li>Building custom tag libraries for JSP and XSP environments (such as Jakarta
+ Taglibs, Struts, Cocoon).</li>
+<li>Consuming XML-based configuration resources (such as Ant build scripts, web
+ application deployment descriptors, Tomcat's <code>server.xml</code>
+ file).</li>
+</ul>
+
+<p>The Java language provides <em>Reflection</em> and <em>Introspection</em>
+APIs (see the <code>java.lang.reflect</code> and <code>java.beans</code>
+packages in the JDK Javadocs). However, these APIs can be quite complex to
+understand and utilize. The proposed <em>BeanUtils</em> component provides
+easy-to-use wrappers around these capabilities.</p>
+
+
+<h3>(1) Scope of the Package</h3>
+
+<p>This proposal is to create a package of Java utility methods for accessing
+and modifying the properties of arbitrary JavaBeans. No dependencies outside
+of the JDK are required, so the use of this package is very lightweight.</p>
+
+<p>In addition to wrapping the reflection and introspection APIs of the
+standard JDK, <em>BeanUtils</em> components shall support a syntax for directly
+accessing <strong>nested</strong> and <strong>indexed</strong> properties, in a
+manner that will be familar to users of scripting languages like JavaScript.
+For example, the following property accessor expressions are supported:</p>
+<ul>
+<li><strong>customer</strong> - Equivalent to <code>getCustomer()</code>.</li>
+<li><strong>customer.address</strong> - Equivalent to
+ <code>getCustomer().getAddress()</code>.</li>
+<li><strong>customer.address[2].street</strong> - Equivalent to
+ <code>getCustomer().getAddress(2).getStreet()</code> (access to indexed
+ properties also works if the underlying property is an array rather than
+ providing indexed getter and setter methods).</li>
+</ul>
+
+
+<h3>(1.5) Interaction With Other Packages</h3>
+
+<p><em>BeanUtils</em> relies only on standard JDK 1.2 (or later) APIs for
+production deployment. It utilizes the JUnit unit testing framework for
+developing and executing unit tests, but this is of interest only to
+developers of the component. BeanUtils will also be a dependency for
+several future proposed components for the Jakarta Commons subproject.</p>
+
+<p>No external configuration files are utilized.</p>
+
+
+<h3>(2) Initial Source of the Package</h3>
+
+<p>The three original Java classes (<code>BeanUtils</code>,
+<code>ConvertUtils</code>, and <code>PropertyUtils</code>) are an integral
+part of the <a href="http://jakarta.apache.org/struts">Struts Framework</a>.
+However, they have very few dependencies on other aspects of Struts, and
+those dependencies have been removed in the proposed code base.
+Once accepted and released as a Jakarta Commons component, Struts will
+be modified to use the Commons version of these classes, and its internal
+versions will be deprecated.</p>
+
+<p>The proposed package name for the new component is
+<code>org.apache.commons.beanutils</code>.</p>
+
+
+<h3>(3) Required Jakarta-Commons Resources</h3>
+
+<ul>
+<li>CVS Repository - New directory <code>beanutils</code> in the
+ <code>jakarta-commons</code> CVS repository. All initial committers
+ are already committers on <code>jakarta-commons</code>, so no
+ additional user setups are required.</li>
+<li>Mailing List - Discussions will take place on the general
+ <em>jakarta-commons at jakarta.apache.org</em> mailing list. To help
+ list subscribers identify messages of interest, it is suggested that
+ the message subject of messages about this component be prefixed with
+ [BeanUtils].</li>
+<li>Bugzilla - New component "BeanUtils" under the "Commons" product
+ category, with appropriate version identifiers as needed.</li>
+<li>Jyve FAQ - New category "commons-beanutils" (when available). </li>
+</ul>
+
+
+<h3>(4) Initial Committers</h3>
+
+<p>The initial committers on the BeanUtils component shall be Craig
+McClanahan and Geir Magnusson Jr.</p>
+
+
+
+</body>
+</html>
diff --git a/trunk/README.txt b/trunk/README.txt
new file mode 100644
index 0000000..74497fc
--- /dev/null
+++ b/trunk/README.txt
@@ -0,0 +1,26 @@
+README
+======
+
+Jars
+----
+
+BeanUtils now comes packaged (into jars) in two different ways:
+
+ an all-in-one jar (commons-beanutils.jar);
+ and as modular component jars:
+ commons-beanutils-core.jar (core classes)
+ commons-beanutils-bean-collections.jar (additional dependency on commons-collections 3.0)
+
+Those who want it all should grab the all-in-one jar (and the optional dependencies)
+whereas those who need minimal dependencies should use commons-beanutils-core.jar
+plus any optional jars they plan to use.
+
+All classes that were in the last release are in commons-beanutils-core except for BeanComparator.
+
+Commons-Collections
+-------------------
+BeanUtils now ships with a small number of commons collections classes.
+This is a temporary measure intended to allow BeanUtils core to be used with
+either commons-collections 2 or commons-collections-3 without a dependency on either.
+It is intended that soon BeanUtils core will have no dependency on any commons collection
+packaged classes.
\ No newline at end of file
diff --git a/trunk/RELEASE-NOTES.txt b/trunk/RELEASE-NOTES.txt
new file mode 100644
index 0000000..638c6b7
--- /dev/null
+++ b/trunk/RELEASE-NOTES.txt
@@ -0,0 +1,90 @@
+$Id: RELEASE-NOTES.txt,v 1.11.2.3 2004/08/01 20:15:19 rdonkin Exp $
+
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+
+
+ Commons BeanUtils Package
+ Version 1.7.0
+ Release Notes
+
+INTRODUCTION:
+============
+
+Beanutils 1.7.0 is a service release aimed at providing compatibility
+with both the commons collections 2.x series of releases
+and the commons collections 3.x series of releases
+
+CHANGES:
+========
+
+Upgraded License To Apache License 2.0
+--------------------------------------
+Beanutils is now released under the Apache License 2.0.
+See http://www.apache.org/licenses/LICENSE-2.0.
+
+Beanification
+-------------
+Creation of objects to back the static utility classes. These object are
+per-context-classloader pseudo-singletons. Each web or enterprise application
+is therefore isolated from changes made to the state of others suing the
+static facades. Greater flexibility of implementation is encourage since users
+can subclass and then set their own implementations. Calls to the static facades
+will then be passed to that implementation.
+
+Removal Of Commons Collections Dependency
+-----------------------------------------
+The commons collections dependency is in the process of being removed
+from core beanutils. This will reduce the number of dependencies for
+the core beanutils and also will allow beanutils to used with both
+collection 2.x and collection 3.x releases.
+
+Documentation Improvements
+--------------------------
+Many thanks to all those kind souls who've contributed documentation :)
+
+
+ENHANCEMENTS:
+=============
+
+BeanAccessLanguageException & NestNullException
+-----------------------------------------------
+Added new subclasses of RuntimeException so that bean access language
+exceptions can be trapped by users.
+
+BeanComparator
+--------------
+Added no-argument constructor for use in bean-centric environments.
+
+ConvertUtilsBean
+----------------
+Added a File converter and registered the File and URL converters by default
+
+
+BUG REPORTS ADDRESSED:
+=====================
+ #14848 Converted localized versions of beanutils and convert utils to use
+ delegated singletons. Now instances with the functionality in these
+ classes can be created.
+ Added public constructors for the utility objects (BeanUtilsBean,
+ PropertyUtilsBean and ConvertUtilsBean). Add public accessor properties
+ for the ConvertUtilsBean and PropertyUtilsBean instances used by a
+ BeanUtilsBean. This allows BeanUtilsBean objects to be created with
+ independent registered converters and independent caches. Also added
+ test cases.
+ #17663 Made BeanUtils.getArrayProperty conversions use ConvertUtils
+ (rather than just toString)
+ #18918 This bug prevented converters from being correctly deregistered
+ #19850 Now cloneBean will deal successfully with DynaBeans.
+
diff --git a/trunk/STATUS.html b/trunk/STATUS.html
new file mode 100755
index 0000000..80feef3
--- /dev/null
+++ b/trunk/STATUS.html
@@ -0,0 +1,131 @@
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<html>
+<head>
+<title>Status File for Jakarta Commons "BeanUtils" Component</title>
+<head>
+<body bgcolor="white">
+
+
+<div align="center">
+<h1>The Jakarta Commons <em>BeanUtils</em> Component</h1>
+$Id: STATUS.html,v 1.16 2004/02/29 21:17:51 scolebourne Exp $<br>
+<a href="#Introduction">[Introduction]</a>
+<a href="#Dependencies">[Dependencies]</a>
+<a href="#Release Info">[Release Info]</a>
+<a href="#Committers">[Committers]</a>
+<a href="#Action Items">[Action Items]</a>
+<br><br>
+</div>
+
+
+<a name="Introduction"></a>
+<h3>1. INTRODUCTION</h3>
+
+<p>The <em>BeanUtils</em> Component contains a set of Java classes that provide
+static utility methods useful in manipulating Java classes that conform to the
+JavaBeans Specification naming patterns for bean properties in a dynamic
+fashion. The following classes are included:</p>
+<ul>
+<li><strong>BeanUtils</strong> - Higher level getters and setters that deal
+ with automatic conversion of String-represented property values to the
+ corresponding data types of the underlying property setters (and vice
+ versa), using the capabilities of the ConvertUtils and PropertyUtils
+ classes.</li>
+<li><strong>ConvertUtils</strong> - Utility methods to convert String arguments
+ to native Java types (and vice versa)</li>.
+<li><strong>PropertyUtils</strong> - Low level getters and setters for bean
+ properties that perform no type conversions at all. Works on
+ <code>List</code> objects as well (Not JavaBeans standard).</li>
+</ul>
+
+<p>An innovative and unique feature of this component is the syntax for
+accessing nested and indexed bean properties. See the Javadocs
+<a href="http://jakarta.apache.org/commons/beanutils/api/">PropertyUtils</a>
+for more information about this feature.</p>
+
+
+<a name="Dependencies"></a>
+<h3>2. DEPENDENCIES</h3>
+
+<p>The <em>BeanUtils</em> component is dependent upon the following external
+components for development and use:</p>
+<ul>
+<li><a href="http://java.sun.com/j2se">Java Development Kit</a>
+ (Version 1.2 or later)</li>
+<li><a href="http://jakarta.apache.org/commons">Collections Classes</a>
+ from the Jakarta Commons Subproject</li>
+<li><a href="http://jakarta.apache.org/commons">Logging Classes</a>
+ from the Jakarta Commons Subproject</li>
+<li><a href="http://www.junit.org">JUnit Testing Framework</a>
+ (Version 3.7 or later) - for unit tests only, not required
+ for deployment</li>
+</ul>
+
+
+<a name="Release Info"></a>
+<h3>3. RELEASE INFO</h3>
+
+<p>Current Release: Version 1.6.1</p>
+
+<p>Planned Next Release: 1.7</p>
+
+
+<a name="Committers"></a>
+<h3>4. COMMITTERS</h3>
+
+<p>The following individuals are the primary developers and maintainers of this
+component. Developers who plan to use <em>BeanUtils</em> in their own
+projects are encouraged to collaborate on the future development of this
+component to ensure that it continues to meet a variety of needs.</p>
+<ul>
+<li><a href="mailto:craigmcc at apache.org">Craig McClanahan</a></li>
+<li><a href="mailto:geirm at apache.org">Geir Magnusson Jr.</a></li>
+<li><a href="mailto:sanders at apache.org">Scott Sanders</a></li>
+<li><a href="mailto:rdonkin at apache.org">Robert Burrell Donkin</a></li>
+<li><a href="mailto:dion at apache.org">dIon Gillard</a></li>
+<li><a href="mailto:jstrachan at apache.org">James Strachan</a></li>
+<li><a href="mailto:yoavs at apache.org">Yoav Shapira</a></li>
+</ul>
+
+
+<a name="Action Items"></a>
+<h3>5. ACTION ITEMS</h3>
+
+<p>The following action items need to be completed prior to the next
+release of this component:</p>
+
+<table border="1">
+
+ <tr>
+ <th width="80%">Action Item</th>
+ <th width="20%">Volunteer</th>
+ </tr>
+
+ <tr>
+ <td><strong>Finish locale-sensitive beans and converters.</strong>.
+ <td align="center">All</td>
+ </tr>
+
+ <tr>
+ <td><strong>Add test cases</strong>. Especially for i18n classes.</td>
+ <td align="center">All</td>
+ </tr>
+
+</table>
+
+</body>
+</html>
diff --git a/trunk/build.properties.sample b/trunk/build.properties.sample
new file mode 100755
index 0000000..592f62b
--- /dev/null
+++ b/trunk/build.properties.sample
@@ -0,0 +1,38 @@
+# Copyright 2001-2004 The Apache Software Foundation
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+# The home directory for the Commons collection classes distribution
+commons-collections.home = ../collections/dist
+
+# The pathname of the collections classes JAR file
+commons-collections.jar = ${commons-collections.home}/commons-collections.jar
+
+# The home directory for the Commons logging classes distribution
+commons-logging.home = ../logging/dist
+
+# The pathname of the Commons Logging JAR file
+commons-logging.jar = ${commons-logging.home}/commons-logging.jar
+
+# The directory containing your binary distribution of JUnit,
+# version 3.7 or later
+junit.home = /usr/local/junit3.7
+
+# The pathname of the "junit.jar" JAR file
+junit.jar = ${junit.home}/junit.jar
+
+
+# Maven properties (for web site build)
+# Those committers using agents may like to use
+#maven.username=rdonkin
+#beanutils.cvs=lserver:rdonkin at cvs.apache.org:/home/cvs
\ No newline at end of file
diff --git a/trunk/build.xml b/trunk/build.xml
new file mode 100755
index 0000000..b174b0f
--- /dev/null
+++ b/trunk/build.xml
@@ -0,0 +1,575 @@
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<project name="Bean Utilities" default="compile" basedir=".">
+
+
+<!--
+ "Bean Utilities" component of the Jakarta Commons Subproject
+ $Id: build.xml,v 1.59.2.3 2004/07/27 21:59:14 rdonkin Exp $
+-->
+
+
+<!-- ========== Initialize Properties ===================================== -->
+
+
+ <property file="build.properties"/> <!-- Component local -->
+ <property file="../build.properties"/> <!-- Commons local -->
+ <property file="${user.home}/build.properties"/> <!-- User local -->
+
+
+<!-- ========== External Dependencies ===================================== -->
+
+ <!-- The home directory for the Commons collection classes distribution -->
+ <property name="commons-logging.home" value="../logging/dist"/>
+
+ <!-- The directory containing your binary distribution of JUnit,
+ version 3.7 or later -->
+ <property name="junit.home" value="/usr/local/junit3.7"/>
+
+
+<!-- ========== Derived Values ============================================ -->
+
+ <!-- The pathname of the "junit.jar" JAR file -->
+ <property name="junit.jar" value="${junit.home}/junit.jar"/>
+
+
+<!-- ========== Component Declarations ==================================== -->
+
+
+ <!-- The name of this component -->
+ <property name="component.name" value="beanutils"/>
+
+ <!-- The primary package name of this component -->
+ <property name="component.package" value="org.apache.commons.beanutils"/>
+
+ <!-- The title of this component -->
+ <property name="component.title" value="Bean Introspection Utilities"/>
+
+ <!-- The current version number of this component -->
+ <property name="component.version" value="1.7"/>
+
+ <!-- The base directory for compilation targets -->
+ <property name="build.home" value="target"/>
+
+ <!-- The base directory for component configuration files -->
+ <property name="conf.home" value="src/conf"/>
+
+ <!-- The base directory for distribution targets -->
+ <property name="dist.home" value="dist"/>
+
+ <!-- The base directory for component sources -->
+ <property name="source.home" value="src/java"/>
+
+ <!-- The base directory for unit test sources -->
+ <property name="test.home" value="src/test"/>
+
+
+<!-- ========== Compiler Defaults ========================================= -->
+
+
+ <!-- Should Java compilations set the 'debug' compiler option? -->
+ <property name="compile.debug" value="true"/>
+
+ <!-- Should Java compilations set the 'deprecation' compiler option? -->
+ <property name="compile.deprecation" value="false"/>
+
+ <!-- Should Java compilations set the 'optimize' compiler option? -->
+ <property name="compile.optimize" value="true"/>
+
+ <!-- Construct compile classpath -->
+ <path id="compile.classpath">
+ <pathelement location="${build.home}/classes"/>
+ <pathelement location="${commons-logging.jar}"/>
+ </path>
+
+
+<!-- ========== Test Execution Defaults =================================== -->
+
+
+ <!-- Construct unit test classpath -->
+ <path id="test.classpath">
+ <pathelement location="${build.home}/classes"/>
+ <pathelement location="${build.home}/tests"/>
+ <pathelement location="${commons-logging.jar}"/>
+ <pathelement location="${junit.jar}"/>
+ </path>
+
+ <!-- Should all tests fail if one does? -->
+ <property name="test.failonerror" value="true"/>
+
+ <!-- The test runner to execute -->
+ <property name="test.runner" value="junit.textui.TestRunner"/>
+
+ <!-- The Commons Logger LogFactory implementation to use -->
+ <property name="test.factory"
+ value="org.apache.commons.logging.impl.LogFactoryImpl"/>
+
+ <!-- The Commons Logger Log implementation to use (standard factory) -->
+ <property name="test.log"
+ value="org.apache.commons.logging.impl.SimpleLog"/>
+
+ <!-- The Commons Logger SimpleLog level for testing -->
+ <property name="test.level" value="error"/>
+
+ <!-- Loop counter for microbenchmarks -->
+ <property name="bench.counter" value="100000"/>
+
+
+<!-- ========== Executable Targets ======================================== -->
+
+
+ <target name="init"
+ description="Initialize and evaluate conditionals">
+ <echo message="-------- ${component.name} ${component.version} --------"/>
+ <filter token="name" value="${component.name}"/>
+ <filter token="package" value="${component.package}"/>
+ <filter token="version" value="${component.version}"/>
+ </target>
+
+
+ <target name="prepare" depends="init"
+ description="Prepare build directory">
+ <mkdir dir="${build.home}"/>
+ <mkdir dir="${build.home}/classes"/>
+ <mkdir dir="${build.home}/conf"/>
+ <mkdir dir="${build.home}/tests"/>
+ </target>
+
+
+ <target name="static" depends="prepare"
+ description="Copy static files to build directory">
+ <tstamp/>
+ <copy todir="${build.home}/conf" filtering="on">
+ <fileset dir="${conf.home}" includes="*.MF"/>
+ </copy>
+ </target>
+
+
+ <target name="compile" depends="static"
+ description="Compile shareable components">
+ <javac srcdir="${source.home}"
+ destdir="${build.home}/classes"
+ debug="${compile.debug}"
+ deprecation="${compile.deprecation}"
+ optimize="${compile.optimize}">
+ <classpath refid="compile.classpath"/>
+ </javac>
+ <copy todir="${build.home}/classes" filtering="on">
+ <fileset dir="${source.home}" excludes="**/*.java"/>
+ </copy>
+ </target>
+
+
+ <target name="compile.tests" depends="compile"
+ description="Compile unit test cases">
+ <javac srcdir="${test.home}"
+ destdir="${build.home}/tests"
+ debug="${compile.debug}"
+ deprecation="${compile.deprecation}"
+ optimize="${compile.optimize}">
+ <classpath refid="test.classpath"/>
+ </javac>
+ <copy todir="${build.home}/tests" filtering="on">
+ <fileset dir="${test.home}" excludes="**/*.java"/>
+ </copy>
+ </target>
+
+
+ <target name="clean"
+ description="Clean build and distribution directories">
+ <delete dir="${build.home}"/>
+ <delete dir="${dist.home}"/>
+ </target>
+
+
+ <target name="all" depends="clean,compile"
+ description="Clean and compile all components"/>
+
+
+ <target name="javadoc" depends="compile"
+ description="Create component Javadoc documentation">
+ <mkdir dir="${dist.home}"/>
+ <mkdir dir="${dist.home}/docs"/>
+ <mkdir dir="${dist.home}/docs/api"/>
+ <javadoc sourcepath="${source.home}"
+ destdir="${dist.home}/docs/api"
+ packagenames="org.apache.commons.*"
+ author="true"
+ private="true"
+ version="true"
+ overview="src/java/overview.html"
+ doctitle="<h1>${component.title} (Version ${component.version})</h1>"
+ windowtitle="${component.title} (Version ${component.version})"
+ bottom="Copyright (c) 2001-2004 - Apache Software Foundation">
+ <classpath refid="compile.classpath"/>
+ </javadoc>
+ </target>
+
+
+ <target name="dist" depends="compile,javadoc"
+ description="Create binary distribution">
+ <mkdir dir="${dist.home}"/>
+ <copy file="LICENSE.txt"
+ todir="${dist.home}"/>
+ <copy file="NOTICE.txt"
+ todir="${dist.home}"/>
+ <copy file="RELEASE-NOTES.txt"
+ todir="${dist.home}"/>
+ <copy file="README.txt"
+ todir="${dist.home}"/>
+ <antcall target="jar"/>
+ <antcall target="bean-collections-dist"/>
+ </target>
+
+
+ <target name="jar" depends="compile"
+ description="Create jar">
+ <mkdir dir="${dist.home}"/>
+ <mkdir dir="${build.home}/classes/META-INF"/>
+ <copy file="LICENSE.txt"
+ tofile="${build.home}/classes/META-INF/LICENSE.txt"/>
+ <copy file="NOTICE.txt"
+ tofile="${build.home}/classes/META-INF/NOTICE.txt"/>
+ <jar jarfile="${dist.home}/commons-beanutils.jar"
+ basedir="${build.home}/classes"
+ manifest="${build.home}/conf/MANIFEST.MF"/>
+ <copy tofile='${dist.home}/commons-beanutils-core.jar' file='${dist.home}/commons-beanutils.jar'/>
+ </target>
+
+
+ <target name="install-jar" depends="jar"
+ description="--> Installs jar file in ${lib.repo}">
+ <copy todir="${lib.repo}" filtering="no">
+ <fileset dir="${dist.home}">
+ <include name="commons-${component.name}.jar"/>
+ </fileset>
+ </copy>
+ </target>
+
+<!-- ========== Sub Component Targets ===================================== -->
+
+ <target name='bean-collections-dist'>
+ <ant antfile='optional/bean-collections/build.xml' target='dist' inheritAll='false'/>
+ </target>
+
+<!-- ========== Unit Test Targets ========================================= -->
+
+
+ <target name="test" depends="compile.tests,
+ test.basic.dynabean,
+ test.wrap.dynabean,
+ test.dyna.property,
+ test.property,
+ test.dyna.bean,
+ test.bean,
+ test.convert,
+ test.method,
+ test.dyna.result,
+ test.dyna.row,
+ test.locale.convert,
+ test.converters,
+ test.beanification,
+ test.lazy.dynaclass,
+ test.lazy.dynabean,
+ test.lazy.dynamap
+ "
+ description="Run all unit test cases">
+ </target>
+
+
+ <target name="test.basic.dynabean" depends="compile.tests">
+ <echo message="Running BasicDynaBean tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.BasicDynaBeanTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.wrap.dynabean" depends="compile.tests">
+ <echo message="Running WrapDynaBean tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.WrapDynaBeanTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.dyna.property" depends="compile.tests">
+ <echo message="Running DynaBean PropertyUtils tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.DynaPropertyUtilsTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.property" depends="compile.tests">
+ <echo message="Running PropertyUtils tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.PropertyUtilsTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.dyna.bean" depends="compile.tests">
+ <echo message="Running DynaBean BeanUtils tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.DynaBeanUtilsTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.bean" depends="compile.tests">
+ <echo message="Running BeanUtils tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.BeanUtilsTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.convert" depends="compile.tests">
+ <echo message="Running ConvertUtils tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.ConvertUtilsTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.method" depends="compile.tests">
+ <echo message="Running MethodUtils tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.MethodUtilsTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.dyna.result" depends="compile.tests">
+ <echo message="Running DynaBean ResultSet tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.DynaResultSetTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.dyna.row" depends="compile.tests">
+ <echo message="Running DynaBean RowSet tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.DynaRowSetTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.locale.convert" depends="compile.tests">
+ <echo message="Running Locale converters tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.locale.LocaleConvertTestSuite"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+
+ <target name="test.converters" depends="compile.tests">
+ <echo message="Running Converters tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.converters.ConverterTestSuite"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.lazy.dynaclass" depends="compile.tests">
+ <echo message="Running LazyDynaClass tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.LazyDynaClassTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.lazy.dynabean" depends="compile.tests">
+ <echo message="Running LazyDynaBean tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.LazyDynaBeanTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.lazy.dynamap" depends="compile.tests">
+ <echo message="Running LazyDynaMap tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.LazyDynaMapTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="test.beanification" depends="compile.tests">
+ <echo message="Running Beanification tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <arg value="org.apache.commons.beanutils.BeanificationTestCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+
+ <!-- ========== Microbenchmark Cases ===================================== -->
+
+
+ <target name="bench" depends="compile.tests,
+ bench.BeanUtils,
+ bench.PropertyUtils
+ "
+ description="Execute microbenchmark cases"/>
+
+
+ <target name="bench.BeanUtils" depends="compile.tests">
+ <echo message="Running BeanUtils microbenchmarks"/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="counter"
+ value="${bench.counter}"/>
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.BeanUtilsBenchCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+ <target name="bench.PropertyUtils" depends="compile.tests">
+ <echo message="Running PropertyUtils microbenchmarks"/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="counter"
+ value="${bench.counter}"/>
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.PropertyUtilsBenchCase"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+
+
+</project>
diff --git a/trunk/maven.xml b/trunk/maven.xml
new file mode 100644
index 0000000..28641d4
--- /dev/null
+++ b/trunk/maven.xml
@@ -0,0 +1,19 @@
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<project default="java:jar"
+ xmlns:j="jelly:core">
+
+</project>
diff --git a/trunk/optional/bean-collections/.cvsignore b/trunk/optional/bean-collections/.cvsignore
new file mode 100644
index 0000000..3ad7dd4
--- /dev/null
+++ b/trunk/optional/bean-collections/.cvsignore
@@ -0,0 +1,4 @@
+*.log
+*.properties
+target
+dist
diff --git a/trunk/optional/bean-collections/LICENSE.txt b/trunk/optional/bean-collections/LICENSE.txt
new file mode 100644
index 0000000..d645695
--- /dev/null
+++ b/trunk/optional/bean-collections/LICENSE.txt
@@ -0,0 +1,202 @@
+
+ Apache License
+ Version 2.0, January 2004
+ http://www.apache.org/licenses/
+
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+ 1. Definitions.
+
+ "License" shall mean the terms and conditions for use, reproduction,
+ and distribution as defined by Sections 1 through 9 of this document.
+
+ "Licensor" shall mean the copyright owner or entity authorized by
+ the copyright owner that is granting the License.
+
+ "Legal Entity" shall mean the union of the acting entity and all
+ other entities that control, are controlled by, or are under common
+ control with that entity. For the purposes of this definition,
+ "control" means (i) the power, direct or indirect, to cause the
+ direction or management of such entity, whether by contract or
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
+ outstanding shares, or (iii) beneficial ownership of such entity.
+
+ "You" (or "Your") shall mean an individual or Legal Entity
+ exercising permissions granted by this License.
+
+ "Source" form shall mean the preferred form for making modifications,
+ including but not limited to software source code, documentation
+ source, and configuration files.
+
+ "Object" form shall mean any form resulting from mechanical
+ transformation or translation of a Source form, including but
+ not limited to compiled object code, generated documentation,
+ and conversions to other media types.
+
+ "Work" shall mean the work of authorship, whether in Source or
+ Object form, made available under the License, as indicated by a
+ copyright notice that is included in or attached to the work
+ (an example is provided in the Appendix below).
+
+ "Derivative Works" shall mean any work, whether in Source or Object
+ form, that is based on (or derived from) the Work and for which the
+ editorial revisions, annotations, elaborations, or other modifications
+ represent, as a whole, an original work of authorship. For the purposes
+ of this License, Derivative Works shall not include works that remain
+ separable from, or merely link (or bind by name) to the interfaces of,
+ the Work and Derivative Works thereof.
+
+ "Contribution" shall mean any work of authorship, including
+ the original version of the Work and any modifications or additions
+ to that Work or Derivative Works thereof, that is intentionally
+ submitted to Licensor for inclusion in the Work by the copyright owner
+ or by an individual or Legal Entity authorized to submit on behalf of
+ the copyright owner. For the purposes of this definition, "submitted"
+ means any form of electronic, verbal, or written communication sent
+ to the Licensor or its representatives, including but not limited to
+ communication on electronic mailing lists, source code control systems,
+ and issue tracking systems that are managed by, or on behalf of, the
+ Licensor for the purpose of discussing and improving the Work, but
+ excluding communication that is conspicuously marked or otherwise
+ designated in writing by the copyright owner as "Not a Contribution."
+
+ "Contributor" shall mean Licensor and any individual or Legal Entity
+ on behalf of whom a Contribution has been received by Licensor and
+ subsequently incorporated within the Work.
+
+ 2. Grant of Copyright License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ copyright license to reproduce, prepare Derivative Works of,
+ publicly display, publicly perform, sublicense, and distribute the
+ Work and such Derivative Works in Source or Object form.
+
+ 3. Grant of Patent License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ (except as stated in this section) patent license to make, have made,
+ use, offer to sell, sell, import, and otherwise transfer the Work,
+ where such license applies only to those patent claims licensable
+ by such Contributor that are necessarily infringed by their
+ Contribution(s) alone or by combination of their Contribution(s)
+ with the Work to which such Contribution(s) was submitted. If You
+ institute patent litigation against any entity (including a
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
+ or a Contribution incorporated within the Work constitutes direct
+ or contributory patent infringement, then any patent licenses
+ granted to You under this License for that Work shall terminate
+ as of the date such litigation is filed.
+
+ 4. Redistribution. You may reproduce and distribute copies of the
+ Work or Derivative Works thereof in any medium, with or without
+ modifications, and in Source or Object form, provided that You
+ meet the following conditions:
+
+ (a) You must give any other recipients of the Work or
+ Derivative Works a copy of this License; and
+
+ (b) You must cause any modified files to carry prominent notices
+ stating that You changed the files; and
+
+ (c) You must retain, in the Source form of any Derivative Works
+ that You distribute, all copyright, patent, trademark, and
+ attribution notices from the Source form of the Work,
+ excluding those notices that do not pertain to any part of
+ the Derivative Works; and
+
+ (d) If the Work includes a "NOTICE" text file as part of its
+ distribution, then any Derivative Works that You distribute must
+ include a readable copy of the attribution notices contained
+ within such NOTICE file, excluding those notices that do not
+ pertain to any part of the Derivative Works, in at least one
+ of the following places: within a NOTICE text file distributed
+ as part of the Derivative Works; within the Source form or
+ documentation, if provided along with the Derivative Works; or,
+ within a display generated by the Derivative Works, if and
+ wherever such third-party notices normally appear. The contents
+ of the NOTICE file are for informational purposes only and
+ do not modify the License. You may add Your own attribution
+ notices within Derivative Works that You distribute, alongside
+ or as an addendum to the NOTICE text from the Work, provided
+ that such additional attribution notices cannot be construed
+ as modifying the License.
+
+ You may add Your own copyright statement to Your modifications and
+ may provide additional or different license terms and conditions
+ for use, reproduction, or distribution of Your modifications, or
+ for any such Derivative Works as a whole, provided Your use,
+ reproduction, and distribution of the Work otherwise complies with
+ the conditions stated in this License.
+
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
+ any Contribution intentionally submitted for inclusion in the Work
+ by You to the Licensor shall be under the terms and conditions of
+ this License, without any additional terms or conditions.
+ Notwithstanding the above, nothing herein shall supersede or modify
+ the terms of any separate license agreement you may have executed
+ with Licensor regarding such Contributions.
+
+ 6. Trademarks. This License does not grant permission to use the trade
+ names, trademarks, service marks, or product names of the Licensor,
+ except as required for reasonable and customary use in describing the
+ origin of the Work and reproducing the content of the NOTICE file.
+
+ 7. Disclaimer of Warranty. Unless required by applicable law or
+ agreed to in writing, Licensor provides the Work (and each
+ Contributor provides its Contributions) on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+ implied, including, without limitation, any warranties or conditions
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+ PARTICULAR PURPOSE. You are solely responsible for determining the
+ appropriateness of using or redistributing the Work and assume any
+ risks associated with Your exercise of permissions under this License.
+
+ 8. Limitation of Liability. In no event and under no legal theory,
+ whether in tort (including negligence), contract, or otherwise,
+ unless required by applicable law (such as deliberate and grossly
+ negligent acts) or agreed to in writing, shall any Contributor be
+ liable to You for damages, including any direct, indirect, special,
+ incidental, or consequential damages of any character arising as a
+ result of this License or out of the use or inability to use the
+ Work (including but not limited to damages for loss of goodwill,
+ work stoppage, computer failure or malfunction, or any and all
+ other commercial damages or losses), even if such Contributor
+ has been advised of the possibility of such damages.
+
+ 9. Accepting Warranty or Additional Liability. While redistributing
+ the Work or Derivative Works thereof, You may choose to offer,
+ and charge a fee for, acceptance of support, warranty, indemnity,
+ or other liability obligations and/or rights consistent with this
+ License. However, in accepting such obligations, You may act only
+ on Your own behalf and on Your sole responsibility, not on behalf
+ of any other Contributor, and only if You agree to indemnify,
+ defend, and hold each Contributor harmless for any liability
+ incurred by, or claims asserted against, such Contributor by reason
+ of your accepting any such warranty or additional liability.
+
+ END OF TERMS AND CONDITIONS
+
+ APPENDIX: How to apply the Apache License to your work.
+
+ To apply the Apache License to your work, attach the following
+ boilerplate notice, with the fields enclosed by brackets "[]"
+ replaced with your own identifying information. (Don't include
+ the brackets!) The text should be enclosed in the appropriate
+ comment syntax for the file format. We also recommend that a
+ file or class name and description of purpose be included on the
+ same "printed page" as the copyright notice for easier
+ identification within third-party archives.
+
+ Copyright [yyyy] [name of copyright owner]
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
diff --git a/trunk/optional/bean-collections/build.properties.sample b/trunk/optional/bean-collections/build.properties.sample
new file mode 100755
index 0000000..6bd1b3a
--- /dev/null
+++ b/trunk/optional/bean-collections/build.properties.sample
@@ -0,0 +1,39 @@
+# Copyright 2001-2004 The Apache Software Foundation
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+# The home directory for the Commons collection classes distribution
+commons-collections.home= ../collections/dist
+
+# The pathname of the collections classes JAR file
+commons-collections.jar = ${commons-collections.home}/commons-collections.jar
+commons-collections-testframework.jar = ${commons-collections.home}/commons-collections-testframework-3.0-dev.jar
+
+# The home directory for the Commons logging classes distribution
+commons-logging.home = ../logging/dist
+
+# The pathname of the Commons Logging JAR file
+commons-logging.jar = ${commons-logging.home}/commons-logging.jar
+
+# The directory containing your binary distribution of JUnit,
+# version 3.7 or later
+junit.home = /usr/local/junit3.7
+
+# The pathname of the "junit.jar" JAR file
+junit.jar = ${junit.home}/junit.jar
+
+
+# Maven properties (for web site build)
+# Those committers using agents may like to use
+#maven.username=rdonkin
+#beanutils.cvs=lserver:rdonkin at cvs.apache.org:/home/cvs
\ No newline at end of file
diff --git a/trunk/optional/bean-collections/build.xml b/trunk/optional/bean-collections/build.xml
new file mode 100755
index 0000000..b5f19cb
--- /dev/null
+++ b/trunk/optional/bean-collections/build.xml
@@ -0,0 +1,301 @@
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<project name="Bean Utilities Bean Collections" default="dist" basedir=".">
+
+
+<!--
+ "Bean Utilities" component of the Jakarta Commons Subproject
+ $Id: build.xml,v 1.2.2.3 2004/07/27 21:59:14 rdonkin Exp $
+-->
+
+
+<!-- ========== Initialize Properties ===================================== -->
+
+
+ <property file="build.properties"/> <!-- Component local -->
+ <property file="../build.properties"/> <!-- Commons local -->
+ <property file="${user.home}/build.properties"/> <!-- User local -->
+
+<!-- ========== External Dependencies ===================================== -->
+
+ <!-- The home directory for the Commons collection classes distribution -->
+ <property name="commons-collections.home" value="../../../collections/dist"/>
+
+ <!-- The home directory for the Commons collection classes distribution -->
+ <property name="commons-logging.home" value="../logging/dist"/>
+
+ <!-- The directory containing your binary distribution of JUnit,
+ version 3.7 or later -->
+ <property name="junit.home" value="/usr/local/junit3.7"/>
+
+ <!-- The pathname of the beanutils core JAR file -->
+ <property name="commons-beanutils-core.jar" value="../../dist/commons-beanutils.jar"/>
+
+
+<!-- ========== Derived Values ============================================ -->
+
+ <!-- The pathname of the collections classes JAR file -->
+ <property name="commons-collections.jar" value="${commons-collections.home}/commons-collections-3.0-dev.jar"/>
+
+ <!-- The pathname of the collections testframework classes JAR file -->
+ <property name="commons-collections-testframework.jar" value="${commons-collections.home}/commons-collections-testframework-3.0-dev.jar"/>
+
+
+ <!-- The pathname of the logging classes JAR file -->
+ <property name="commons-logging.jar" value="${commons-logging.home}/commons-logging.jar"/>
+
+ <!-- The pathname of the "junit.jar" JAR file -->
+ <property name="junit.jar" value="${junit.home}/junit.jar"/>
+
+
+<!-- ========== Component Declarations ==================================== -->
+
+
+ <!-- The name of this component -->
+ <property name="component.name" value="beanutils"/>
+
+ <!-- The primary package name of this component -->
+ <property name="component.package" value="org.apache.commons.beanutils"/>
+
+ <!-- The title of this component -->
+ <property name="component.title" value="Bean Utilities Bean Collections"/>
+
+ <!-- The current version number of this component -->
+ <property name="component.version" value="1.7"/>
+
+ <!-- The base directory for compilation targets -->
+ <property name="build.home" value="target"/>
+
+ <!-- The base directory for component configuration files -->
+ <property name="conf.home" value="src/conf"/>
+
+ <!-- The base directory for distribution targets -->
+ <property name="dist.home" value="dist"/>
+
+ <!-- The base directory for component sources -->
+ <property name="source.home" value="src/java"/>
+
+ <!-- The base directory for unit test sources -->
+ <property name="test.home" value="src/test"/>
+
+
+<!-- ========== Compiler Defaults ========================================= -->
+
+
+ <!-- Should Java compilations set the 'debug' compiler option? -->
+ <property name="compile.debug" value="true"/>
+
+ <!-- Should Java compilations set the 'deprecation' compiler option? -->
+ <property name="compile.deprecation" value="false"/>
+
+ <!-- Should Java compilations set the 'optimize' compiler option? -->
+ <property name="compile.optimize" value="true"/>
+
+ <!-- Construct compile classpath -->
+ <path id="compile.classpath">
+ <pathelement location="${build.home}/classes"/>
+ <pathelement location="${commons-logging.jar}"/>
+ <pathelement location="${commons-collections.jar}"/>
+ <pathelement location="${commons-beanutils-core.jar}"/>
+ </path>
+
+
+<!-- ========== Test Execution Defaults =================================== -->
+
+
+ <!-- Construct unit test classpath -->
+ <path id="test.classpath">
+ <pathelement location="${build.home}/classes"/>
+ <pathelement location="${build.home}/tests"/>
+ <pathelement location="${commons-logging.jar}"/>
+ <pathelement location="${commons-collections.jar}"/>
+ <pathelement location="${commons-collections-testframework.jar}"/>
+ <pathelement location="${commons-beanutils-core.jar}"/>
+ <pathelement location="${junit.jar}"/>
+ </path>
+
+ <!-- Should all tests fail if one does? -->
+ <property name="test.failonerror" value="true"/>
+
+ <!-- The test runner to execute -->
+ <property name="test.runner" value="junit.textui.TestRunner"/>
+
+ <!-- The Commons Logger LogFactory implementation to use -->
+ <property name="test.factory"
+ value="org.apache.commons.logging.impl.LogFactoryImpl"/>
+
+ <!-- The Commons Logger Log implementation to use (standard factory) -->
+ <property name="test.log"
+ value="org.apache.commons.logging.impl.SimpleLog"/>
+
+ <!-- The Commons Logger SimpleLog level for testing -->
+ <property name="test.level" value="error"/>
+
+ <!-- Loop counter for microbenchmarks -->
+ <property name="bench.counter" value="100000"/>
+
+
+<!-- ========== Executable Targets ======================================== -->
+
+
+ <target name="init"
+ description="Initialize and evaluate conditionals">
+ <echo message="-------- ${component.name} ${component.version} --------"/>
+ <filter token="name" value="${component.name}"/>
+ <filter token="package" value="${component.package}"/>
+ <filter token="version" value="${component.version}"/>
+ </target>
+
+
+ <target name="prepare" depends="init"
+ description="Prepare build directory">
+ <mkdir dir="${build.home}"/>
+ <mkdir dir="${build.home}/classes"/>
+ <mkdir dir="${build.home}/conf"/>
+ <mkdir dir="${build.home}/tests"/>
+ </target>
+
+
+ <target name="static" depends="prepare"
+ description="Copy static files to build directory">
+ <tstamp/>
+ <copy todir="${build.home}/conf" filtering="on">
+ <fileset dir="${conf.home}" includes="*.MF"/>
+ </copy>
+ </target>
+
+
+ <target name="compile" depends="static"
+ description="Compile shareable components">
+ <javac srcdir="${source.home}"
+ destdir="${build.home}/classes"
+ debug="${compile.debug}"
+ deprecation="${compile.deprecation}"
+ optimize="${compile.optimize}">
+ <classpath refid="compile.classpath"/>
+ </javac>
+ <copy todir="${build.home}/classes" filtering="on">
+ <fileset dir="${source.home}" excludes="**/*.java"/>
+ </copy>
+ </target>
+
+
+ <target name="compile.tests" depends="compile"
+ description="Compile unit test cases">
+ <javac srcdir="${test.home}"
+ destdir="${build.home}/tests"
+ debug="${compile.debug}"
+ deprecation="${compile.deprecation}"
+ optimize="${compile.optimize}">
+ <classpath refid="test.classpath"/>
+ </javac>
+ <copy todir="${build.home}/tests" filtering="on">
+ <fileset dir="${test.home}" excludes="**/*.java"/>
+ </copy>
+ </target>
+
+
+ <target name="clean"
+ description="Clean build and distribution directories">
+ <delete dir="${build.home}"/>
+ <delete dir="${dist.home}"/>
+ </target>
+
+
+ <target name="all" depends="clean,compile"
+ description="Clean and compile all components"/>
+
+ <target name="dist" depends="clean,compile,javadoc,jar"
+ description="Clean, compile and jar">
+ <copy todir='../../dist' file= '${dist.home}/commons-beanutils-bean-collections.jar'/>
+ </target>
+
+ <target name="jar" depends="compile"
+ description="Create jar">
+ <mkdir dir="${dist.home}"/>
+ <mkdir dir="${build.home}/classes/META-INF"/>
+ <copy file="../../LICENSE.txt"
+ tofile="${build.home}/classes/META-INF/LICENSE.txt"/>
+ <copy file="../../NOTICE.txt"
+ tofile="${build.home}/classes/META-INF/NOTICE.txt"/>
+ <jar jarfile="${dist.home}/commons-beanutils-bean-collections.jar"
+ basedir="${build.home}/classes"
+ manifest="${build.home}/conf/MANIFEST.MF"/>
+ <jar jarfile="../../dist/commons-beanutils.jar"
+ basedir="${build.home}/classes"
+ update='true'/>
+ </target>
+
+
+ <target name="javadoc" depends="compile"
+ description="Create component Javadoc documentation">
+ <mkdir dir="${dist.home}"/>
+ <mkdir dir="${dist.home}/docs"/>
+ <mkdir dir="${dist.home}/docs/api"/>
+ <mkdir dir="../../dist/docs/"/>
+ <mkdir dir="../../dist/docs/bean-collections"/>
+ <javadoc sourcepath="${source.home}"
+ destdir="${dist.home}/docs/api"
+ packagenames="org.apache.commons.*"
+ author="true"
+ private="true"
+ version="true"
+ overview="src/java/overview.html"
+ doctitle="<h1>${component.title} (Version ${component.version})</h1>"
+ windowtitle="${component.title} (Version ${component.version})"
+ bottom="Copyright (c) 2001-2004 - Apache Software Foundation">
+ <classpath refid="compile.classpath"/>
+ </javadoc>
+ <copy
+ todir='../../dist/docs/bean-collections'>
+ <fileset dir='${dist.home}/docs/api'/>
+ </copy>
+ </target>
+
+ <target name="install-jar" depends="jar"
+ description="--> Installs jar file in ${lib.repo}">
+ <copy todir="${lib.repo}" filtering="no">
+ <fileset dir="${dist.home}">
+ <include name="commons-${component.name}.jar"/>
+ </fileset>
+ </copy>
+ </target>
+
+
+<!-- ========== Unit Test Targets ========================================= -->
+
+
+ <target name="test" depends="compile.tests,
+ test.bean.collections"
+ description="Run all unit test cases">
+ </target>
+
+ <target name="test.bean.collections" depends="compile.tests">
+ <echo message="Running bean collection utilities tests ..."/>
+ <java classname="${test.runner}" fork="yes"
+ failonerror="${test.failonerror}">
+ <sysproperty key="org.apache.commons.logging.LogFactory"
+ value="${test.factory}"/>
+ <sysproperty key="org.apache.commons.logging.Log"
+ value="${test.log}"/>
+ <sysproperty key="org.apache.commons.logging.simplelog.defaultlog"
+ value="${test.level}"/>
+ <arg value="org.apache.commons.beanutils.BeanCollectionsTestSuite"/>
+ <classpath refid="test.classpath"/>
+ </java>
+ </target>
+
+</project>
diff --git a/trunk/optional/bean-collections/project.xml b/trunk/optional/bean-collections/project.xml
new file mode 100644
index 0000000..583a140
--- /dev/null
+++ b/trunk/optional/bean-collections/project.xml
@@ -0,0 +1,50 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<project>
+ <extend>../../project.xml</extend>
+ <pomVersion>3</pomVersion>
+ <id>commons-beanutils-bean-collections</id>
+ <name>BeanUtils Bean Collections</name>
+ <currentVersion>1.7</currentVersion>
+ <inceptionYear>2000</inceptionYear>
+ <shortDescription>Commons BeanUtils Bean Collections</shortDescription>
+ <description>Extensions of commons collections focussing on collections of beans</description>
+
+ <siteDirectory>/www/jakarta.apache.org/commons/beanutils/bean-collections</siteDirectory>
+
+ <dependencies>
+ <dependency>
+ <id>commons-logging</id>
+ <version>1.0.3</version>
+ </dependency>
+ <dependency>
+ <id>commons-collections</id>
+ <version>3.0</version>
+ </dependency>
+ <dependency>
+ <id>commons-beanutils</id>
+ <version>1.6</version>
+ </dependency>
+ <!-- Test Only -->
+ <dependency>
+ <artifactId>commons-collections-testframework</artifactId>
+ <groupId>commons-collections</groupId>
+ <version>SNAPSHOT</version>
+ </dependency>
+
+ </dependencies>
+</project>
diff --git a/trunk/optional/bean-collections/src/conf/MANIFEST.MF b/trunk/optional/bean-collections/src/conf/MANIFEST.MF
new file mode 100644
index 0000000..d0e0431
--- /dev/null
+++ b/trunk/optional/bean-collections/src/conf/MANIFEST.MF
@@ -0,0 +1,7 @@
+Extension-Name: org.apache.commons.beanutils-bean-collections
+Specification-Title: Jakarta Commons Beanutils
+Specification-Vendor: Apache Software Foundation
+Specification-Version: 1.6
+Implementation-Title: org.apache.commons.beanutils-bean-collections
+Implementation-Vendor: Apache Software Foundation
+Implementation-Version: 1.6
diff --git a/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanComparator.java b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanComparator.java
new file mode 100644
index 0000000..74ec49a
--- /dev/null
+++ b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanComparator.java
@@ -0,0 +1,191 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import java.io.Serializable;
+import java.util.Comparator;
+import org.apache.commons.beanutils.PropertyUtils;
+import org.apache.commons.collections.comparators.ComparableComparator;
+
+/**
+ * <p>
+ * This comparator compares two beans by the specified bean property.
+ * It is also possible to compare beans based on nested, indexed,
+ * combined, mapped bean properties. Please see the {@link PropertyUtilsBean}
+ * documentation for all property name possibilities.
+ *
+ * </p><p>
+ * <strong>Note:</strong> The BeanComparator passes the values of the specified
+ * bean property to a ComparableComparator, if no comparator is
+ * specified in the constructor. If you are comparing two beans based
+ * on a property that could contain "null" values, a suitable <code>Comparator</code>
+ * or <code>ComparatorChain</code> should be supplied in the constructor.
+ * </p>
+ *
+ * @author <a href"mailto:epugh at upstate.com">Eric Pugh</a>
+ * @author Tim O'Brien
+ */
+public class BeanComparator implements Comparator, Serializable {
+
+ private String property;
+ private Comparator comparator;
+
+ /**
+ * <p>Constructs a Bean Comparator without a property set.
+ * </p><p>
+ * <strong>Note</strong> that this is intended to be used
+ * only in bean-centric environments.
+ * </p><p>
+ * Until {@link #setProperty} is called with a non-null value.
+ * this comparator will compare the Objects only.
+ * </p>
+ */
+ public BeanComparator() {
+ this( null );
+ }
+
+ /**
+ * <p>Constructs a property-based comparator for beans.
+ * This compares two beans by the property
+ * specified in the property parameter. This constructor creates
+ * a <code>BeanComparator</code> that uses a <code>ComparableComparator</code>
+ * to compare the property values.
+ * </p>
+ *
+ * <p>Passing "null" to this constructor will cause the BeanComparator
+ * to compare objects based on natural order, that is
+ * <code>java.lang.Comparable</code>.
+ * </p>
+ *
+ * @param property String Name of a bean property, which may contain the
+ * name of a simple, nested, indexed, mapped, or combined
+ * property. See {@link PropertyUtilsBean} for property query language syntax.
+ * If the property passed in is null then the actual objects will be compared
+ */
+ public BeanComparator( String property ) {
+ this( property, ComparableComparator.getInstance() );
+ }
+
+ /**
+ * Constructs a property-based comparator for beans.
+ * This constructor creates
+ * a BeanComparator that uses the supplied Comparator to compare
+ * the property values.
+ *
+ * @param property Name of a bean property, can contain the name
+ * of a simple, nested, indexed, mapped, or combined
+ * property. See {@link PropertyUtilsBean} for property query language
+ * syntax.
+ * @param comparator BeanComparator will pass the values of the
+ * specified bean property to this Comparator.
+ * If your bean property is not a comparable or
+ * contains null values, a suitable comparator
+ * may be supplied in this constructor.
+ */
+ public BeanComparator( String property, Comparator comparator ) {
+ setProperty( property );
+ this.comparator = comparator;
+ }
+
+ /**
+ * Sets the method to be called to compare two JavaBeans
+ *
+ * @param property String method name to call to compare
+ * If the property passed in is null then the actual objects will be compared
+ */
+ public void setProperty( String property ) {
+ this.property = property;
+ }
+
+
+ /**
+ * Gets the property attribute of the BeanComparator
+ *
+ * @return String method name to call to compare.
+ * A null value indicates that the actual objects will be compared
+ */
+ public String getProperty() {
+ return property;
+ }
+
+
+ /**
+ * Gets the Comparator being used to compare beans.
+ */
+ public Comparator getComparator() {
+ return comparator;
+ }
+
+
+ /**
+ * Compare two JavaBeans by their shared property.
+ * If {@link #getProperty} is null then the actual objects will be compared.
+ *
+ * @param o1 Object The first bean to get data from to compare against
+ * @param o2 Object The second bean to get data from to compare
+ * @return int negative or positive based on order
+ */
+ public int compare( Object o1, Object o2 ) {
+
+ if ( property == null ) {
+ // compare the actual objects
+ return comparator.compare( o1, o2 );
+ }
+
+ try {
+ Object value1 = PropertyUtils.getProperty( o1, property );
+ Object value2 = PropertyUtils.getProperty( o2, property );
+ return comparator.compare( value1, value2 );
+ }
+ catch ( Exception e ) {
+ throw new ClassCastException( e.toString() );
+ }
+ }
+
+ /**
+ * Two <code>BeanComparator</code>'s are equals if and only if
+ * the wrapped comparators and the property names to be compared
+ * are equal.
+ */
+ public boolean equals(Object o) {
+ if (this == o) return true;
+ if (!(o instanceof BeanComparator)) return false;
+
+ final BeanComparator beanComparator = (BeanComparator) o;
+
+ if (!comparator.equals(beanComparator.comparator)) return false;
+ if (property != null)
+ {
+ if (!property.equals(beanComparator.property)) return false;
+ }
+ else
+ {
+ return (beanComparator.property == null);
+ }
+
+ return true;
+ }
+
+ /**
+ * Hashcode compatible with equals.
+ */
+ public int hashCode() {
+ int result;
+ result = comparator.hashCode();
+ return result;
+ }
+}
diff --git a/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanMap.java b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanMap.java
new file mode 100644
index 0000000..74ea05a
--- /dev/null
+++ b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanMap.java
@@ -0,0 +1,803 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.beanutils;
+
+import java.beans.BeanInfo;
+import java.beans.IntrospectionException;
+import java.beans.Introspector;
+import java.beans.PropertyDescriptor;
+import java.lang.reflect.Constructor;
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Method;
+import java.util.AbstractMap;
+import java.util.AbstractSet;
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.Set;
+
+import org.apache.commons.collections.list.UnmodifiableList;
+import org.apache.commons.collections.keyvalue.AbstractMapEntry;
+import org.apache.commons.collections.set.UnmodifiableSet;
+import org.apache.commons.collections.Transformer;
+
+/**
+ * An implementation of Map for JavaBeans which uses introspection to
+ * get and put properties in the bean.
+ * <p>
+ * If an exception occurs during attempts to get or set a property then the
+ * property is considered non existent in the Map
+ *
+ * @version $Revision: 1.2.2.2 $ $Date: 2004/06/22 21:07:02 $
+ *
+ * @author James Strachan
+ * @author Stephen Colebourne
+ */
+public class BeanMap extends AbstractMap implements Cloneable {
+
+ private transient Object bean;
+
+ private transient HashMap readMethods = new HashMap();
+ private transient HashMap writeMethods = new HashMap();
+ private transient HashMap types = new HashMap();
+
+ /**
+ * An empty array. Used to invoke accessors via reflection.
+ */
+ public static final Object[] NULL_ARGUMENTS = {};
+
+ /**
+ * Maps primitive Class types to transformers. The transformer
+ * transform strings into the appropriate primitive wrapper.
+ */
+ public static HashMap defaultTransformers = new HashMap();
+
+ static {
+ defaultTransformers.put(
+ Boolean.TYPE,
+ new Transformer() {
+ public Object transform( Object input ) {
+ return Boolean.valueOf( input.toString() );
+ }
+ }
+ );
+ defaultTransformers.put(
+ Character.TYPE,
+ new Transformer() {
+ public Object transform( Object input ) {
+ return new Character( input.toString().charAt( 0 ) );
+ }
+ }
+ );
+ defaultTransformers.put(
+ Byte.TYPE,
+ new Transformer() {
+ public Object transform( Object input ) {
+ return Byte.valueOf( input.toString() );
+ }
+ }
+ );
+ defaultTransformers.put(
+ Short.TYPE,
+ new Transformer() {
+ public Object transform( Object input ) {
+ return Short.valueOf( input.toString() );
+ }
+ }
+ );
+ defaultTransformers.put(
+ Integer.TYPE,
+ new Transformer() {
+ public Object transform( Object input ) {
+ return Integer.valueOf( input.toString() );
+ }
+ }
+ );
+ defaultTransformers.put(
+ Long.TYPE,
+ new Transformer() {
+ public Object transform( Object input ) {
+ return Long.valueOf( input.toString() );
+ }
+ }
+ );
+ defaultTransformers.put(
+ Float.TYPE,
+ new Transformer() {
+ public Object transform( Object input ) {
+ return Float.valueOf( input.toString() );
+ }
+ }
+ );
+ defaultTransformers.put(
+ Double.TYPE,
+ new Transformer() {
+ public Object transform( Object input ) {
+ return Double.valueOf( input.toString() );
+ }
+ }
+ );
+ }
+
+
+ // Constructors
+ //-------------------------------------------------------------------------
+
+ /**
+ * Constructs a new empty <code>BeanMap</code>.
+ */
+ public BeanMap() {
+ }
+
+ /**
+ * Constructs a new <code>BeanMap</code> that operates on the
+ * specified bean. If the given bean is <code>null</code>, then
+ * this map will be empty.
+ *
+ * @param bean the bean for this map to operate on
+ */
+ public BeanMap(Object bean) {
+ this.bean = bean;
+ initialise();
+ }
+
+ // Map interface
+ //-------------------------------------------------------------------------
+
+ /**
+ * Renders a string representation of this object.
+ * @return a <code>String</code> representation of this object
+ */
+ public String toString() {
+ return "BeanMap<" + String.valueOf(bean) + ">";
+ }
+
+ /**
+ * Clone this bean map using the following process:
+ *
+ * <ul>
+ * <li>If there is no underlying bean, return a cloned BeanMap without a
+ * bean.
+ *
+ * <li>Since there is an underlying bean, try to instantiate a new bean of
+ * the same type using Class.newInstance().
+ *
+ * <li>If the instantiation fails, throw a CloneNotSupportedException
+ *
+ * <li>Clone the bean map and set the newly instantiated bean as the
+ * underlying bean for the bean map.
+ *
+ * <li>Copy each property that is both readable and writable from the
+ * existing object to a cloned bean map.
+ *
+ * <li>If anything fails along the way, throw a
+ * CloneNotSupportedException.
+ *
+ * <ul>
+ */
+ public Object clone() throws CloneNotSupportedException {
+ BeanMap newMap = (BeanMap)super.clone();
+
+ if(bean == null) {
+ // no bean, just an empty bean map at the moment. return a newly
+ // cloned and empty bean map.
+ return newMap;
+ }
+
+ Object newBean = null;
+ Class beanClass = null;
+ try {
+ beanClass = bean.getClass();
+ newBean = beanClass.newInstance();
+ } catch (Exception e) {
+ // unable to instantiate
+ throw new CloneNotSupportedException
+ ("Unable to instantiate the underlying bean \"" +
+ beanClass.getName() + "\": " + e);
+ }
+
+ try {
+ newMap.setBean(newBean);
+ } catch (Exception exception) {
+ throw new CloneNotSupportedException
+ ("Unable to set bean in the cloned bean map: " +
+ exception);
+ }
+
+ try {
+ // copy only properties that are readable and writable. If its
+ // not readable, we can't get the value from the old map. If
+ // its not writable, we can't write a value into the new map.
+ Iterator readableKeys = readMethods.keySet().iterator();
+ while(readableKeys.hasNext()) {
+ Object key = readableKeys.next();
+ if(getWriteMethod(key) != null) {
+ newMap.put(key, get(key));
+ }
+ }
+ } catch (Exception exception) {
+ throw new CloneNotSupportedException
+ ("Unable to copy bean values to cloned bean map: " +
+ exception);
+ }
+
+ return newMap;
+ }
+
+ /**
+ * Puts all of the writable properties from the given BeanMap into this
+ * BeanMap. Read-only and Write-only properties will be ignored.
+ *
+ * @param map the BeanMap whose properties to put
+ */
+ public void putAllWriteable(BeanMap map) {
+ Iterator readableKeys = map.readMethods.keySet().iterator();
+ while (readableKeys.hasNext()) {
+ Object key = readableKeys.next();
+ if (getWriteMethod(key) != null) {
+ this.put(key, map.get(key));
+ }
+ }
+ }
+
+
+ /**
+ * This method reinitializes the bean map to have default values for the
+ * bean's properties. This is accomplished by constructing a new instance
+ * of the bean which the map uses as its underlying data source. This
+ * behavior for <code>clear()</code> differs from the Map contract in that
+ * the mappings are not actually removed from the map (the mappings for a
+ * BeanMap are fixed).
+ */
+ public void clear() {
+ if(bean == null) return;
+
+ Class beanClass = null;
+ try {
+ beanClass = bean.getClass();
+ bean = beanClass.newInstance();
+ }
+ catch (Exception e) {
+ throw new UnsupportedOperationException( "Could not create new instance of class: " + beanClass );
+ }
+ }
+
+ /**
+ * Returns true if the bean defines a property with the given name.
+ * <p>
+ * The given name must be a <code>String</code>; if not, this method
+ * returns false. This method will also return false if the bean
+ * does not define a property with that name.
+ * <p>
+ * Write-only properties will not be matched as the test operates against
+ * property read methods.
+ *
+ * @param name the name of the property to check
+ * @return false if the given name is null or is not a <code>String</code>;
+ * false if the bean does not define a property with that name; or
+ * true if the bean does define a property with that name
+ */
+ public boolean containsKey(Object name) {
+ Method method = getReadMethod(name);
+ return method != null;
+ }
+
+ /**
+ * Returns true if the bean defines a property whose current value is
+ * the given object.
+ *
+ * @param value the value to check
+ * @return false true if the bean has at least one property whose
+ * current value is that object, false otherwise
+ */
+ public boolean containsValue(Object value) {
+ // use default implementation
+ return super.containsValue(value);
+ }
+
+ /**
+ * Returns the value of the bean's property with the given name.
+ * <p>
+ * The given name must be a {@link String} and must not be
+ * null; otherwise, this method returns <code>null</code>.
+ * If the bean defines a property with the given name, the value of
+ * that property is returned. Otherwise, <code>null</code> is
+ * returned.
+ * <p>
+ * Write-only properties will not be matched as the test operates against
+ * property read methods.
+ *
+ * @param name the name of the property whose value to return
+ * @return the value of the property with that name
+ */
+ public Object get(Object name) {
+ if ( bean != null ) {
+ Method method = getReadMethod( name );
+ if ( method != null ) {
+ try {
+ return method.invoke( bean, NULL_ARGUMENTS );
+ }
+ catch ( IllegalAccessException e ) {
+ logWarn( e );
+ }
+ catch ( IllegalArgumentException e ) {
+ logWarn( e );
+ }
+ catch ( InvocationTargetException e ) {
+ logWarn( e );
+ }
+ catch ( NullPointerException e ) {
+ logWarn( e );
+ }
+ }
+ }
+ return null;
+ }
+
+ /**
+ * Sets the bean property with the given name to the given value.
+ *
+ * @param name the name of the property to set
+ * @param value the value to set that property to
+ * @return the previous value of that property
+ * @throws IllegalArgumentException if the given name is null;
+ * if the given name is not a {@link String}; if the bean doesn't
+ * define a property with that name; or if the bean property with
+ * that name is read-only
+ */
+ public Object put(Object name, Object value) throws IllegalArgumentException, ClassCastException {
+ if ( bean != null ) {
+ Object oldValue = get( name );
+ Method method = getWriteMethod( name );
+ if ( method == null ) {
+ throw new IllegalArgumentException( "The bean of type: "+ bean.getClass().getName() + " has no property called: " + name );
+ }
+ try {
+ Object[] arguments = createWriteMethodArguments( method, value );
+ method.invoke( bean, arguments );
+
+ Object newValue = get( name );
+ firePropertyChange( name, oldValue, newValue );
+ }
+ catch ( InvocationTargetException e ) {
+ logInfo( e );
+ throw new IllegalArgumentException( e.getMessage() );
+ }
+ catch ( IllegalAccessException e ) {
+ logInfo( e );
+ throw new IllegalArgumentException( e.getMessage() );
+ }
+ return oldValue;
+ }
+ return null;
+ }
+
+ /**
+ * Returns the number of properties defined by the bean.
+ *
+ * @return the number of properties defined by the bean
+ */
+ public int size() {
+ return readMethods.size();
+ }
+
+
+ /**
+ * Get the keys for this BeanMap.
+ * <p>
+ * Write-only properties are <b>not</b> included in the returned set of
+ * property names, although it is possible to set their value and to get
+ * their type.
+ *
+ * @return BeanMap keys. The Set returned by this method is not
+ * modifiable.
+ */
+ public Set keySet() {
+ return UnmodifiableSet.decorate(readMethods.keySet());
+ }
+
+ /**
+ * Gets a Set of MapEntry objects that are the mappings for this BeanMap.
+ * <p>
+ * Each MapEntry can be set but not removed.
+ *
+ * @return the unmodifiable set of mappings
+ */
+ public Set entrySet() {
+ return UnmodifiableSet.decorate(new AbstractSet() {
+ public Iterator iterator() {
+ return entryIterator();
+ }
+ public int size() {
+ return BeanMap.this.readMethods.size();
+ }
+ });
+ }
+
+ /**
+ * Returns the values for the BeanMap.
+ *
+ * @return values for the BeanMap. The returned collection is not
+ * modifiable.
+ */
+ public Collection values() {
+ ArrayList answer = new ArrayList( readMethods.size() );
+ for ( Iterator iter = valueIterator(); iter.hasNext(); ) {
+ answer.add( iter.next() );
+ }
+ return UnmodifiableList.decorate(answer);
+ }
+
+
+ // Helper methods
+ //-------------------------------------------------------------------------
+
+ /**
+ * Returns the type of the property with the given name.
+ *
+ * @param name the name of the property
+ * @return the type of the property, or <code>null</code> if no such
+ * property exists
+ */
+ public Class getType(String name) {
+ return (Class) types.get( name );
+ }
+
+ /**
+ * Convenience method for getting an iterator over the keys.
+ * <p>
+ * Write-only properties will not be returned in the iterator.
+ *
+ * @return an iterator over the keys
+ */
+ public Iterator keyIterator() {
+ return readMethods.keySet().iterator();
+ }
+
+ /**
+ * Convenience method for getting an iterator over the values.
+ *
+ * @return an iterator over the values
+ */
+ public Iterator valueIterator() {
+ final Iterator iter = keyIterator();
+ return new Iterator() {
+ public boolean hasNext() {
+ return iter.hasNext();
+ }
+ public Object next() {
+ Object key = iter.next();
+ return get(key);
+ }
+ public void remove() {
+ throw new UnsupportedOperationException( "remove() not supported for BeanMap" );
+ }
+ };
+ }
+
+ /**
+ * Convenience method for getting an iterator over the entries.
+ *
+ * @return an iterator over the entries
+ */
+ public Iterator entryIterator() {
+ final Iterator iter = keyIterator();
+ return new Iterator() {
+ public boolean hasNext() {
+ return iter.hasNext();
+ }
+ public Object next() {
+ Object key = iter.next();
+ Object value = get(key);
+ return new Entry( BeanMap.this, key, value );
+ }
+ public void remove() {
+ throw new UnsupportedOperationException( "remove() not supported for BeanMap" );
+ }
+ };
+ }
+
+
+ // Properties
+ //-------------------------------------------------------------------------
+
+ /**
+ * Returns the bean currently being operated on. The return value may
+ * be null if this map is empty.
+ *
+ * @return the bean being operated on by this map
+ */
+ public Object getBean() {
+ return bean;
+ }
+
+ /**
+ * Sets the bean to be operated on by this map. The given value may
+ * be null, in which case this map will be empty.
+ *
+ * @param newBean the new bean to operate on
+ */
+ public void setBean( Object newBean ) {
+ bean = newBean;
+ reinitialise();
+ }
+
+ /**
+ * Returns the accessor for the property with the given name.
+ *
+ * @param name the name of the property
+ * @return the accessor method for the property, or null
+ */
+ public Method getReadMethod(String name) {
+ return (Method) readMethods.get(name);
+ }
+
+ /**
+ * Returns the mutator for the property with the given name.
+ *
+ * @param name the name of the property
+ * @return the mutator method for the property, or null
+ */
+ public Method getWriteMethod(String name) {
+ return (Method) writeMethods.get(name);
+ }
+
+
+ // Implementation methods
+ //-------------------------------------------------------------------------
+
+ /**
+ * Returns the accessor for the property with the given name.
+ *
+ * @param name the name of the property
+ * @return null if the name is null; null if the name is not a
+ * {@link String}; null if no such property exists; or the accessor
+ * method for that property
+ */
+ protected Method getReadMethod( Object name ) {
+ return (Method) readMethods.get( name );
+ }
+
+ /**
+ * Returns the mutator for the property with the given name.
+ *
+ * @param name the name of the
+ * @return null if the name is null; null if the name is not a
+ * {@link String}; null if no such property exists; null if the
+ * property is read-only; or the mutator method for that property
+ */
+ protected Method getWriteMethod( Object name ) {
+ return (Method) writeMethods.get( name );
+ }
+
+ /**
+ * Reinitializes this bean. Called during {@link #setBean(Object)}.
+ * Does introspection to find properties.
+ */
+ protected void reinitialise() {
+ readMethods.clear();
+ writeMethods.clear();
+ types.clear();
+ initialise();
+ }
+
+ private void initialise() {
+ if(getBean() == null) return;
+
+ Class beanClass = getBean().getClass();
+ try {
+ //BeanInfo beanInfo = Introspector.getBeanInfo( bean, null );
+ BeanInfo beanInfo = Introspector.getBeanInfo( beanClass );
+ PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();
+ if ( propertyDescriptors != null ) {
+ for ( int i = 0; i < propertyDescriptors.length; i++ ) {
+ PropertyDescriptor propertyDescriptor = propertyDescriptors[i];
+ if ( propertyDescriptor != null ) {
+ String name = propertyDescriptor.getName();
+ Method readMethod = propertyDescriptor.getReadMethod();
+ Method writeMethod = propertyDescriptor.getWriteMethod();
+ Class aType = propertyDescriptor.getPropertyType();
+
+ if ( readMethod != null ) {
+ readMethods.put( name, readMethod );
+ }
+ if ( writeMethods != null ) {
+ writeMethods.put( name, writeMethod );
+ }
+ types.put( name, aType );
+ }
+ }
+ }
+ }
+ catch ( IntrospectionException e ) {
+ logWarn( e );
+ }
+ }
+
+ /**
+ * Called during a successful {@link #put(Object,Object)} operation.
+ * Default implementation does nothing. Override to be notified of
+ * property changes in the bean caused by this map.
+ *
+ * @param key the name of the property that changed
+ * @param oldValue the old value for that property
+ * @param newValue the new value for that property
+ */
+ protected void firePropertyChange( Object key, Object oldValue, Object newValue ) {
+ }
+
+ // Implementation classes
+ //-------------------------------------------------------------------------
+
+ /**
+ * Map entry used by {@link BeanMap}.
+ */
+ protected static class Entry extends AbstractMapEntry {
+ private BeanMap owner;
+
+ /**
+ * Constructs a new <code>Entry</code>.
+ *
+ * @param owner the BeanMap this entry belongs to
+ * @param key the key for this entry
+ * @param value the value for this entry
+ */
+ protected Entry( BeanMap owner, Object key, Object value ) {
+ super( key, value );
+ this.owner = owner;
+ }
+
+ /**
+ * Sets the value.
+ *
+ * @param value the new value for the entry
+ * @return the old value for the entry
+ */
+ public Object setValue(Object value) {
+ Object key = getKey();
+ Object oldValue = owner.get( key );
+
+ owner.put( key, value );
+ Object newValue = owner.get( key );
+ super.setValue( newValue );
+ return oldValue;
+ }
+ }
+
+ /**
+ * Creates an array of parameters to pass to the given mutator method.
+ * If the given object is not the right type to pass to the method
+ * directly, it will be converted using {@link #convertType(Class,Object)}.
+ *
+ * @param method the mutator method
+ * @param value the value to pass to the mutator method
+ * @return an array containing one object that is either the given value
+ * or a transformed value
+ * @throws IllegalAccessException if {@link #convertType(Class,Object)}
+ * raises it
+ * @throws IllegalArgumentException if any other exception is raised
+ * by {@link #convertType(Class,Object)}
+ */
+ protected Object[] createWriteMethodArguments( Method method, Object value ) throws IllegalAccessException, ClassCastException {
+ try {
+ if ( value != null ) {
+ Class[] types = method.getParameterTypes();
+ if ( types != null && types.length > 0 ) {
+ Class paramType = types[0];
+ if ( ! paramType.isAssignableFrom( value.getClass() ) ) {
+ value = convertType( paramType, value );
+ }
+ }
+ }
+ Object[] answer = { value };
+ return answer;
+ }
+ catch ( InvocationTargetException e ) {
+ logInfo( e );
+ throw new IllegalArgumentException( e.getMessage() );
+ }
+ catch ( InstantiationException e ) {
+ logInfo( e );
+ throw new IllegalArgumentException( e.getMessage() );
+ }
+ }
+
+ /**
+ * Converts the given value to the given type. First, reflection is
+ * is used to find a public constructor declared by the given class
+ * that takes one argument, which must be the precise type of the
+ * given value. If such a constructor is found, a new object is
+ * created by passing the given value to that constructor, and the
+ * newly constructed object is returned.<P>
+ *
+ * If no such constructor exists, and the given type is a primitive
+ * type, then the given value is converted to a string using its
+ * {@link Object#toString() toString()} method, and that string is
+ * parsed into the correct primitive type using, for instance,
+ * {@link Integer#valueOf(String)} to convert the string into an
+ * <code>int</code>.<P>
+ *
+ * If no special constructor exists and the given type is not a
+ * primitive type, this method returns the original value.
+ *
+ * @param newType the type to convert the value to
+ * @param value the value to convert
+ * @return the converted value
+ * @throws NumberFormatException if newType is a primitive type, and
+ * the string representation of the given value cannot be converted
+ * to that type
+ * @throws InstantiationException if the constructor found with
+ * reflection raises it
+ * @throws InvocationTargetException if the constructor found with
+ * reflection raises it
+ * @throws IllegalAccessException never
+ * @throws IllegalArgumentException never
+ */
+ protected Object convertType( Class newType, Object value )
+ throws InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
+
+ // try call constructor
+ Class[] types = { value.getClass() };
+ try {
+ Constructor constructor = newType.getConstructor( types );
+ Object[] arguments = { value };
+ return constructor.newInstance( arguments );
+ }
+ catch ( NoSuchMethodException e ) {
+ // try using the transformers
+ Transformer transformer = getTypeTransformer( newType );
+ if ( transformer != null ) {
+ return transformer.transform( value );
+ }
+ return value;
+ }
+ }
+
+ /**
+ * Returns a transformer for the given primitive type.
+ *
+ * @param aType the primitive type whose transformer to return
+ * @return a transformer that will convert strings into that type,
+ * or null if the given type is not a primitive type
+ */
+ protected Transformer getTypeTransformer( Class aType ) {
+ return (Transformer) defaultTransformers.get( aType );
+ }
+
+ /**
+ * Logs the given exception to <code>System.out</code>. Used to display
+ * warnings while accessing/mutating the bean.
+ *
+ * @param ex the exception to log
+ */
+ protected void logInfo(Exception ex) {
+ // Deliberately do not use LOG4J or Commons Logging to avoid dependencies
+ System.out.println( "INFO: Exception: " + ex );
+ }
+
+ /**
+ * Logs the given exception to <code>System.err</code>. Used to display
+ * errors while accessing/mutating the bean.
+ *
+ * @param ex the exception to log
+ */
+ protected void logWarn(Exception ex) {
+ // Deliberately do not use LOG4J or Commons Logging to avoid dependencies
+ System.out.println( "WARN: Exception: " + ex );
+ ex.printStackTrace();
+ }
+}
diff --git a/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanPredicate.java b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanPredicate.java
new file mode 100644
index 0000000..f09d4b0
--- /dev/null
+++ b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanPredicate.java
@@ -0,0 +1,121 @@
+/*
+ * Copyright 2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import org.apache.commons.collections.Predicate;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.lang.reflect.InvocationTargetException;
+
+/**
+ * <p>Predicate implementation that applies the given <code>Predicate</code>
+ * to the result of calling the given property getter.
+ * </p>
+ */
+public class BeanPredicate implements Predicate {
+
+ private final Log log = LogFactory.getLog(this.getClass());
+
+ /** Name of the property whose value will be predicated */
+ private String propertyName;
+ /** <code>Predicate</code> to be applied to the property value */
+ private Predicate predicate;
+
+ /**
+ * Constructs a <code>BeanPredicate</code> that applies the given
+ * <code>Predicate</code> to the named property value.
+ * @param propertyName the name of the property whose value is to be predicated,
+ * not null
+ * @param predicate the <code>Predicate</code> to be applied,
+ * not null
+ */
+ public BeanPredicate(String propertyName, Predicate predicate) {
+ this.propertyName = propertyName;
+ this.predicate = predicate;
+ }
+
+ /**
+ * Evaluates the given object by applying the {@link #getPredicate()}
+ * to a property value named by {@link #getPropertyName()}.
+ * @throws IllegalAccessException when the property cannot be evaluated
+ */
+ public boolean evaluate(Object object) {
+
+ boolean evaluation = false;
+
+ try {
+ Object propValue = PropertyUtils.getProperty( object, propertyName );
+ evaluation = predicate.evaluate(propValue);
+ } catch (IllegalArgumentException e) {
+ final String errorMsg = "Problem during evaluation.";
+ log.error("ERROR: " + errorMsg, e);
+ throw e;
+ } catch (IllegalAccessException e) {
+ final String errorMsg = "Unable to access the property provided.";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ } catch (InvocationTargetException e) {
+ final String errorMsg = "Exception occurred in property's getter";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ } catch (NoSuchMethodException e) {
+ final String errorMsg = "Property not found.";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ }
+
+ return evaluation;
+ }
+
+ /**
+ * Gets the name of the property whose value is to be predicated.
+ * in the evaluation.
+ * @return the property name, not null
+ */
+ public String getPropertyName() {
+ return propertyName;
+ }
+
+ /**
+ * Sets the name of the property whose value is to be predicated.
+ * @param propertyName the name of the property whose value is to be predicated,
+ * not null
+ */
+ public void setPropertyName(String propertyName) {
+ this.propertyName = propertyName;
+ }
+
+ /**
+ * Gets the <code>Predicate</code> to be applied to the value of the named property
+ * during {@link #evaluate}.
+ * @return <code>Predicate</code>, not null
+ */
+ public Predicate getPredicate() {
+ return predicate;
+ }
+
+ /**
+ * Sets the <code>Predicate</code> to be applied to the value of the named property
+ * during {@link evaluate}.
+ * @param predicate <code>Predicate</code>, not null
+ */
+ public void setPredicate(Predicate predicate) {
+ this.predicate = predicate;
+ }
+
+}
diff --git a/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanPropertyValueChangeClosure.java b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanPropertyValueChangeClosure.java
new file mode 100644
index 0000000..4d3e4ec
--- /dev/null
+++ b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanPropertyValueChangeClosure.java
@@ -0,0 +1,224 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import org.apache.commons.collections.Closure;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.lang.reflect.InvocationTargetException;
+
+
+/**
+ * <p><code>Closure</code> that sets a property.</p>
+ * <p>
+ * An implementation of <code>org.apache.commons.collections.Closure</code> that updates
+ * a specified property on the object provided with a specified value.
+ * The <code>BeanPropertyValueChangeClosure</code> constructor takes two parameters which determine
+ * what property will be updated and with what value.
+ * <dl>
+ * <dt>
+ * <b><code><pre>public BeanPropertyValueChangeClosure( String propertyName, Object propertyValue )</pre></code></b>
+ * </dt>
+ * <dd>
+ * Will create a <code>Closure</code> that will update an object by setting the property
+ * specified by <code>propertyName</code> to the value specified by <code>propertyValue</code>.
+ * </dd>
+ * </dl>
+ *
+ * <p/>
+ * <strong>Note:</strong> Property names can be a simple, nested, indexed, or mapped property as defined by
+ * <code>org.apache.commons.beanutils.PropertyUtils</code>. If any object in the property path
+ * specified by <code>propertyName</code> is <code>null</code> then the outcome is based on the
+ * value of the <code>ignoreNull</code> attribute.
+ *
+ * <p/>
+ * A typical usage might look like:
+ * <code><pre>
+ * // create the closure
+ * BeanPropertyValueChangeClosure closure =
+ * new BeanPropertyValueChangeClosure( "activeEmployee", Boolean.TRUE );
+ *
+ * // update the Collection
+ * CollectionUtils.forAllDo( peopleCollection, closure );
+ * </pre></code>
+ * <p/>
+ *
+ * This would take a <code>Collection</code> of person objects and update the
+ * <code>activeEmployee</code> property of each object in the <code>Collection</code> to
+ * <code>true</code>. Assuming...
+ * <ul>
+ * <li>
+ * The top level object in the <code>peopleCollection</code> is an object which represents a
+ * person.
+ * </li>
+ * <li>
+ * The person object has a <code>setActiveEmployee( boolean )</code> method which updates
+ * the value for the object's <code>activeEmployee</code> property.
+ * </li>
+ * </ul>
+ *
+ * @author Norm Deane
+ * @see org.apache.commons.beanutils.PropertyUtils
+ * @see org.apache.commons.collections.Closure
+ */
+public class BeanPropertyValueChangeClosure implements Closure {
+
+ /** For logging. */
+ private final Log log = LogFactory.getLog(this.getClass());
+
+ /**
+ * The name of the property which will be updated when this <code>Closure</code> executes.
+ */
+ private String propertyName;
+
+ /**
+ * The value that the property specified by <code>propertyName</code>
+ * will be updated to when this <code>Closure</code> executes.
+ */
+ private Object propertyValue;
+
+ /**
+ * Determines whether <code>null</code> objects in the property path will genenerate an
+ * <code>IllegalArgumentException</code> or not. If set to <code>true</code> then if any objects
+ * in the property path leading up to the target property evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged but
+ * not rethrown. If set to <code>false</code> then if any objects in the property path leading
+ * up to the target property evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged and
+ * rethrown.
+ */
+ private boolean ignoreNull;
+
+ /**
+ * Constructor which takes the name of the property to be changed, the new value to set
+ * the property to, and assumes <code>ignoreNull</code> to be <code>false</code>.
+ *
+ * @param propertyName The name of the property that will be updated with the value specified by
+ * <code>propertyValue</code>.
+ * @param propertyValue The value that <code>propertyName</code> will be set to on the target
+ * object.
+ * @throws IllegalArgumentException If the propertyName provided is null or empty.
+ */
+ public BeanPropertyValueChangeClosure(String propertyName, Object propertyValue) {
+ this(propertyName, propertyValue, false);
+ }
+
+ /**
+ * Constructor which takes the name of the property to be changed, the new value to set
+ * the property to and a boolean which determines whether <code>null</code> objects in the
+ * property path will genenerate an <code>IllegalArgumentException</code> or not.
+ *
+ * @param propertyName The name of the property that will be updated with the value specified by
+ * <code>propertyValue</code>.
+ * @param propertyValue The value that <code>propertyName</code> will be set to on the target
+ * object.
+ * @param ignoreNull Determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not.
+ * @throws IllegalArgumentException If the propertyName provided is null or empty.
+ */
+ public BeanPropertyValueChangeClosure(String propertyName, Object propertyValue, boolean ignoreNull) {
+ super();
+
+ if ((propertyName != null) && (propertyName.length() > 0)) {
+ this.propertyName = propertyName;
+ this.propertyValue = propertyValue;
+ this.ignoreNull = ignoreNull;
+ } else {
+ throw new IllegalArgumentException("propertyName cannot be null or empty");
+ }
+ }
+
+ /**
+ * Updates the target object provided using the property update criteria provided when this
+ * <code>BeanPropertyValueChangeClosure</code> was constructed. If any object in the property
+ * path leading up to the target property is <code>null</code> then the outcome will be based on
+ * the value of the <code>ignoreNull</code> attribute. By default, <code>ignoreNull</code> is
+ * <code>false</code> and would result in an <code>IllegalArgumentException</code> if an object
+ * in the property path leading up to the target property is <code>null</code>.
+ *
+ * @param object The object to be updated.
+ * @throws IllegalArgumentException If an IllegalAccessException, InvocationTargetException, or
+ * NoSuchMethodException is thrown when trying to access the property specified on the object
+ * provided. Or if an object in the property path provided is <code>null</code> and
+ * <code>ignoreNull</code> is set to <code>false</code>.
+ */
+ public void execute(Object object) {
+
+ try {
+ PropertyUtils.setProperty(object, propertyName, propertyValue);
+ } catch (IllegalArgumentException e) {
+ final String errorMsg = "Unable to execute Closure. Null value encountered in property path...";
+
+ if (ignoreNull) {
+ log.warn("WARNING: " + errorMsg, e);
+ } else {
+ log.error("ERROR: " + errorMsg, e);
+ throw e;
+ }
+ } catch (IllegalAccessException e) {
+ final String errorMsg = "Unable to access the property provided.";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ } catch (InvocationTargetException e) {
+ final String errorMsg = "Exception occurred in property's getter";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ } catch (NoSuchMethodException e) {
+ final String errorMsg = "Property not found";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ }
+ }
+
+ /**
+ * Returns the name of the property which will be updated when this <code>Closure</code> executes.
+ *
+ * @return The name of the property which will be updated when this <code>Closure</code> executes.
+ */
+ public String getPropertyName() {
+ return propertyName;
+ }
+
+ /**
+ * Returns the value that the property specified by <code>propertyName</code>
+ * will be updated to when this <code>Closure</code> executes.
+ *
+ * @return The value that the property specified by <code>propertyName</code>
+ * will be updated to when this <code>Closure</code> executes.
+ */
+ public Object getPropertyValue() {
+ return propertyValue;
+ }
+
+ /**
+ * Returns the flag that determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not. If set to <code>true</code> then
+ * if any objects in the property path leading up to the target property evaluate to
+ * <code>null</code> then the <code>IllegalArgumentException</code> throw by
+ * <code>PropertyUtils</code> will be logged but not rethrown. If set to <code>false</code> then
+ * if any objects in the property path leading up to the target property evaluate to
+ * <code>null</code> then the <code>IllegalArgumentException</code> throw by
+ * <code>PropertyUtils</code> will be logged and rethrown.
+ *
+ * @return The flag that determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not.
+ */
+ public boolean isIgnoreNull() {
+ return ignoreNull;
+ }
+}
diff --git a/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanPropertyValueEqualsPredicate.java b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanPropertyValueEqualsPredicate.java
new file mode 100644
index 0000000..d7f544f
--- /dev/null
+++ b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanPropertyValueEqualsPredicate.java
@@ -0,0 +1,275 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import org.apache.commons.collections.Predicate;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.lang.reflect.InvocationTargetException;
+
+
+/**
+ * <p><code>Predicate</code> that evaluates a property value against a specified value.</p>
+ * <p>
+ * An implementation of <code>org.apache.commons.collections.Predicate</code> that evaluates a
+ * property value on the object provided against a specified value and returns <code>true</code>
+ * if equal; <code>false</code> otherwise.
+ * The <code>BeanPropertyValueEqualsPredicate</code> constructor takes two parameters which
+ * determine what property will be evaluated on the target object and what its expected value should
+ * be.
+ * <dl>
+ * <dt>
+ * <strong><code><pre>public BeanPropertyValueEqualsPredicate( String propertyName, Object propertyValue )</pre></code></strong>
+ * </dt>
+ * <dd>
+ * Will create a <code>Predicate</code> that will evaluate the target object and return
+ * <code>true</code> if the property specified by <code>propertyName</code> has a value which
+ * is equal to the the value specified by <code>propertyValue</code>. Or return
+ * <code>false</code> otherwise.
+ * </dd>
+ * </dl>
+ * </p>
+ * <p>
+ * <strong>Note:</strong> Property names can be a simple, nested, indexed, or mapped property as defined by
+ * <code>org.apache.commons.beanutils.PropertyUtils</code>. If any object in the property path
+ * specified by <code>propertyName</code> is <code>null</code> then the outcome is based on the
+ * value of the <code>ignoreNull</code> attribute.
+ * </p>
+ * <p>
+ * A typical usage might look like:
+ * <code><pre>
+ * // create the closure
+ * BeanPropertyValueEqualsPredicate predicate =
+ * new BeanPropertyValueEqualsPredicate( "activeEmployee", Boolean.FALSE );
+ *
+ * // filter the Collection
+ * CollectionUtils.filter( peopleCollection, predicate );
+ * </pre></code>
+ * </p>
+ * <p>
+ * This would take a <code>Collection</code> of person objects and filter out any people whose
+ * <code>activeEmployee</code> property is <code>false</code>. Assuming...
+ * <ul>
+ * <li>
+ * The top level object in the <code>peeopleCollection</code> is an object which represents a
+ * person.
+ * </li>
+ * <li>
+ * The person object has a <code>getActiveEmployee()</code> method which returns
+ * the boolean value for the object's <code>activeEmployee</code> property.
+ * </li>
+ * </ul>
+ * </p>
+ * <p>
+ * Another typical usage might look like:
+ * <code><pre>
+ * // create the closure
+ * BeanPropertyValueEqualsPredicate predicate =
+ * new BeanPropertyValueEqualsPredicate( "personId", "456-12-1234" );
+ *
+ * // search the Collection
+ * CollectionUtils.find( peopleCollection, predicate );
+ * </pre><code>
+ * </p>
+ * <p>
+ * This would search a <code>Collection</code> of person objects and return the first object whose
+ * <code>personId</code> property value equals <code>456-12-1234</code>. Assuming...
+ * <ul>
+ * <li>
+ * The top level object in the <code>peeopleCollection</code> is an object which represents a
+ * person.
+ * </li>
+ * <li>
+ * The person object has a <code>getPersonId()</code> method which returns
+ * the value for the object's <code>personId</code> property.
+ * </li>
+ * </ul>
+ * </p>
+ *
+ * @author Norm Deane
+ * @see org.apache.commons.beanutils.PropertyUtils
+ * @see org.apache.commons.collections.Predicate
+ */
+public class BeanPropertyValueEqualsPredicate implements Predicate {
+
+ /** For logging. */
+ private final Log log = LogFactory.getLog(this.getClass());
+
+ /**
+ * The name of the property which will be evaluated when this <code>Predicate</code> is executed.
+ */
+ private String propertyName;
+
+ /**
+ * The value that the property specified by <code>propertyName</code>
+ * will be compared to when this <code>Predicate</code> executes.
+ */
+ private Object propertyValue;
+
+ /**
+ * <p>Should <code>null</code> objects in the property path be ignored?</p>
+ * <p>
+ * Determines whether <code>null</code> objects in the property path will genenerate an
+ * <code>IllegalArgumentException</code> or not. If set to <code>true</code> then if any objects
+ * in the property path evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged but
+ * not rethrown and <code>false</code> will be returned. If set to <code>false</code> then if
+ * any objects in the property path evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged and
+ * rethrown.
+ * </p>
+ */
+ private boolean ignoreNull;
+
+ /**
+ * Constructor which takes the name of the property, its expected value to be used in evaluation,
+ * and assumes <code>ignoreNull</code> to be <code>false</code>.
+ *
+ * @param propertyName The name of the property that will be evaluated against the expected value.
+ * @param propertyValue The value to use in object evaluation.
+ * @throws IllegalArgumentException If the property name provided is null or empty.
+ */
+ public BeanPropertyValueEqualsPredicate(String propertyName, Object propertyValue) {
+ this(propertyName, propertyValue, false);
+ }
+
+ /**
+ * Constructor which takes the name of the property, its expected value
+ * to be used in evaluation, and a boolean which determines whether <code>null</code> objects in
+ * the property path will genenerate an <code>IllegalArgumentException</code> or not.
+ *
+ * @param propertyName The name of the property that will be evaluated against the expected value.
+ * @param propertyValue The value to use in object evaluation.
+ * @param ignoreNull Determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not.
+ * @throws IllegalArgumentException If the property name provided is null or empty.
+ */
+ public BeanPropertyValueEqualsPredicate(String propertyName, Object propertyValue, boolean ignoreNull) {
+ super();
+
+ if ((propertyName != null) && (propertyName.length() > 0)) {
+ this.propertyName = propertyName;
+ this.propertyValue = propertyValue;
+ this.ignoreNull = ignoreNull;
+ } else {
+ throw new IllegalArgumentException("propertyName cannot be null or empty");
+ }
+ }
+
+ /**
+ * Evaulates the object provided against the criteria specified when this
+ * <code>BeanPropertyValueEqualsPredicate</code> was constructed. Equality is based on
+ * either reference or logical equality as defined by the property object's equals method. If
+ * any object in the property path leading up to the target property is <code>null</code> then
+ * the outcome will be based on the value of the <code>ignoreNull</code> attribute. By default,
+ * <code>ignoreNull</code> is <code>false</code> and would result in an
+ * <code>IllegalArgumentException</code> if an object in the property path leading up to the
+ * target property is <code>null</code>.
+ *
+ * @param object The object to be evaluated.
+ * @return True if the object provided meets all the criteria for this <code>Predicate</code>;
+ * false otherwise.
+ * @throws IllegalArgumentException If an IllegalAccessException, InvocationTargetException, or
+ * NoSuchMethodException is thrown when trying to access the property specified on the object
+ * provided. Or if an object in the property path provided is <code>null</code> and
+ * <code>ignoreNull</code> is set to <code>false</code>.
+ */
+ public boolean evaluate(Object object) {
+
+ boolean evaluation = false;
+
+ try {
+ evaluation = evaluateValue(propertyValue,
+ PropertyUtils.getProperty(object, propertyName));
+ } catch (IllegalArgumentException e) {
+ final String errorMsg = "Problem during evaluation. Null value encountered in property path...";
+
+ if (ignoreNull) {
+ log.warn("WARNING: " + errorMsg, e);
+ } else {
+ log.error("ERROR: " + errorMsg, e);
+ throw e;
+ }
+ } catch (IllegalAccessException e) {
+ final String errorMsg = "Unable to access the property provided.";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ } catch (InvocationTargetException e) {
+ final String errorMsg = "Exception occurred in property's getter";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ } catch (NoSuchMethodException e) {
+ final String errorMsg = "Property not found.";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ }
+
+ return evaluation;
+ }
+
+ /**
+ * Utility method which evaluates whether the actual property value equals the expected property
+ * value.
+ *
+ * @param expected The expected value.
+ * @param actual The actual value.
+ * @return True if they are equal; false otherwise.
+ */
+ private boolean evaluateValue(Object expected, Object actual) {
+ return (expected == actual) || ((expected != null) && expected.equals(actual));
+ }
+
+ /**
+ * Returns the name of the property which will be evaluated when this <code>Predicate</code> is
+ * executed.
+ *
+ * @return The name of the property which will be evaluated when this <code>Predicate</code> is
+ * executed.
+ */
+ public String getPropertyName() {
+ return propertyName;
+ }
+
+ /**
+ * Returns the value that the property specified by <code>propertyName</code> will be compared to
+ * when this <code>Predicate</code> executes.
+ *
+ * @return The value that the property specified by <code>propertyName</code> will be compared to
+ * when this <code>Predicate</code> executes.
+ */
+ public Object getPropertyValue() {
+ return propertyValue;
+ }
+
+ /**
+ * Returns the flag which determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not. If set to <code>true</code> then
+ * if any objects in the property path evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged but
+ * not rethrown and <code>false</code> will be returned. If set to <code>false</code> then if
+ * any objects in the property path evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged and
+ * rethrown.
+ *
+ * @return The flag which determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not.
+ */
+ public boolean isIgnoreNull() {
+ return ignoreNull;
+ }
+}
diff --git a/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanToPropertyValueTransformer.java b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanToPropertyValueTransformer.java
new file mode 100644
index 0000000..f427d09
--- /dev/null
+++ b/trunk/optional/bean-collections/src/java/org/apache/commons/beanutils/BeanToPropertyValueTransformer.java
@@ -0,0 +1,205 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import org.apache.commons.collections.Transformer;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.lang.reflect.InvocationTargetException;
+
+
+/**
+ * <p><code>Transformer</code> that outputs a property value.</p>
+ *
+ * <p>An implementation of <code>org.apache.commons.collections.Transformer</code> that transforms
+ * the object provided by returning the value of a specified property of the object. The
+ * constructor for <code>BeanToPropertyValueTransformer</code> requires the name of the property
+ * that will be used in the transformation. The property can be a simple, nested, indexed, or
+ * mapped property as defined by <code>org.apache.commons.beanutils.PropertyUtils</code>. If any
+ * object in the property path specified by <code>propertyName</code> is <code>null</code> then the
+ * outcome is based on the value of the <code>ignoreNull</code> attribute.
+ * </p>
+ *
+ * <p>
+ * A typical usage might look like:
+ * <code><pre>
+ * // create the transformer
+ * BeanToPropertyValueTransformer transformer = new BeanToPropertyValueTransformer( "person.address.city" );
+ *
+ * // transform the Collection
+ * Collection peoplesCities = CollectionUtils.collect( peopleCollection, transformer );
+ * </pre></code>
+ * </p>
+ *
+ * <p>
+ * This would take a <code>Collection</code> of person objects and return a <code>Collection</code>
+ * of objects which represents the cities in which each person lived. Assuming...
+ * <ul>
+ * <li>
+ * The top level object in the <code>peeopleCollection</code> is an object which represents a
+ * person.
+ * </li>
+ * <li>
+ * The person object has a <code>getAddress()</code> method which returns an object which
+ * represents a person's address.
+ * </li>
+ * <li>
+ * The address object has a <code>getCity()</code> method which returns an object which
+ * represents the city in which a person lives.
+ * </li>
+ * </ul>
+ *
+ * @author Norm Deane
+ * @see org.apache.commons.beanutils.PropertyUtils
+ * @see org.apache.commons.collections.Transformer
+ */
+public class BeanToPropertyValueTransformer implements Transformer {
+
+ /** For logging. */
+ private final Log log = LogFactory.getLog(this.getClass());
+
+ /** The name of the property that will be used in the transformation of the object. */
+ private String propertyName;
+
+ /**
+ * <p>Should null objects on the property path throw an <code>IllegalArgumentException</code>?</p>
+ * <p>
+ * Determines whether <code>null</code> objects in the property path will genenerate an
+ * <code>IllegalArgumentException</code> or not. If set to <code>true</code> then if any objects
+ * in the property path evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged but
+ * not rethrown and <code>null</code> will be returned. If set to <code>false</code> then if any
+ * objects in the property path evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged and
+ * rethrown.
+ * </p>
+ */
+ private boolean ignoreNull;
+
+ /**
+ * Constructs a Transformer which does not ignore nulls.
+ * Constructor which takes the name of the property that will be used in the transformation and
+ * assumes <code>ignoreNull</code> to be <code>false</code>.
+ *
+ * @param propertyName The name of the property that will be used in the transformation.
+ * @throws IllegalArgumentException If the <code>propertyName</code> is <code>null</code> or
+ * empty.
+ */
+ public BeanToPropertyValueTransformer(String propertyName) {
+ this(propertyName, false);
+ }
+
+ /**
+ * Constructs a Transformer and sets ignoreNull.
+ * Constructor which takes the name of the property that will be used in the transformation and
+ * a boolean which determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not.
+ *
+ * @param propertyName The name of the property that will be used in the transformation.
+ * @param ignoreNull Determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not.
+ * @throws IllegalArgumentException If the <code>propertyName</code> is <code>null</code> or
+ * empty.
+ */
+ public BeanToPropertyValueTransformer(String propertyName, boolean ignoreNull) {
+ super();
+
+ if ((propertyName != null) && (propertyName.length() > 0)) {
+ this.propertyName = propertyName;
+ this.ignoreNull = ignoreNull;
+ } else {
+ throw new IllegalArgumentException(
+ "propertyName cannot be null or empty");
+ }
+ }
+
+ /**
+ * Returns the value of the property named in the transformer's constructor for
+ * the object provided. If any object in the property path leading up to the target property is
+ * <code>null</code> then the outcome will be based on the value of the <code>ignoreNull</code>
+ * attribute. By default, <code>ignoreNull</code> is <code>false</code> and would result in an
+ * <code>IllegalArgumentException</code> if an object in the property path leading up to the
+ * target property is <code>null</code>.
+ *
+ * @param object The object to be transformed.
+ * @return The value of the property named in the transformer's constructor for the object
+ * provided.
+ * @throws IllegalArgumentException If an IllegalAccessException, InvocationTargetException, or
+ * NoSuchMethodException is thrown when trying to access the property specified on the object
+ * provided. Or if an object in the property path provided is <code>null</code> and
+ * <code>ignoreNull</code> is set to <code>false</code>.
+ */
+ public Object transform(Object object) {
+
+ Object propertyValue = null;
+
+ try {
+ propertyValue = PropertyUtils.getProperty(object, propertyName);
+ } catch (IllegalArgumentException e) {
+ final String errorMsg = "Problem during transformation. Null value encountered in property path...";
+
+ if (ignoreNull) {
+ log.warn("WARNING: " + errorMsg, e);
+ } else {
+ log.error("ERROR: " + errorMsg, e);
+ throw e;
+ }
+ } catch (IllegalAccessException e) {
+ final String errorMsg = "Unable to access the property provided.";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ } catch (InvocationTargetException e) {
+ final String errorMsg = "Exception occurred in property's getter";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ } catch (NoSuchMethodException e) {
+ final String errorMsg = "No property found for name [" +
+ propertyName + "]";
+ log.error(errorMsg, e);
+ throw new IllegalArgumentException(errorMsg);
+ }
+
+ return propertyValue;
+ }
+
+ /**
+ * Returns the name of the property that will be used in the transformation of the bean.
+ *
+ * @return The name of the property that will be used in the transformation of the bean.
+ */
+ public String getPropertyName() {
+ return propertyName;
+ }
+
+ /**
+ * Returns the flag which determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not. If set to <code>true</code> then
+ * if any objects in the property path evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged but
+ * not rethrown and <code>null</code> will be returned. If set to <code>false</code> then if any
+ * objects in the property path evaluate to <code>null</code> then the
+ * <code>IllegalArgumentException</code> throw by <code>PropertyUtils</code> will be logged and
+ * rethrown.
+ *
+ * @return The flag which determines whether <code>null</code> objects in the property path will
+ * genenerate an <code>IllegalArgumentException</code> or not.
+ */
+ public boolean isIgnoreNull() {
+ return ignoreNull;
+ }
+}
diff --git a/trunk/optional/bean-collections/src/java/overview.html b/trunk/optional/bean-collections/src/java/overview.html
new file mode 100644
index 0000000..b0f4f9c
--- /dev/null
+++ b/trunk/optional/bean-collections/src/java/overview.html
@@ -0,0 +1,19 @@
+<html>
+<head>
+<title>Overview Documentation for COMMONS-BEANUTILS BEAN COLLECTIONS</title>
+</head>
+<body bgcolor="white">
+<p>The <em>Bean Introspection Utilities</em> component of the Jakarta Commons
+subproject offers low-level utility classes that assist in getting and setting
+property values on Java classes that follow the naming design patterns outlined
+in the JavaBeans Specification, as well as mechanisms for dynamically defining
+and accessing bean properties.</p>
+<p>
+The Bean-Collections optional distribution contains objects which apply these
+techniques to collections. They rely on commons-collections 3.0 (or higher).
+</p>
+
+<p>The documentation for this optional package is included with the core Beanutils
+documentation.</p>
+</body>
+</html>
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/AbstractParent.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/AbstractParent.java
new file mode 100644
index 0000000..0687a3f
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/AbstractParent.java
@@ -0,0 +1,53 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+public abstract class AbstractParent {
+
+ private Child child;
+
+ public Child getChild()
+ {
+ return child;
+ }
+
+ /**
+ * Method which matches signature but which has wrong parameters
+ */
+ public String testAddChild(String badParameter) {
+ return null;
+ }
+
+ /**
+ * Method which matches signature but which has wrong parameters
+ */
+ public String testAddChild2(String ignore, String badParameter) {
+ return null;
+ }
+
+ public String testAddChild(Child child) {
+ this.child = child;
+ return child.getName();
+ }
+
+
+ public String testAddChild2(String ignore, Child child) {
+ this.child = child;
+ return child.getName();
+ }
+
+}
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/AlphaBean.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/AlphaBean.java
new file mode 100644
index 0000000..50bf3c5
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/AlphaBean.java
@@ -0,0 +1,41 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+public class AlphaBean extends AbstractParent implements Child {
+
+ private String name;
+
+ public AlphaBean() {}
+
+ public AlphaBean(String name) {
+ setName(name);
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public void setName(String name) {
+ this.name = name;
+ }
+
+ /**
+ * Used for testing that correct exception is thrown.
+ */
+ public void bogus(String badParameter){}
+}
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanCollectionsTestSuite.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanCollectionsTestSuite.java
new file mode 100644
index 0000000..577f776
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanCollectionsTestSuite.java
@@ -0,0 +1,57 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.util.HashMap;
+import java.util.Map;
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * <p>Test suite which runs all the test cases application to beans in collections.
+ *
+ * @author <a href="mailto:epugh at upstate.com">Eric Pugh</a>
+ * @version $Revision: 1.2 $
+ */
+
+public class BeanCollectionsTestSuite extends TestCase {
+
+ public BeanCollectionsTestSuite(java.lang.String testName) {
+ super(testName);
+ }
+
+ public static void main(java.lang.String[] args) {
+ junit.textui.TestRunner.run(suite());
+ }
+
+ public static junit.framework.Test suite() {
+ TestSuite suite = new TestSuite();
+ suite.addTestSuite(BeanComparatorTestCase.class);
+ suite.addTestSuite(BeanToPropertyValueTransformerTest.class);
+ suite.addTestSuite(BeanPropertyValueEqualsPredicateTest.class);
+ suite.addTestSuite(BeanPropertyValueChangeClosureTest.class);
+ suite.addTestSuite(TestBeanMap.class);
+
+ return suite;
+ }
+}
+
+
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanComparatorTestCase.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanComparatorTestCase.java
new file mode 100644
index 0000000..cc236c0
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanComparatorTestCase.java
@@ -0,0 +1,261 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.util.HashMap;
+import java.util.Map;
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * <p>
+ * Test Case for the BeanComparator class.
+ *
+ * @author <a href="mailto:epugh at upstate.com">Eric Pugh</a>
+ * @version $Revision: 1.1 $
+ */
+
+public class BeanComparatorTestCase extends TestCase {
+
+ // ---------------------------------------------------- Instance Variables
+
+ /**
+ * The test beans for each test.
+ */
+ protected TestBean bean = null;
+ protected AlphaBean alphaBean1 = null;
+ protected AlphaBean alphaBean2 = null;
+
+ // The test BeanComparator
+ protected BeanComparator beanComparator = null;
+
+
+
+
+
+ // ---------------------------------------------------------- Constructors
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public BeanComparatorTestCase(String name) {
+ super(name);
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() {
+ bean = new TestBean();
+ alphaBean1 = new AlphaBean("alphaBean1");
+ alphaBean2 = new AlphaBean("alphaBean2");
+
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(BeanComparatorTestCase.class));
+ }
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ bean = null;
+ alphaBean1 = null;
+ alphaBean2 = null;
+ beanComparator = null;
+ }
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ /**
+ * tests comparing two beans via their name using the default Comparator
+ */
+ public void testSimpleCompare() {
+ try {
+ beanComparator = new BeanComparator("name");
+ int result = beanComparator.compare(alphaBean1, alphaBean2);
+ assertTrue("Comparator did not sort properly. Result:" + result,result==-1);
+
+ }
+ catch (Exception e) {
+ fail("Exception");
+ }
+ }
+
+ /**
+ * tests comparing two beans via their name using the default Comparator, but the inverse
+ */
+ public void testSimpleCompareInverse() {
+ try {
+ beanComparator = new BeanComparator("name");
+ int result = beanComparator.compare(alphaBean2, alphaBean1);
+ assertTrue("Comparator did not sort properly. Result:" + result,result==1);
+
+ }
+ catch (Exception e) {
+ fail("Exception" + e);
+ }
+ }
+
+ /**
+ * tests comparing two beans via their name using the default Comparator where they have the same value.
+ */
+ public void testCompareIdentical() {
+ try {
+ alphaBean1 = new AlphaBean("alphabean");
+ alphaBean2 = new AlphaBean("alphabean");
+ beanComparator = new BeanComparator("name");
+ int result = beanComparator.compare(alphaBean1, alphaBean2);
+ assertTrue("Comparator did not sort properly. Result:" + result,result==0);
+
+ }
+ catch (Exception e) {
+ fail("Exception");
+ }
+ }
+
+ /**
+ * tests comparing one bean against itself.
+ */
+ public void testCompareBeanAgainstSelf() {
+ try {
+ beanComparator = new BeanComparator("name");
+ int result = beanComparator.compare(alphaBean1, alphaBean1);
+ assertTrue("Comparator did not sort properly. Result:" + result,result==0);
+
+ }
+ catch (Exception e) {
+ fail("Exception");
+ }
+ }
+
+ /**
+ * tests comparing two beans via their name using the default Comparator, but with one of the beans
+ * being null.
+ */
+ public void testCompareWithNulls() {
+ try {
+ beanComparator = new BeanComparator("name");
+ beanComparator.compare(alphaBean2, null);
+
+ // DEP not sure if this is the best way to test an exception?
+ fail("Should not be able to compare a null value.");
+
+ }
+ catch (Exception e) {
+
+ }
+ }
+
+ /**
+ * tests comparing two beans who don't have a property
+ */
+ public void testCompareOnMissingProperty() {
+ try {
+ beanComparator = new BeanComparator("bogusName");
+ beanComparator.compare(alphaBean2, alphaBean1);
+ fail("should not be able to compare");
+
+
+ }
+ catch (ClassCastException cce){
+ assertTrue("Wrong exception was thrown.",cce.toString().indexOf("Unknown property") > -1);
+ }
+ catch (Exception e) {
+ fail("Exception" + e);
+ }
+ }
+
+ /**
+ * tests comparing two beans on a boolean property, which is not possible.
+ */
+ public void testCompareOnBooleanProperty() {
+ try {
+ TestBean testBeanA = new TestBean();
+ TestBean testBeanB = new TestBean();
+
+ testBeanA.setBooleanProperty(true);
+ testBeanB.setBooleanProperty(false);
+
+ beanComparator = new BeanComparator("booleanProperty");
+ beanComparator.compare(testBeanA, testBeanB);
+
+ fail("BeanComparator should throw an exception when comparing two booleans.");
+
+ }
+ catch (ClassCastException cce){
+ ; // Expected result
+ }
+ catch (Exception e) {
+ fail("Exception" + e);
+ }
+ }
+
+ /**
+ * tests comparing two beans on a boolean property, then changing the property and testing
+ */
+ public void testSetProperty() {
+ try {
+ TestBean testBeanA = new TestBean();
+ TestBean testBeanB = new TestBean();
+
+ testBeanA.setDoubleProperty(5.5);
+ testBeanB.setDoubleProperty(1.0);
+
+ beanComparator = new BeanComparator("doubleProperty");
+ int result = beanComparator.compare(testBeanA, testBeanB);
+
+ assertTrue("Comparator did not sort properly. Result:" + result,result==1);
+
+ testBeanA.setStringProperty("string 1");
+ testBeanB.setStringProperty("string 2");
+
+ beanComparator.setProperty("stringProperty");
+
+ result = beanComparator.compare(testBeanA, testBeanB);
+
+ assertTrue("Comparator did not sort properly. Result:" + result,result==-1);
+
+ }
+ catch (ClassCastException cce){
+ fail("ClassCaseException " + cce.toString());
+ }
+ catch (Exception e) {
+ fail("Exception" + e);
+ }
+ }
+}
+
+
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanPredicateTestCase.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanPredicateTestCase.java
new file mode 100644
index 0000000..d5ee508
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanPredicateTestCase.java
@@ -0,0 +1,61 @@
+/*
+ * Copyright 2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import junit.framework.TestCase;
+
+import org.apache.commons.collections.functors.EqualPredicate;
+import org.apache.commons.collections.functors.InstanceofPredicate;
+import org.apache.commons.collections.functors.NotPredicate;
+import org.apache.commons.collections.functors.NullPredicate;
+
+public class BeanPredicateTestCase extends TestCase {
+
+ public BeanPredicateTestCase(String name) {
+ super(name);
+ }
+
+ public void testEqual() {
+ BeanPredicate predicate =
+ new BeanPredicate("stringProperty",new EqualPredicate("foo"));
+ assertTrue(predicate.evaluate(new TestBean("foo")));
+ assertTrue(!predicate.evaluate(new TestBean("bar")));
+ }
+
+ public void testNotEqual() {
+ BeanPredicate predicate =
+ new BeanPredicate("stringProperty",new NotPredicate( new EqualPredicate("foo")));
+ assertTrue(!predicate.evaluate(new TestBean("foo")));
+ assertTrue(predicate.evaluate(new TestBean("bar")));
+ }
+
+ public void testInstanceOf() {
+ BeanPredicate predicate =
+ new BeanPredicate("stringProperty",new InstanceofPredicate( String.class ));
+ assertTrue(predicate.evaluate(new TestBean("foo")));
+ assertTrue(predicate.evaluate(new TestBean("bar")));
+ }
+
+ public void testNull() {
+ BeanPredicate predicate =
+ new BeanPredicate("stringProperty", NullPredicate.INSTANCE);
+ String nullString = null;
+ assertTrue(predicate.evaluate(new TestBean(nullString)));
+ assertTrue(!predicate.evaluate(new TestBean("bar")));
+ }
+
+}
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanPropertyValueChangeClosureTest.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanPropertyValueChangeClosureTest.java
new file mode 100644
index 0000000..fdc0cef
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanPropertyValueChangeClosureTest.java
@@ -0,0 +1,321 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import junit.framework.TestCase;
+
+
+/**
+ * Test cases for <code>BeanPropertyValueChangeClosure</code>.
+ *
+ * @author Norm Deane
+ */
+public class BeanPropertyValueChangeClosureTest extends TestCase {
+
+ private static final Integer expectedIntegerValue = new Integer(123);
+ private static final Float expectedFloatValue = new Float(123.123f);
+ private static final Double expectedDoubleValue = new Double(567879.12344d);
+ private static final Boolean expectedBooleanValue = Boolean.TRUE;
+ private static final Byte expectedByteValue = new Byte("12");
+
+ /**
+ * Constructor for BeanPropertyValueChangeClosureTest.
+ *
+ * @param name Name of this test case.
+ */
+ public BeanPropertyValueChangeClosureTest(String name) {
+ super(name);
+ }
+
+ /**
+ * Test execute with simple float property and Float value.
+ */
+ public void testExecuteWithSimpleFloatPropertyAndFloatValue() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("floatProperty", expectedFloatValue).execute(testBean);
+ assertTrue(expectedFloatValue.floatValue() == testBean.getFloatProperty());
+ }
+
+ /**
+ * Test execute with simple float property and String value.
+ */
+ public void testExecuteWithSimpleFloatPropertyAndStringValue() {
+ try {
+ new BeanPropertyValueChangeClosure("floatProperty", "123").execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with simple float property and Double value.
+ */
+ public void testExecuteWithSimpleFloatPropertyAndDoubleValue() {
+ try {
+ new BeanPropertyValueChangeClosure("floatProperty", expectedDoubleValue).execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with simple float property and Integer value.
+ */
+ public void testExecuteWithSimpleFloatPropertyAndIntegerValue() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("floatProperty", expectedIntegerValue).execute(testBean);
+ assertTrue(expectedIntegerValue.floatValue() == testBean.getFloatProperty());
+ }
+
+ /**
+ * Test execute with simple double property and Double value.
+ */
+ public void testExecuteWithSimpleDoublePropertyAndDoubleValue() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("doubleProperty", expectedDoubleValue).execute(testBean);
+ assertTrue(expectedDoubleValue.doubleValue() == testBean.getDoubleProperty());
+ }
+
+ /**
+ * Test execute with simple double property and String value.
+ */
+ public void testExecuteWithSimpleDoublePropertyAndStringValue() {
+ try {
+ new BeanPropertyValueChangeClosure("doubleProperty", "123").execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with simple double property and Float value.
+ */
+ public void testExecuteWithSimpleDoublePropertyAndFloatValue() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("doubleProperty", expectedFloatValue).execute(testBean);
+ assertTrue(expectedFloatValue.doubleValue() == testBean.getDoubleProperty());
+ }
+
+ /**
+ * Test execute with simple double property and Integer value.
+ */
+ public void testExecuteWithSimpleDoublePropertyAndIntegerValue() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("doubleProperty", expectedIntegerValue).execute(testBean);
+ assertTrue(expectedIntegerValue.doubleValue() == testBean.getDoubleProperty());
+ }
+
+ /**
+ * Test execute with simple int property and Double value.
+ */
+ public void testExecuteWithSimpleIntPropertyAndDoubleValue() {
+ try {
+ new BeanPropertyValueChangeClosure("intProperty", expectedDoubleValue).execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with simple int property and String value.
+ */
+ public void testExecuteWithSimpleIntPropertyAndStringValue() {
+ try {
+ new BeanPropertyValueChangeClosure("intProperty", "123").execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with simple int property and Float value.
+ */
+ public void testExecuteWithSimpleIntPropertyAndFloatValue() {
+ try {
+ new BeanPropertyValueChangeClosure("intProperty", expectedFloatValue).execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with simple int property and Integer value.
+ */
+ public void testExecuteWithSimpleIntPropertyAndIntegerValue() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("intProperty", expectedIntegerValue).execute(testBean);
+ assertTrue(expectedIntegerValue.intValue() == testBean.getIntProperty());
+ }
+
+ /**
+ * Test execute with simple boolean property and Boolean value.
+ */
+ public void testExecuteWithSimpleBooleanPropertyAndBooleanValue() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("booleanProperty", expectedBooleanValue).execute(testBean);
+ assertTrue(expectedBooleanValue.booleanValue() == testBean.getBooleanProperty());
+ }
+
+ /**
+ * Test execute with simple boolean property and String value.
+ */
+ public void testExecuteWithSimpleBooleanPropertyAndStringValue() {
+ try {
+ new BeanPropertyValueChangeClosure("booleanProperty", "true").execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with simple byte property and Byte value.
+ */
+ public void testExecuteWithSimpleBytePropertyAndByteValue() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("byteProperty", expectedByteValue).execute(testBean);
+ assertTrue(expectedByteValue.byteValue() == testBean.getByteProperty());
+ }
+
+ /**
+ * Test execute with simple boolean property and String value.
+ */
+ public void testExecuteWithSimpleBytePropertyAndStringValue() {
+ try {
+ new BeanPropertyValueChangeClosure("byteProperty", "foo").execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with simple primitive property and null value.
+ */
+ public void testExecuteWithSimplePrimitivePropertyAndNullValue() {
+ try {
+ new BeanPropertyValueChangeClosure("intProperty", null).execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (NullPointerException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with read only property.
+ */
+ public void testExecuteWithReadOnlyProperty() {
+ try {
+ new BeanPropertyValueChangeClosure("readOnlyProperty", "foo").execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with write only property.
+ */
+ public void testExecuteWithWriteOnlyProperty() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("writeOnlyProperty", "foo").execute(testBean);
+ assertEquals("foo", testBean.getWriteOnlyPropertyValue());
+ }
+
+ /**
+ * Test execute with a nested property.
+ */
+ public void testExecuteWithNestedProperty() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("nested.stringProperty", "bar").execute(testBean);
+ assertEquals("bar", testBean.getNested().getStringProperty());
+ }
+
+ /**
+ * Test execute with a nested property and null in the property path.
+ */
+ public void testExecuteWithNullInPropertyPath() {
+ try {
+ new BeanPropertyValueChangeClosure("anotherNested.stringProperty", "foo").execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+
+ /**
+ * Test execute with a nested property and null in the property path and ignoreNull = true.
+ */
+ public void testExecuteWithNullInPropertyPathAngIgnoreTrue() {
+ TestBean testBean = new TestBean();
+
+ // create a closure that will attempt to set a property on the null bean in the path
+ BeanPropertyValueChangeClosure closure = new BeanPropertyValueChangeClosure("anotherNested.stringProperty",
+ "Should ignore exception", true);
+
+ try {
+ closure.execute(testBean);
+ } catch (IllegalArgumentException e) {
+ fail("Should have ignored the exception.");
+ }
+ }
+
+ /**
+ * Test execute with indexed property.
+ */
+ public void testExecuteWithIndexedProperty() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("intIndexed[0]", expectedIntegerValue).execute(testBean);
+ assertTrue(expectedIntegerValue.intValue() == testBean.getIntIndexed(0));
+ }
+
+ /**
+ * Test execute with mapped property.
+ */
+ public void testExecuteWithMappedProperty() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("mappedProperty(fred)", "barney").execute(testBean);
+ assertEquals("barney", testBean.getMappedProperty("fred"));
+ }
+
+ /**
+ * Test execute with a simple String property.
+ */
+ public void testExecuteWithSimpleStringProperty() {
+ TestBean testBean = new TestBean();
+ new BeanPropertyValueChangeClosure("stringProperty", "barney").execute(testBean);
+ assertEquals("barney", testBean.getStringProperty());
+ }
+
+ /**
+ * Test execute with an invalid property name.
+ */
+ public void testExecuteWithInvalidPropertyName() {
+ try {
+ new BeanPropertyValueChangeClosure("bogusProperty", "foo").execute(new TestBean());
+ fail("Should have thrown an IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* this is what we expect */
+ }
+ }
+}
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanPropertyValueEqualsPredicateTest.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanPropertyValueEqualsPredicateTest.java
new file mode 100644
index 0000000..112b236
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanPropertyValueEqualsPredicateTest.java
@@ -0,0 +1,261 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import junit.framework.TestCase;
+
+
+/**
+ * Test cases for <code>BeanPropertyValueEqualsPredicateTest</code>.
+ *
+ * @author Norm Deane
+ */
+public class BeanPropertyValueEqualsPredicateTest extends TestCase {
+
+ private static final Integer expectedIntegerValue = new Integer(123);
+ private static final Float expectedFloatValue = new Float(123.123f);
+ private static final Double expectedDoubleValue = new Double(567879.12344d);
+ private static final Boolean expectedBooleanValue = Boolean.TRUE;
+ private static final Byte expectedByteValue = new Byte("12");
+
+ /**
+ * Constructor for BeanPropertyValueEqualsPredicateTest.
+ *
+ * @param name Name of this test case.
+ */
+ public BeanPropertyValueEqualsPredicateTest(String name) {
+ super(name);
+ }
+
+ /**
+ * Test evaluate with simple String property.
+ */
+ public void testEvaluateWithSimpleStringProperty() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("stringProperty","foo");
+ assertTrue(predicate.evaluate(new TestBean("foo")));
+ assertTrue(!predicate.evaluate(new TestBean("bar")));
+ }
+
+ /**
+ * Test evaluate with simple String property and null values.
+ */
+ public void testEvaluateWithSimpleStringPropertyWithNullValues() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("stringProperty",null);
+ assertTrue(predicate.evaluate(new TestBean((String) null)));
+ assertTrue(!predicate.evaluate(new TestBean("bar")));
+ }
+
+ /**
+ * Test evaluate with nested property.
+ */
+ public void testEvaluateWithNestedProperty() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("anotherNested.stringProperty","match");
+ TestBean testBean = new TestBean();
+ TestBean nestedBean = new TestBean("match");
+ testBean.setAnotherNested(nestedBean);
+ assertTrue(predicate.evaluate(testBean));
+ testBean.setAnotherNested(new TestBean("no-match"));
+ assertTrue(!predicate.evaluate(testBean));
+ }
+
+ /**
+ * Test evaluate with null in property path and ignore=false.
+ */
+ public void testEvaluateWithNullInPath() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("anotherNested.stringProperty","foo");
+ try {
+ // try to evaluate the predicate
+ predicate.evaluate(new TestBean());
+ fail("Should have throw IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* ignore this is what should happen */
+ }
+ }
+
+ /**
+ * Test evaluate with null in property path and ignore=true.
+ */
+ public void testEvaluateWithNullInPathAndIgnoreTrue() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("anotherNested.stringProperty","foo", true);
+ try {
+ assertTrue(!predicate.evaluate(new TestBean()));
+ } catch (IllegalArgumentException e) {
+ fail("Should not have throw IllegalArgumentException");
+ }
+ }
+
+ /**
+ * Test evaluate with int property.
+ */
+ public void testEvaluateWithIntProperty() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("intProperty",expectedIntegerValue);
+ assertTrue(predicate.evaluate(new TestBean(expectedIntegerValue.intValue())));
+ assertTrue(!predicate.evaluate(new TestBean(expectedIntegerValue.intValue() - 1)));
+ }
+
+ /**
+ * Test evaluate with float property.
+ */
+ public void testEvaluateWithFloatProperty() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("floatProperty",expectedFloatValue);
+ assertTrue(predicate.evaluate(new TestBean(expectedFloatValue.floatValue())));
+ assertTrue(!predicate.evaluate(new TestBean(expectedFloatValue.floatValue() - 1)));
+ }
+
+ /**
+ * Test evaluate with double property.
+ */
+ public void testEvaluateWithDoubleProperty() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("doubleProperty",expectedDoubleValue);
+ assertTrue(predicate.evaluate(new TestBean(expectedDoubleValue.doubleValue())));
+ assertTrue(!predicate.evaluate(new TestBean(expectedDoubleValue.doubleValue() - 1)));
+ }
+
+ /**
+ * Test evaluate with boolean property.
+ */
+ public void testEvaluateWithBooleanProperty() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("booleanProperty",expectedBooleanValue);
+ assertTrue(predicate.evaluate(new TestBean(expectedBooleanValue.booleanValue())));
+ assertTrue(!predicate.evaluate(new TestBean(!expectedBooleanValue.booleanValue())));
+ }
+
+ /**
+ * Test evaluate with byte property.
+ */
+ public void testEvaluateWithByteProperty() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("byteProperty",expectedByteValue);
+ TestBean testBean = new TestBean();
+ testBean.setByteProperty(expectedByteValue.byteValue());
+ assertTrue(predicate.evaluate(testBean));
+ testBean.setByteProperty((byte) (expectedByteValue.byteValue() - 1));
+ assertTrue(!predicate.evaluate(testBean));
+ }
+
+ /**
+ * Test evaluate with mapped property.
+ */
+ public void testEvaluateWithMappedProperty() {
+ // try a key that is in the map
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("mappedProperty(test-key)","match");
+ TestBean testBean = new TestBean();
+ testBean.setMappedProperty("test-key", "match");
+ assertTrue(predicate.evaluate(testBean));
+ testBean.setMappedProperty("test-key", "no-match");
+ assertTrue(!predicate.evaluate(testBean));
+
+ // try a key that isn't in the map
+ predicate = new BeanPropertyValueEqualsPredicate("mappedProperty(invalid-key)", "match");
+ assertTrue(!predicate.evaluate(testBean));
+ }
+
+ /**
+ * Test evaluate with indexed property.
+ */
+ public void testEvaluateWithIndexedProperty() {
+ // try a valid index
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("intIndexed[0]",expectedIntegerValue);
+ TestBean testBean = new TestBean();
+ testBean.setIntIndexed(0, expectedIntegerValue.intValue());
+ assertTrue(predicate.evaluate(testBean));
+ testBean.setIntIndexed(0, expectedIntegerValue.intValue() - 1);
+ assertTrue(!predicate.evaluate(testBean));
+
+ // try an invalid index
+ predicate = new BeanPropertyValueEqualsPredicate("intIndexed[999]", "exception-ahead");
+
+ try {
+ assertTrue(!predicate.evaluate(testBean));
+ } catch (ArrayIndexOutOfBoundsException e) {
+ /* this is what should happen */
+ }
+ }
+
+ /**
+ * Test evaluate with primitive property and null value.
+ */
+ public void testEvaluateWithPrimitiveAndNull() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("intProperty",null);
+ assertTrue(!predicate.evaluate(new TestBean(0)));
+
+ predicate = new BeanPropertyValueEqualsPredicate("booleanProperty", null);
+ assertTrue(!predicate.evaluate(new TestBean(true)));
+
+ predicate = new BeanPropertyValueEqualsPredicate("floatProperty", null);
+ assertTrue(!predicate.evaluate(new TestBean(expectedFloatValue.floatValue())));
+ }
+
+ /**
+ * Test evaluate with nested mapped property.
+ */
+ public void testEvaluateWithNestedMappedProperty() {
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("anotherNested.mappedProperty(test-key)","match");
+ TestBean testBean = new TestBean();
+ TestBean nestedBean = new TestBean();
+ nestedBean.setMappedProperty("test-key", "match");
+ testBean.setAnotherNested(nestedBean);
+ assertTrue(predicate.evaluate(testBean));
+ nestedBean.setMappedProperty("test-key", "no-match");
+ assertTrue(!predicate.evaluate(testBean));
+ }
+
+ /**
+ * Test evaluate with write only property.
+ */
+ public void testEvaluateWithWriteOnlyProperty() {
+ try {
+ new BeanPropertyValueEqualsPredicate("writeOnlyProperty", null).evaluate(new TestBean());
+ } catch (IllegalArgumentException e) {
+ /* This is what should happen */
+ }
+ }
+
+ /**
+ * Test evaluate with read only property.
+ */
+ public void testEvaluateWithReadOnlyProperty() {
+ TestBean testBean = new TestBean();
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate("readOnlyProperty",testBean.getReadOnlyProperty());
+ assertTrue(predicate.evaluate(new TestBean()));
+ }
+
+ /**
+ * Test evaluate with an invalid property name.
+ */
+ public void testEvaluateWithInvalidPropertyName() {
+ try {
+ new BeanPropertyValueEqualsPredicate("bogusProperty", null).evaluate(new TestBean());
+ } catch (IllegalArgumentException e) {
+ /* This is what should happen */
+ }
+ }
+}
\ No newline at end of file
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanToPropertyValueTransformerTest.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanToPropertyValueTransformerTest.java
new file mode 100644
index 0000000..c4df21c
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/BeanToPropertyValueTransformerTest.java
@@ -0,0 +1,246 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import junit.framework.TestCase;
+
+
+/**
+ * Test cases for <code>BeanToPropertyValueTransformer</code>.
+ *
+ * @author Norm Deane
+ */
+public class BeanToPropertyValueTransformerTest extends TestCase {
+
+ private static final Integer expectedIntegerValue = new Integer(123);
+ private static final Long expectedLongValue = new Long(123);
+ private static final Float expectedFloatValue = new Float(123.123f);
+ private static final Double expectedDoubleValue = new Double(567879.12344d);
+ private static final Boolean expectedBooleanValue = Boolean.TRUE;
+ private static final Byte expectedByteValue = new Byte("12");
+
+ /**
+ * Constructor for BeanToPropertyValueTransformerTest.
+ *
+ * @param name Name of this test case.
+ */
+ public BeanToPropertyValueTransformerTest(String name) {
+ super(name);
+ }
+
+ /**
+ * Test transform with simple String property.
+ */
+ public void testTransformWithSimpleStringProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("stringProperty");
+ TestBean testBean = new TestBean("foo");
+ assertEquals("foo", transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with simple String property and null value.
+ *
+ */
+ public void testTransformWithSimpleStringPropertyAndNullValue() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("stringProperty");
+ TestBean testBean = new TestBean((String) null);
+ assertNull(transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with simple int property.
+ */
+ public void testTransformWithSimpleIntProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("intProperty");
+ TestBean testBean = new TestBean(expectedIntegerValue.intValue());
+ assertEquals(expectedIntegerValue, transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with simple long property.
+ */
+ public void testTransformWithSimpleLongProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("longProperty");
+ TestBean testBean = new TestBean();
+ testBean.setLongProperty(expectedLongValue.longValue());
+ assertEquals(expectedLongValue, transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with simple float property.
+ */
+ public void testTransformWithSimpleFloatProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("floatProperty");
+ TestBean testBean = new TestBean(expectedFloatValue.floatValue());
+ assertEquals(expectedFloatValue, transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with simple double property.
+ */
+ public void testTransformWithSimpleDoubleProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("doubleProperty");
+ TestBean testBean = new TestBean(expectedDoubleValue.doubleValue());
+ assertEquals(expectedDoubleValue, transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with simple byte property.
+ */
+ public void testTransformWithSimpleByteProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("byteProperty");
+ TestBean testBean = new TestBean();
+ testBean.setByteProperty(expectedByteValue.byteValue());
+ assertEquals(expectedByteValue, transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with simple boolean property.
+ */
+ public void testTransformWithSimpleBooleanProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("booleanProperty");
+ TestBean testBean = new TestBean(expectedBooleanValue.booleanValue());
+ assertEquals(expectedBooleanValue, transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with write only property.
+ */
+ public void testTransformWithWriteOnlyProperty() {
+ try {
+ new BeanToPropertyValueTransformer("writeOnlyProperty").transform(new TestBean());
+ } catch (IllegalArgumentException e) {
+ /* This is what should happen */
+ }
+ }
+
+ /**
+ * Test transform with read only property.
+ */
+ public void testTransformWithReadOnlyProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("readOnlyProperty");
+ TestBean testBean = new TestBean();
+ assertEquals(testBean.getReadOnlyProperty(), transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with invalid property.
+ */
+ public void testTransformWithInvalidProperty() {
+ try {
+ new BeanToPropertyValueTransformer("bogusProperty").transform(new TestBean());
+ } catch (IllegalArgumentException e) {
+ /* This is what should happen */
+ }
+ }
+
+ /**
+ * Test transform with nested property.
+ */
+ public void testTransformWithNestedProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("anotherNested.stringProperty");
+ TestBean testBean = new TestBean();
+ TestBean nestedBean = new TestBean("foo");
+ testBean.setAnotherNested(nestedBean);
+ assertEquals("foo", transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with mapped property.
+ */
+ public void testTransformWithMappedProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("mappedProperty(test-key)");
+ TestBean testBean = new TestBean();
+
+ // try a valid key
+ testBean.setMappedProperty("test-key", "test-value");
+ assertEquals("test-value", transformer.transform(testBean));
+
+ // now try an invalid key
+ transformer = new BeanToPropertyValueTransformer("mappedProperty(bogus-key)");
+ assertEquals(null, transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with indexed property.
+ */
+ public void testTransformWithIndexedProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("intIndexed[0]");
+ TestBean testBean = new TestBean();
+ testBean.setIntIndexed(0, expectedIntegerValue.intValue());
+ assertEquals(expectedIntegerValue, transformer.transform(testBean));
+
+ // test index out of range
+ transformer = new BeanToPropertyValueTransformer("intIndexed[9999]");
+
+ try {
+ transformer.transform(testBean);
+ fail("Should have thrown an ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException e) {
+ /* this is what should happen */
+ }
+ }
+
+ /**
+ * Test transform with nested indexed property.
+ */
+ public void testTransformWithNestedIndexedProperty() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("anotherNested.intIndexed[0]");
+ TestBean testBean = new TestBean();
+ TestBean nestedBean = new TestBean();
+ nestedBean.setIntIndexed(0, expectedIntegerValue.intValue());
+ testBean.setAnotherNested(nestedBean);
+ assertEquals(expectedIntegerValue, transformer.transform(testBean));
+ }
+
+ /**
+ * Test transform with null in property path.
+ */
+ public void testTransformWithNullInPath() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("anotherNested.stringProperty");
+
+ try {
+ transformer.transform(new TestBean());
+ fail("Should have throw IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ /* ignore this is what should happen */
+ }
+ }
+
+ /**
+ * Test transform with null in property path and ignore = true.
+ */
+ public void testTransformWithNullInPathAndIgnoreTrue() {
+ BeanToPropertyValueTransformer transformer =
+ new BeanToPropertyValueTransformer("anotherNested.stringProperty",true);
+ assertEquals(null, transformer.transform(new TestBean()));
+ }
+}
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/Child.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/Child.java
new file mode 100644
index 0000000..7fb49ac
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/Child.java
@@ -0,0 +1,23 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+public interface Child {
+
+ public String getName();
+}
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/TestBean.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/TestBean.java
new file mode 100755
index 0000000..f0703aa
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/TestBean.java
@@ -0,0 +1,572 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+
+/**
+ * General purpose test bean for JUnit tests for the "beanutils" component.
+ *
+ * @author Craig R. McClanahan
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 20:01:16 $
+ */
+
+public class TestBean {
+
+ // ----------------------------------------------------------- Constructors
+
+ public TestBean() {
+ }
+
+ public TestBean(String stringProperty) {
+ setStringProperty(stringProperty);
+ }
+
+ public TestBean(float floatProperty) {
+ setFloatProperty(floatProperty);
+ }
+
+ public TestBean(boolean booleanProperty) {
+ setBooleanProperty(booleanProperty);
+ }
+
+ public TestBean(Boolean booleanSecond) {
+ setBooleanSecond(booleanSecond.booleanValue());
+ }
+
+ public TestBean(float floatProperty, String stringProperty) {
+ setFloatProperty(floatProperty);
+ setStringProperty(stringProperty);
+ }
+
+ public TestBean(boolean booleanProperty, String stringProperty) {
+ setBooleanProperty(booleanProperty);
+ setStringProperty(stringProperty);
+ }
+
+ public TestBean(Boolean booleanSecond, String stringProperty) {
+ setBooleanSecond(booleanSecond.booleanValue());
+ setStringProperty(stringProperty);
+ }
+
+ public TestBean(Integer intProperty) {
+ setIntProperty(intProperty.intValue());
+ }
+
+ public TestBean(double doubleProperty) {
+ setDoubleProperty(doubleProperty);
+ }
+
+ TestBean(int intProperty) {
+ setIntProperty(intProperty);
+ }
+
+ protected TestBean(boolean booleanProperty, boolean booleanSecond, String stringProperty) {
+ setBooleanProperty(booleanProperty);
+ setBooleanSecond(booleanSecond);
+ setStringProperty(stringProperty);
+ }
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ * A boolean property.
+ */
+ private boolean booleanProperty = true;
+
+ public boolean getBooleanProperty() {
+ return (booleanProperty);
+ }
+
+ public void setBooleanProperty(boolean booleanProperty) {
+ this.booleanProperty = booleanProperty;
+ }
+
+
+ /**
+ * A boolean property that uses an "is" method for the getter.
+ */
+ private boolean booleanSecond = true;
+
+ public boolean isBooleanSecond() {
+ return (booleanSecond);
+ }
+
+ public void setBooleanSecond(boolean booleanSecond) {
+ this.booleanSecond = booleanSecond;
+ }
+
+
+ /**
+ * A byte property.
+ */
+ private byte byteProperty = (byte) 121;
+
+ public byte getByteProperty() {
+ return (this.byteProperty);
+ }
+
+ public void setByteProperty(byte byteProperty) {
+ this.byteProperty = byteProperty;
+ }
+
+
+ /**
+ * A double property.
+ */
+ private double doubleProperty = 321.0;
+
+ public double getDoubleProperty() {
+ return (this.doubleProperty);
+ }
+
+ public void setDoubleProperty(double doubleProperty) {
+ this.doubleProperty = doubleProperty;
+ }
+
+
+ /**
+ * An "indexed property" accessible via both array and subscript
+ * based getters and setters.
+ */
+ private String dupProperty[] =
+ { "Dup 0", "Dup 1", "Dup 2", "Dup 3", "Dup 4" };
+
+ public String[] getDupProperty() {
+ return (this.dupProperty);
+ }
+
+ public String getDupProperty(int index) {
+ return (this.dupProperty[index]);
+ }
+
+ public void setDupProperty(int index, String value) {
+ this.dupProperty[index] = value;
+ }
+
+ public void setDupProperty(String dupProperty[]) {
+ this.dupProperty = dupProperty;
+ }
+
+
+ /**
+ * A float property.
+ */
+ private float floatProperty = (float) 123.0;
+
+ public float getFloatProperty() {
+ return (this.floatProperty);
+ }
+
+ public void setFloatProperty(float floatProperty) {
+ this.floatProperty = floatProperty;
+ }
+
+
+ /**
+ * An integer array property accessed as an array.
+ */
+ private int intArray[] = { 0, 10, 20, 30, 40 };
+
+ public int[] getIntArray() {
+ return (this.intArray);
+ }
+
+ public void setIntArray(int intArray[]) {
+ this.intArray = intArray;
+ }
+
+
+ /**
+ * An integer array property accessed as an indexed property.
+ */
+ private int intIndexed[] = { 0, 10, 20, 30, 40 };
+
+ public int getIntIndexed(int index) {
+ return (intIndexed[index]);
+ }
+
+ public void setIntIndexed(int index, int value) {
+ intIndexed[index] = value;
+ }
+
+
+ /**
+ * An integer property.
+ */
+ private int intProperty = 123;
+
+ public int getIntProperty() {
+ return (this.intProperty);
+ }
+
+ public void setIntProperty(int intProperty) {
+ this.intProperty = intProperty;
+ }
+
+
+ /**
+ * A List property accessed as an indexed property.
+ */
+ private static List listIndexed = new ArrayList();
+
+ static {
+ listIndexed.add("String 0");
+ listIndexed.add("String 1");
+ listIndexed.add("String 2");
+ listIndexed.add("String 3");
+ listIndexed.add("String 4");
+ }
+
+ public List getListIndexed() {
+ return (listIndexed);
+ }
+
+
+ /**
+ * A long property.
+ */
+ private long longProperty = 321;
+
+ public long getLongProperty() {
+ return (this.longProperty);
+ }
+
+ public void setLongProperty(long longProperty) {
+ this.longProperty = longProperty;
+ }
+
+
+ /**
+ * A mapped property with only a getter and setter for a Map.
+ */
+ private Map mapProperty = null;
+
+ public Map getMapProperty() {
+ // Create the map the very first time
+ if (mapProperty == null) {
+ mapProperty = new HashMap();
+ mapProperty.put("First Key", "First Value");
+ mapProperty.put("Second Key", "Second Value");
+ }
+ return (mapProperty);
+ }
+
+ public void setMapProperty(Map mapProperty) {
+ // Create the map the very first time
+ if (mapProperty == null) {
+ mapProperty = new HashMap();
+ mapProperty.put("First Key", "First Value");
+ mapProperty.put("Second Key", "Second Value");
+ }
+ this.mapProperty = mapProperty;
+ }
+
+
+ /**
+ * A mapped property that has String keys and Object values.
+ */
+ private HashMap mappedObjects = null;
+
+ public Object getMappedObjects(String key) {
+ // Create the map the very first time
+ if (mappedObjects == null) {
+ mappedObjects = new HashMap();
+ mappedObjects.put("First Key", "First Value");
+ mappedObjects.put("Second Key", "Second Value");
+ }
+ return (mappedObjects.get(key));
+ }
+
+ public void setMappedObjects(String key, Object value) {
+ // Create the map the very first time
+ if (mappedObjects == null) {
+ mappedObjects = new HashMap();
+ mappedObjects.put("First Key", "First Value");
+ mappedObjects.put("Second Key", "Second Value");
+ }
+ mappedObjects.put(key, value);
+ }
+
+
+ /**
+ * A mapped property that has String keys and String values.
+ */
+ private HashMap mappedProperty = null;
+
+ public String getMappedProperty(String key) {
+ // Create the map the very first time
+ if (mappedProperty == null) {
+ mappedProperty = new HashMap();
+ mappedProperty.put("First Key", "First Value");
+ mappedProperty.put("Second Key", "Second Value");
+ }
+ return ((String) mappedProperty.get(key));
+ }
+
+ public void setMappedProperty(String key, String value) {
+ // Create the map the very first time
+ if (mappedProperty == null) {
+ mappedProperty = new HashMap();
+ mappedProperty.put("First Key", "First Value");
+ mappedProperty.put("Second Key", "Second Value");
+ }
+ mappedProperty.put(key, value);
+ }
+
+
+ /**
+ * A mapped property that has String keys and int values.
+ */
+ private HashMap mappedIntProperty = null;
+
+ public int getMappedIntProperty(String key) {
+ // Create the map the very first time
+ if (mappedProperty == null) {
+ mappedProperty = new HashMap();
+ mappedProperty.put("One", new Integer(1));
+ mappedProperty.put("Two", new Integer(2));
+ }
+ Integer x = (Integer) mappedIntProperty.get(key);
+ return ((x == null) ? 0 : x.intValue());
+ }
+
+ public void setMappedIntProperty(String key, int value) {
+ mappedIntProperty.put(key, new Integer(value));
+ }
+
+
+ /**
+ * A nested reference to another test bean (populated as needed).
+ */
+ private TestBean nested = null;
+
+ public TestBean getNested() {
+ if (nested == null)
+ nested = new TestBean();
+ return (nested);
+ }
+
+ /**
+ * Another nested reference to another test bean,
+ */
+ private TestBean anotherNested = null;
+
+ public TestBean getAnotherNested() {
+ return anotherNested;
+ }
+
+ public void setAnotherNested( TestBean anotherNested ) {
+ this.anotherNested = anotherNested;
+ }
+
+ /*
+ * Another nested reference to a bean containing mapp properties
+ */
+ class MappedTestBean {
+ public void setValue(String key,String val) { }
+ public String getValue(String key) { return "Mapped Value"; }
+ }
+
+ private MappedTestBean mappedNested = null;
+
+ public MappedTestBean getMappedNested() {
+ if (mappedNested == null)
+ {
+ mappedNested = new MappedTestBean();
+ }
+ return mappedNested;
+ }
+
+ /**
+ * A String property with an initial value of null.
+ */
+ private String nullProperty = null;
+
+ public String getNullProperty() {
+ return (this.nullProperty);
+ }
+
+ public void setNullProperty(String nullProperty) {
+ this.nullProperty = nullProperty;
+ }
+
+
+ /**
+ * A read-only String property.
+ */
+ private String readOnlyProperty = "Read Only String Property";
+
+ public String getReadOnlyProperty() {
+ return (this.readOnlyProperty);
+ }
+
+
+ /**
+ * A short property.
+ */
+ private short shortProperty = (short) 987;
+
+ public short getShortProperty() {
+ return (this.shortProperty);
+ }
+
+ public void setShortProperty(short shortProperty) {
+ this.shortProperty = shortProperty;
+ }
+
+
+ /**
+ * A String array property accessed as a String.
+ */
+ private String stringArray[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+
+ public String[] getStringArray() {
+ return (this.stringArray);
+ }
+
+ public void setStringArray(String stringArray[]) {
+ this.stringArray = stringArray;
+ }
+
+
+ /**
+ * A String array property accessed as an indexed property.
+ */
+ private String stringIndexed[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+
+ public String getStringIndexed(int index) {
+ return (stringIndexed[index]);
+ }
+
+ public void setStringIndexed(int index, String value) {
+ stringIndexed[index] = value;
+ }
+
+
+ /**
+ * A String property.
+ */
+ private String stringProperty = "This is a string";
+
+ public String getStringProperty() {
+ return (this.stringProperty);
+ }
+
+ public void setStringProperty(String stringProperty) {
+ this.stringProperty = stringProperty;
+ }
+
+
+ /**
+ * A write-only String property.
+ */
+ private String writeOnlyProperty = "Write Only String Property";
+
+ public String getWriteOnlyPropertyValue() {
+ return (this.writeOnlyProperty);
+ }
+
+ public void setWriteOnlyProperty(String writeOnlyProperty) {
+ this.writeOnlyProperty = writeOnlyProperty;
+ }
+
+
+ // ------------------------------------------------------ Invalid Properties
+
+
+ /**
+ * <p>An invalid property that has two boolean getters (getInvalidBoolean
+ * and isInvalidBoolean) plus a String setter (setInvalidBoolean). By the
+ * rules described in the JavaBeans Specification, this will be considered
+ * a read-only boolean property, using isInvalidBoolean() as the getter.</p>
+ */
+ private boolean invalidBoolean = false;
+
+ public boolean getInvalidBoolean() {
+ return (this.invalidBoolean);
+ }
+
+ public boolean isInvalidBoolean() {
+ return (this.invalidBoolean);
+ }
+
+ public void setInvalidBoolean(String invalidBoolean) {
+ if ("true".equalsIgnoreCase(invalidBoolean) ||
+ "yes".equalsIgnoreCase(invalidBoolean) ||
+ "1".equalsIgnoreCase(invalidBoolean)) {
+ this.invalidBoolean = true;
+ } else {
+ this.invalidBoolean = false;
+ }
+ }
+
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * A static variable that is accessed and updated via static methods
+ * for MethodUtils testing.
+ */
+ private static int counter = 0;
+
+
+ /**
+ * Return the current value of the counter.
+ */
+ public static int currentCounter() {
+
+ return (counter);
+
+ }
+
+
+ /**
+ * Increment the current value of the counter by 1.
+ */
+ public static void incrementCounter() {
+
+ incrementCounter(1);
+
+ }
+
+
+ /**
+ * Increment the current value of the counter by the specified amount.
+ *
+ * @param amount Amount to be added to the current counter
+ */
+ public static void incrementCounter(int amount) {
+
+ counter += amount;
+
+ }
+
+
+}
diff --git a/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/TestBeanMap.java b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/TestBeanMap.java
new file mode 100644
index 0000000..afce252
--- /dev/null
+++ b/trunk/optional/bean-collections/src/test/org/apache/commons/beanutils/TestBeanMap.java
@@ -0,0 +1,344 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.beanutils;
+
+import java.io.Serializable;
+import java.lang.reflect.Method;
+import java.util.Map;
+
+import junit.framework.Test;
+import junit.textui.TestRunner;
+
+import org.apache.commons.collections.map.AbstractTestMap;
+import org.apache.commons.collections.BulkTest;
+
+/**
+ * Test cases for BeanMap
+ *
+ * @version $Revision: 1.1 $ $Date: 2004/05/11 22:46:26 $
+ *
+ * @author Morgan Delagrange
+ * @author Stephen Colebourne
+ */
+public class TestBeanMap extends AbstractTestMap {
+
+ public TestBeanMap(String testName) {
+ super(testName);
+ }
+
+ public static void main(String[] args) {
+ TestRunner.run(suite());
+ }
+
+ public static Test suite() {
+ return BulkTest.makeSuite(TestBeanMap.class);
+ }
+
+/*
+ note to self. The getter and setter methods were generated by copying the
+ field declarations and using the following regular expression search and
+ replace:
+
+ From:
+ private \(.*\) some\(.*\);
+ To:
+ public \1 getSome\2Value() {
+ return some\2;
+ }
+ public void setSome\2Value(\1 value) {
+ some\2 = value;
+ }
+
+ Also note: The sample keys and mappings were generated manually.
+*/
+
+
+ public static class BeanWithProperties implements Serializable {
+ private int someInt;
+ private long someLong;
+ private double someDouble;
+ private float someFloat;
+ private short someShort;
+ private byte someByte;
+ private char someChar;
+ private Integer someInteger;
+ private String someString;
+ private Object someObject;
+
+ public int getSomeIntValue() {
+ return someInt;
+ }
+ public void setSomeIntValue(int value) {
+ someInt = value;
+ }
+
+ public long getSomeLongValue() {
+ return someLong;
+ }
+ public void setSomeLongValue(long value) {
+ someLong = value;
+ }
+
+ public double getSomeDoubleValue() {
+ return someDouble;
+ }
+ public void setSomeDoubleValue(double value) {
+ someDouble = value;
+ }
+
+ public float getSomeFloatValue() {
+ return someFloat;
+ }
+ public void setSomeFloatValue(float value) {
+ someFloat = value;
+ }
+
+ public short getSomeShortValue() {
+ return someShort;
+ }
+ public void setSomeShortValue(short value) {
+ someShort = value;
+ }
+
+ public byte getSomeByteValue() {
+ return someByte;
+ }
+ public void setSomeByteValue(byte value) {
+ someByte = value;
+ }
+
+ public char getSomeCharValue() {
+ return someChar;
+ }
+ public void setSomeCharValue(char value) {
+ someChar = value;
+ }
+
+ public String getSomeStringValue() {
+ return someString;
+ }
+ public void setSomeStringValue(String value) {
+ someString = value;
+ }
+
+ public Integer getSomeIntegerValue() {
+ return someInteger;
+ }
+ public void setSomeIntegerValue(Integer value) {
+ someInteger = value;
+ }
+
+ public Object getSomeObjectValue() {
+ return someObject;
+ }
+ public void setSomeObjectValue(Object value) {
+ someObject = value;
+ }
+ }
+
+ // note to self. The Sample keys were generated by copying the field
+ // declarations and using the following regular expression search and replace:
+ //
+ // From:
+ // private \(.*\) some\(.*\);
+ // To:
+ // "some\2Value",
+ //
+ // Then, I manually added the "class" key, which is a property that exists for
+ // all beans (and all objects for that matter.
+ public Object[] getSampleKeys() {
+ Object[] keys = new Object[] {
+ "someIntValue",
+ "someLongValue",
+ "someDoubleValue",
+ "someFloatValue",
+ "someShortValue",
+ "someByteValue",
+ "someCharValue",
+ "someIntegerValue",
+ "someStringValue",
+ "someObjectValue",
+ "class",
+ };
+ return keys;
+ }
+
+ /**
+ * An object value that will be stored in the bean map as a value. Need
+ * to save this externally so that we can make sure the object instances
+ * are equivalent since getSampleValues() would otherwise construct a new
+ * and different Object each time.
+ **/
+ private Object objectInFullMap = new Object();
+
+ // note to self: the sample values were created manually
+ public Object[] getSampleValues() {
+ Object[] values = new Object[] {
+ new Integer(1234),
+ new Long(1298341928234L),
+ new Double(123423.34),
+ new Float(1213332.12f),
+ new Short((short)134),
+ new Byte((byte)10),
+ new Character('a'),
+ new Integer(1432),
+ "SomeStringValue",
+ objectInFullMap,
+ BeanWithProperties.class,
+ };
+ return values;
+ }
+
+ public Object[] getNewSampleValues() {
+ Object[] values = new Object[] {
+ new Integer(223),
+ new Long(23341928234L),
+ new Double(23423.34),
+ new Float(213332.12f),
+ new Short((short)234),
+ new Byte((byte)20),
+ new Character('b'),
+ new Integer(232),
+ "SomeNewStringValue",
+ new Object(),
+ null,
+ };
+ return values;
+ }
+
+ /**
+ * Values is a dead copy in BeanMap, so refresh each time.
+ */
+ public void verifyValues() {
+ values = map.values();
+ super.verifyValues();
+ }
+
+ /**
+ * The mappings in a BeanMap are fixed on the properties the underlying
+ * bean has. Adding and removing mappings is not possible, thus this
+ * method is overridden to return false.
+ */
+ public boolean isPutAddSupported() {
+ return false;
+ }
+
+ /**
+ * The mappings in a BeanMap are fixed on the properties the underlying
+ * bean has. Adding and removing mappings is not possible, thus this
+ * method is overridden to return false.
+ */
+ public boolean isRemoveSupported() {
+ return false;
+ }
+
+ public Map makeFullMap() {
+ // note: These values must match (i.e. .equals() must return true)
+ // those returned from getSampleValues().
+ BeanWithProperties bean = new BeanWithProperties();
+ bean.setSomeIntValue(1234);
+ bean.setSomeLongValue(1298341928234L);
+ bean.setSomeDoubleValue(123423.34);
+ bean.setSomeFloatValue(1213332.12f);
+ bean.setSomeShortValue((short)134);
+ bean.setSomeByteValue((byte)10);
+ bean.setSomeCharValue('a');
+ bean.setSomeIntegerValue(new Integer(1432));
+ bean.setSomeStringValue("SomeStringValue");
+ bean.setSomeObjectValue(objectInFullMap);
+ return new BeanMap(bean);
+ }
+
+ public Map makeEmptyMap() {
+ return new BeanMap();
+ }
+
+ public String[] ignoredTests() {
+ // Ignore the serialization tests on collection views.
+ return new String[] {
+ "TestBeanMap.bulkTestMapEntrySet.testCanonicalEmptyCollectionExists",
+ "TestBeanMap.bulkTestMapEntrySet.testCanonicalFullCollectionExists",
+ "TestBeanMap.bulkTestMapKeySet.testCanonicalEmptyCollectionExists",
+ "TestBeanMap.bulkTestMapKeySet.testCanonicalFullCollectionExists",
+ "TestBeanMap.bulkTestMapValues.testCanonicalEmptyCollectionExists",
+ "TestBeanMap.bulkTestMapValues.testCanonicalFullCollectionExists",
+ "TestBeanMap.bulkTestMapEntrySet.testSimpleSerialization",
+ "TestBeanMap.bulkTestMapKeySet.testSimpleSerialization",
+ "TestBeanMap.bulkTestMapEntrySet.testSerializeDeserializeThenCompare",
+ "TestBeanMap.bulkTestMapKeySet.testSerializeDeserializeThenCompare"
+ };
+ }
+
+ /**
+ * Need to override this method because the "clear()" method on the bean
+ * map just returns the bean properties to their default states. It does
+ * not actually remove the mappings as per the map contract. The default
+ * testClear() methods checks that the clear method throws an
+ * UnsupportedOperationException since this class is not add/remove
+ * modifiable. In our case though, we do not always throw that exception.
+ */
+ public void testMapClear() {
+ //TODO: make sure a call to BeanMap.clear returns the bean to its
+ //default initialization values.
+ }
+
+ /**
+ * Need to override this method because the "put()" method on the bean
+ * doesn't work for this type of Map.
+ */
+ public void testMapPut() {
+ // see testBeanMapPutAllWriteable
+ }
+
+ public void testBeanMapClone() {
+ BeanMap map = (BeanMap)makeFullMap();
+ try {
+ BeanMap map2 = (BeanMap)((BeanMap)map).clone();
+
+ // make sure containsKey is working to verify the bean was cloned
+ // ok, and the read methods were properly initialized
+ Object[] keys = getSampleKeys();
+ for(int i = 0; i < keys.length; i++) {
+ assertTrue("Cloned BeanMap should contain the same keys",
+ map2.containsKey(keys[i]));
+ }
+ } catch (CloneNotSupportedException exception) {
+ fail("BeanMap.clone() should not throw a " +
+ "CloneNotSupportedException when clone should succeed.");
+ }
+ }
+
+ public void testBeanMapPutAllWriteable() {
+ BeanMap map1 = (BeanMap)makeFullMap();
+ BeanMap map2 = (BeanMap)makeFullMap();
+ map2.put("someIntValue", new Integer(0));
+ map1.putAllWriteable(map2);
+ assertEquals(map1.get("someIntValue"), new Integer(0));
+ }
+
+ public void testMethodAccessor() throws Exception {
+ BeanMap map = (BeanMap) makeFullMap();
+ Method method = BeanWithProperties.class.getDeclaredMethod("getSomeIntegerValue", null);
+ assertEquals(method, map.getReadMethod("someIntegerValue"));
+ }
+
+ public void testMethodMutator() throws Exception {
+ BeanMap map = (BeanMap) makeFullMap();
+ Method method = BeanWithProperties.class.getDeclaredMethod("setSomeIntegerValue", new Class[] {Integer.class});
+ assertEquals(method, map.getWriteMethod("someIntegerValue"));
+ }
+
+}
diff --git a/trunk/optional/bean-collections/xdocs/index.xml b/trunk/optional/bean-collections/xdocs/index.xml
new file mode 100644
index 0000000..f998bf9
--- /dev/null
+++ b/trunk/optional/bean-collections/xdocs/index.xml
@@ -0,0 +1,85 @@
+<?xml version="1.0"?>
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+
+<document>
+
+ <properties>
+ <title>Commons</title>
+ <author email="commons-dev at jakarta.apache.org">Commons Documentation Team</author>
+ </properties>
+
+ <body>
+
+<section name="Commons BeanUtils Bean Collections">
+
+<p>
+Dealing with collections of beans is a common feature of development in Java.
+A lot of time is spent creating bean-specific implementations
+for standard collection manipulators such as functors and comparators.
+</p>
+
+<p>
+<code>BeanUtils Bean collections</code> is a library intended to
+improve developer productivity by using sophisticated bean introspection
+(from <code>BeanUtils</code>)
+to allow general library classes to be used (rather than creating bean specific
+implementation classes).
+</p>
+
+<p>
+But won't this be slower? Yes, reflection is slower than direct references
+but in real life applications, this is typically insignificant. So why not use
+these classes to speed development and then substitute faster implementations only
+where the profiler indicates there is a problem :)
+</p>
+ <subsection name='Quick-Quick Guide To Functors'>
+ <p>
+Many of the classes in <code>bean-collections</code> are functor implementations
+specialized for bean properties. This is a very quick guide (aimed to let java developer
+know why functors make for elegant code).
+ </p>
+ <p>
+The word <code>functor</code> is mathematical in origin but comes into object oriented
+development from functional and logic coding. In Java terms, a functor
+is a function that is encapsulated as an object (and so can be manipulated as an object).
+This allows elegant, concise and powerful techniques to be used. For example,
+<a href='http://jakarta.apache.org/commons/collections'>Commons Collections</a>
+contains utilities that allow functions (as functors) to be easily applied to
+Collections.
+ </p>
+ <p>
+This is actually pretty useful when it comes to collections of beans. It's a common
+problem to want to extract information from a collection of beans or to change all properties
+to a particular value. Functors can be a particularly elegant way to do this. So try them!
+You might just like them!
+ </p>
+ <p>
+For more information about functors, please read the introduction to the
+<a href='http://jakarta.apache.org/commons/sandbox/functor/'>Commons Functor component</a>.
+ </p>
+ </subsection>
+</section>
+<section name='Releases'>
+ <p>
+BeanUtils Bean-Collections is distributed as an optional jar within the main
+beanutils distribution. For details, see the
+<a href='http://jakarta.apache.org/commons/beanutils/index.html'>main BeanUtils website</a>
+ </p>
+</section>
+
+</body>
+</document>
diff --git a/trunk/optional/bean-collections/xdocs/navigation.xml b/trunk/optional/bean-collections/xdocs/navigation.xml
new file mode 100644
index 0000000..a34db5c
--- /dev/null
+++ b/trunk/optional/bean-collections/xdocs/navigation.xml
@@ -0,0 +1,40 @@
+<?xml version="1.0" encoding="ISO-8859-1"?>
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<!DOCTYPE org.apache.commons.menus SYSTEM '../../../../commons-build/menus/menus.dtd'>
+<project name="BeanUtils Bean Collections">
+
+ <title>BeanUtils Bean Collections</title>
+ <organizationLogo href="/images/jakarta-logo-blue.gif">
+ Jakarta
+ </organizationLogo>
+
+ <body>
+ <links>
+ <item name="BeanUtils Core"
+ href="http://jakarta.apache.org/commons/beanutils"/>
+ <item name="Jakarta Commons"
+ href="http://jakarta.apache.org/commons/"/>
+ </links>
+
+ <menu name="Commons BeanUtils Bean Collections">
+ <item name="Overview" href="/index.html"/>
+ </menu>
+
+ &common-menus;
+
+ </body>
+</project>
diff --git a/trunk/project.properties b/trunk/project.properties
new file mode 100644
index 0000000..a3e4ab9
--- /dev/null
+++ b/trunk/project.properties
@@ -0,0 +1,23 @@
+# Copyright 2001-2004 The Apache Software Foundation
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+# documentation properties
+maven.xdoc.jsl=../commons-build/commons-site.jsl
+maven.xdoc.date=left
+maven.xdoc.version=${pom.currentVersion}
+maven.xdoc.developmentProcessUrl=http://jakarta.apache.org/commons/charter.html
+maven.xdoc.poweredby.image=maven-feather.png
+
+
+beanutils.cvs=pserver:anoncvs at cvs.apache.org:/home/cvspublic
\ No newline at end of file
diff --git a/trunk/project.xml b/trunk/project.xml
new file mode 100644
index 0000000..3a6681e
--- /dev/null
+++ b/trunk/project.xml
@@ -0,0 +1,194 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<project>
+ <pomVersion>3</pomVersion>
+ <id>commons-beanutils</id>
+ <name>BeanUtils</name>
+ <currentVersion>1.7</currentVersion>
+ <inceptionYear>2000</inceptionYear>
+ <shortDescription>Commons BeanUtils</shortDescription>
+ <description>BeanUtils provides an easy-to-use but flexible wrapper around reflection and introspection.</description>
+ <logo>/images/logo.png</logo>
+
+ <package>org.apache.commons.beanutils.*</package>
+ <url>http://jakarta.apache.org/commons/beanutils/index.html</url>
+
+ <organization>
+ <name>The Apache Software Foundation</name>
+ <url>http://jakarta.apache.org</url>
+ <logo>http://jakarta.apache.org/images/original-jakarta-logo.gif</logo>
+ </organization>
+
+ <licenses>
+ <license>
+ <name>The Apache Software License, Version 2.0</name>
+ <url>/LICENSE.txt</url>
+ <distribution>repo</distribution>
+ </license>
+ </licenses>
+
+ <gumpRepositoryId>jakarta</gumpRepositoryId>
+ <issueTrackingUrl>http://issues.apache.org/bugzilla/</issueTrackingUrl>
+ <siteAddress>jakarta.apache.org</siteAddress>
+ <siteDirectory>/www/jakarta.apache.org/commons/${pom.artifactId.substring(8)}/</siteDirectory>
+ <distributionDirectory>/www/jakarta.apache.org/builds/jakarta-commons/${pom.artifactId.substring(8)}/</distributionDirectory>
+
+ <repository>
+ <connection>scm:cvs:pserver:anoncvs at cvs.apache.org:/home/cvspublic:jakarta-commons/${pom.artifactId.substring(8)}</connection>
+ <url>http://cvs.apache.org/viewcvs/jakarta-commons/${pom.artifactId.substring(8)}/</url>
+ </repository>
+
+ <mailingLists>
+ <mailingList>
+ <name>Commons Dev List</name>
+ <subscribe>commons-dev-subscribe at jakarta.apache.org</subscribe>
+ <unsubscribe>commons-dev-unsubscribe at jakarta.apache.org</unsubscribe>
+ <archive>http://nagoya.apache.org/eyebrowse/SummarizeList?listName=commons-dev@jakarta.apache.org</archive>
+ </mailingList>
+ <mailingList>
+ <name>Commons User List</name>
+ <subscribe>commons-user-subscribe at jakarta.apache.org</subscribe>
+ <unsubscribe>commons-user-unsubscribe at jakarta.apache.org</unsubscribe>
+ <archive>http://nagoya.apache.org/eyebrowse/SummarizeList?listName=commons-user@jakarta.apache.org</archive>
+ </mailingList>
+ </mailingLists>
+
+ <developers>
+ <developer>
+ <name>Robert Burrell Donkin</name>
+ <id>rdonkin</id>
+ <email>rdonkin at apache.org</email>
+ <organization>Apache Software Foundation</organization>
+ </developer>
+ <developer>
+ <name>dIon Gillard</name>
+ <id>dion</id>
+ <email>dion at apache.org</email>
+ <organization>Apache Software Foundation</organization>
+ </developer>
+ <developer>
+ <name>Craig McClanahan</name>
+ <id>craigmcc</id>
+ <email>craigmcc at apache.org</email>
+ <organization>Apache Software Foundation</organization>
+ </developer>
+ <developer>
+ <name>Geir Magnusson Jr.</name>
+ <id>geirm</id>
+ <email>geirm at apache.org</email>
+ <organization>Apache Software Foundation</organization>
+ </developer>
+ <developer>
+ <name>Scott Sanders</name>
+ <id>sanders</id>
+ <email>sanders at apache.org</email>
+ <organization>Apache Software Foundation</organization>
+ </developer>
+ <developer>
+ <name>James Strachan</name>
+ <id>jstrachan</id>
+ <email>jstrachan at apache.org</email>
+ <organization>Apache Software Foundation</organization>
+ </developer>
+ <developer>
+ <name>Rodney Waldhoff</name>
+ <id>rwaldhoff</id>
+ <email>rwaldhoff at apache.org</email>
+ <organization>Apache Software Foundation</organization>
+ </developer>
+ <developer>
+ <name>Martin van den Bemt</name>
+ <id>mvdb</id>
+ <email>mvdb at apache.org</email>
+ <organization>Apache Software Foundation</organization>
+ </developer>
+ <developer>
+ <name>Yoav Shapira</name>
+ <id>yoavs</id>
+ <email>yoavs at apache.org</email>
+ <organization>Apache Software Foundation</organization>
+ </developer>
+ </developers>
+
+ <contributors>
+ <contributor>
+ <name>Paul Jack</name>
+ <email></email>
+ </contributor>
+ <contributor>
+ <name>Stephen Colebourne</name>
+ <email></email>
+ </contributor>
+ <contributor>
+ <name>Berin Loritsch</name>
+ <email></email>
+ </contributor>
+ </contributors>
+
+ <dependencies>
+ <dependency>
+ <id>commons-logging</id>
+ <version>1.0.3</version>
+ </dependency>
+ </dependencies>
+
+ <build>
+ <nagEmailAddress>commons-dev at jakarta.apache.org</nagEmailAddress>
+ <sourceDirectory>src/java</sourceDirectory>
+ <unitTestSourceDirectory>src/test</unitTestSourceDirectory>
+ <integrationUnitTestSourceDirectory/>
+ <aspectSourceDirectory/>
+
+ <!-- Unit test classes -->
+ <unitTest>
+ <includes>
+ <include>**/*TestCase.java</include>
+ </includes>
+ <resources>
+ <resource>
+ <directory>${pom.build.unitTestSourceDirectory}</directory>
+ <includes>
+ <include>**/*.xml</include>
+ </includes>
+ </resource>
+ </resources>
+ </unitTest>
+
+ <!-- Integration unit test classes -->
+ <integrationUnitTestPatterns></integrationUnitTestPatterns>
+
+ <resources>
+ <includes>
+ <include>**/*.properties</include>
+ </includes>
+ </resources>
+ <jars/>
+ </build>
+
+ <reports>
+ <report>maven-changelog-plugin</report>
+ <report>maven-changes-plugin</report>
+ <report>maven-developer-activity-plugin</report>
+ <report>maven-file-activity-plugin</report>
+ <report>maven-javadoc-plugin</report>
+ <report>maven-jdepend-plugin</report>
+ <report>maven-junit-report-plugin</report>
+ <report>maven-jxr-plugin</report>
+ <report>maven-license-plugin</report>
+ <report>maven-tasklist-plugin</report>
+ </reports>
+</project>
diff --git a/trunk/src/conf/MANIFEST.MF b/trunk/src/conf/MANIFEST.MF
new file mode 100644
index 0000000..19e1075
--- /dev/null
+++ b/trunk/src/conf/MANIFEST.MF
@@ -0,0 +1,7 @@
+Extension-Name: org.apache.commons.beanutils
+Specification-Title: Jakarta Commons Beanutils
+Specification-Vendor: Apache Software Foundation
+Specification-Version: 1.6
+Implementation-Title: org.apache.commons.beanutils
+Implementation-Vendor: Apache Software Foundation
+Implementation-Version: 1.6
diff --git a/trunk/src/java/org/apache/commons/beanutils/BasicDynaBean.java b/trunk/src/java/org/apache/commons/beanutils/BasicDynaBean.java
new file mode 100644
index 0000000..7f7dd4a
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/BasicDynaBean.java
@@ -0,0 +1,402 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.io.Serializable;
+import java.lang.reflect.Array;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+
+/**
+ * <p>Minimal implementation of the <code>DynaBean</code> interface. Can be
+ * used as a convenience base class for more sophisticated implementations.</p>
+ *
+ * <p><strong>IMPLEMENTATION NOTE</strong> - Instances of this class that are
+ * accessed from multiple threads simultaneously need to be synchronized.</p>
+ *
+ * <p><strong>IMPLEMENTATION NOTE</strong> - Instances of this class can be
+ * successfully serialized and deserialized <strong>ONLY</strong> if all
+ * property values are <code>Serializable</code>.</p>
+ *
+ * @author Craig McClanahan
+ * @version $Revision: 1.11 $ $Date: 2004/02/28 13:18:33 $
+ */
+
+public class BasicDynaBean implements DynaBean, Serializable {
+
+
+ // ---------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new <code>DynaBean</code> associated with the specified
+ * <code>DynaClass</code> instance.
+ *
+ * @param dynaClass The DynaClass we are associated with
+ */
+ public BasicDynaBean(DynaClass dynaClass) {
+
+ super();
+ this.dynaClass = dynaClass;
+
+ }
+
+
+ // ---------------------------------------------------- Instance Variables
+
+
+ /**
+ * The <code>DynaClass</code> "base class" that this DynaBean
+ * is associated with.
+ */
+ protected DynaClass dynaClass = null;
+
+
+ /**
+ * The set of property values for this DynaBean, keyed by property name.
+ */
+ protected HashMap values = new HashMap();
+
+
+ // ------------------------------------------------------ DynaBean Methods
+
+
+ /**
+ * Does the specified mapped property contain a value for the specified
+ * key value?
+ *
+ * @param name Name of the property to check
+ * @param key Name of the key to check
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public boolean contains(String name, String key) {
+
+ Object value = values.get(name);
+ if (value == null) {
+ throw new NullPointerException
+ ("No mapped value for '" + name + "(" + key + ")'");
+ } else if (value instanceof Map) {
+ return (((Map) value).containsKey(key));
+ } else {
+ throw new IllegalArgumentException
+ ("Non-mapped property for '" + name + "(" + key + ")'");
+ }
+
+ }
+
+
+ /**
+ * Return the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public Object get(String name) {
+
+ // Return any non-null value for the specified property
+ Object value = values.get(name);
+ if (value != null) {
+ return (value);
+ }
+
+ // Return a null value for a non-primitive property
+ Class type = getDynaProperty(name).getType();
+ if (!type.isPrimitive()) {
+ return (value);
+ }
+
+ // Manufacture default values for primitive properties
+ if (type == Boolean.TYPE) {
+ return (Boolean.FALSE);
+ } else if (type == Byte.TYPE) {
+ return (new Byte((byte) 0));
+ } else if (type == Character.TYPE) {
+ return (new Character((char) 0));
+ } else if (type == Double.TYPE) {
+ return (new Double((double) 0.0));
+ } else if (type == Float.TYPE) {
+ return (new Float((float) 0.0));
+ } else if (type == Integer.TYPE) {
+ return (new Integer((int) 0));
+ } else if (type == Long.TYPE) {
+ return (new Long((int) 0));
+ } else if (type == Short.TYPE) {
+ return (new Short((short) 0));
+ } else {
+ return (null);
+ }
+
+ }
+
+
+ /**
+ * Return the value of an indexed property with the specified name.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param index Index of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ * @exception NullPointerException if no array or List has been
+ * initialized for this property
+ */
+ public Object get(String name, int index) {
+
+ Object value = values.get(name);
+ if (value == null) {
+ throw new NullPointerException
+ ("No indexed value for '" + name + "[" + index + "]'");
+ } else if (value.getClass().isArray()) {
+ return (Array.get(value, index));
+ } else if (value instanceof List) {
+ return ((List) value).get(index);
+ } else {
+ throw new IllegalArgumentException
+ ("Non-indexed property for '" + name + "[" + index + "]'");
+ }
+
+ }
+
+
+ /**
+ * Return the value of a mapped property with the specified name,
+ * or <code>null</code> if there is no value for the specified key.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param key Key of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public Object get(String name, String key) {
+
+ Object value = values.get(name);
+ if (value == null) {
+ throw new NullPointerException
+ ("No mapped value for '" + name + "(" + key + ")'");
+ } else if (value instanceof Map) {
+ return (((Map) value).get(key));
+ } else {
+ throw new IllegalArgumentException
+ ("Non-mapped property for '" + name + "(" + key + ")'");
+ }
+
+ }
+
+
+ /**
+ * Return the <code>DynaClass</code> instance that describes the set of
+ * properties available for this DynaBean.
+ */
+ public DynaClass getDynaClass() {
+
+ return (this.dynaClass);
+
+ }
+
+
+ /**
+ * Remove any existing value for the specified key on the
+ * specified mapped property.
+ *
+ * @param name Name of the property for which a value is to
+ * be removed
+ * @param key Key of the value to be removed
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public void remove(String name, String key) {
+
+ Object value = values.get(name);
+ if (value == null) {
+ throw new NullPointerException
+ ("No mapped value for '" + name + "(" + key + ")'");
+ } else if (value instanceof Map) {
+ ((Map) value).remove(key);
+ } else {
+ throw new IllegalArgumentException
+ ("Non-mapped property for '" + name + "(" + key + ")'");
+ }
+
+ }
+
+
+ /**
+ * Set the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception NullPointerException if an attempt is made to set a
+ * primitive property to null
+ */
+ public void set(String name, Object value) {
+
+ DynaProperty descriptor = getDynaProperty(name);
+ if (value == null) {
+ if (descriptor.getType().isPrimitive()) {
+ throw new NullPointerException
+ ("Primitive value for '" + name + "'");
+ }
+ } else if (!isAssignable(descriptor.getType(), value.getClass())) {
+ throw new ConversionException
+ ("Cannot assign value of type '" +
+ value.getClass().getName() +
+ "' to property '" + name + "' of type '" +
+ descriptor.getType().getName() + "'");
+ }
+ values.put(name, value);
+
+ }
+
+
+ /**
+ * Set the value of an indexed property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param index Index of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ */
+ public void set(String name, int index, Object value) {
+
+ Object prop = values.get(name);
+ if (prop == null) {
+ throw new NullPointerException
+ ("No indexed value for '" + name + "[" + index + "]'");
+ } else if (prop.getClass().isArray()) {
+ Array.set(prop, index, value);
+ } else if (prop instanceof List) {
+ try {
+ ((List) prop).set(index, value);
+ } catch (ClassCastException e) {
+ throw new ConversionException(e.getMessage());
+ }
+ } else {
+ throw new IllegalArgumentException
+ ("Non-indexed property for '" + name + "[" + index + "]'");
+ }
+
+ }
+
+
+ /**
+ * Set the value of a mapped property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param key Key of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public void set(String name, String key, Object value) {
+
+ Object prop = values.get(name);
+ if (prop == null) {
+ throw new NullPointerException
+ ("No mapped value for '" + name + "(" + key + ")'");
+ } else if (prop instanceof Map) {
+ ((Map) prop).put(key, value);
+ } else {
+ throw new IllegalArgumentException
+ ("Non-mapped property for '" + name + "(" + key + ")'");
+ }
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * Return the property descriptor for the specified property name.
+ *
+ * @param name Name of the property for which to retrieve the descriptor
+ *
+ * @exception IllegalArgumentException if this is not a valid property
+ * name for our DynaClass
+ */
+ protected DynaProperty getDynaProperty(String name) {
+
+ DynaProperty descriptor = getDynaClass().getDynaProperty(name);
+ if (descriptor == null) {
+ throw new IllegalArgumentException
+ ("Invalid property name '" + name + "'");
+ }
+ return (descriptor);
+
+ }
+
+
+ /**
+ * Is an object of the source class assignable to the destination class?
+ *
+ * @param dest Destination class
+ * @param source Source class
+ */
+ protected boolean isAssignable(Class dest, Class source) {
+
+ if (dest.isAssignableFrom(source) ||
+ ((dest == Boolean.TYPE) && (source == Boolean.class)) ||
+ ((dest == Byte.TYPE) && (source == Byte.class)) ||
+ ((dest == Character.TYPE) && (source == Character.class)) ||
+ ((dest == Double.TYPE) && (source == Double.class)) ||
+ ((dest == Float.TYPE) && (source == Float.class)) ||
+ ((dest == Integer.TYPE) && (source == Integer.class)) ||
+ ((dest == Long.TYPE) && (source == Long.class)) ||
+ ((dest == Short.TYPE) && (source == Short.class))) {
+ return (true);
+ } else {
+ return (false);
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/BasicDynaClass.java b/trunk/src/java/org/apache/commons/beanutils/BasicDynaClass.java
new file mode 100644
index 0000000..7ab61fe
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/BasicDynaClass.java
@@ -0,0 +1,290 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.io.Serializable;
+import java.lang.reflect.Constructor;
+import java.lang.reflect.InvocationTargetException;
+import java.util.HashMap;
+
+
+/**
+ * <p>Minimal implementation of the <code>DynaClass</code> interface. Can be
+ * used as a convenience base class for more sophisticated implementations.</p> *
+ * <p><strong>IMPLEMENTATION NOTE</strong> - The <code>DynaBean</code>
+ * implementation class supplied to our constructor MUST have a one-argument
+ * constructor of its own that accepts a <code>DynaClass</code>. This is
+ * used to associate the DynaBean instance with this DynaClass.</p>
+ *
+ * @author Craig McClanahan
+ * @version $Revision: 1.11 $ $Date: 2004/02/28 13:18:33 $
+ */
+
+public class BasicDynaClass implements DynaClass, Serializable {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new BasicDynaClass with default parameters.
+ */
+ public BasicDynaClass() {
+
+ this(null, null, null);
+
+ }
+
+
+ /**
+ * Construct a new BasicDynaClass with the specified parameters.
+ *
+ * @param name Name of this DynaBean class
+ * @param dynaBeanClass The implementation class for new instances
+ */
+ public BasicDynaClass(String name, Class dynaBeanClass) {
+
+ this(name, dynaBeanClass, null);
+
+ }
+
+
+ /**
+ * Construct a new BasicDynaClass with the specified parameters.
+ *
+ * @param name Name of this DynaBean class
+ * @param dynaBeanClass The implementation class for new intances
+ * @param properties Property descriptors for the supported properties
+ */
+ public BasicDynaClass(String name, Class dynaBeanClass,
+ DynaProperty properties[]) {
+
+ super();
+ if (name != null)
+ this.name = name;
+ if (dynaBeanClass == null)
+ dynaBeanClass = BasicDynaBean.class;
+ setDynaBeanClass(dynaBeanClass);
+ if (properties != null)
+ setProperties(properties);
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The constructor of the <code>dynaBeanClass</code> that we will use
+ * for creating new instances.
+ */
+ protected transient Constructor constructor = null;
+
+
+ /**
+ * The method signature of the constructor we will use to create
+ * new DynaBean instances.
+ */
+ protected static Class constructorTypes[] = { DynaClass.class };
+
+
+ /**
+ * The argument values to be passed to the constructore we will use
+ * to create new DynaBean instances.
+ */
+ protected Object constructorValues[] = { this };
+
+
+ /**
+ * The <code>DynaBean</code> implementation class we will use for
+ * creating new instances.
+ */
+ protected Class dynaBeanClass = BasicDynaBean.class;
+
+
+ /**
+ * The "name" of this DynaBean class.
+ */
+ protected String name = this.getClass().getName();
+
+
+ /**
+ * The set of dynamic properties that are part of this DynaClass.
+ */
+ protected DynaProperty properties[] = new DynaProperty[0];
+
+
+ /**
+ * The set of dynamic properties that are part of this DynaClass,
+ * keyed by the property name. Individual descriptor instances will
+ * be the same instances as those in the <code>properties</code> list.
+ */
+ protected HashMap propertiesMap = new HashMap();
+
+
+ // ------------------------------------------------------ DynaClass Methods
+
+
+ /**
+ * Return the name of this DynaClass (analogous to the
+ * <code>getName()</code> method of <code>java.lang.Class</code), which
+ * allows the same <code>DynaClass</code> implementation class to support
+ * different dynamic classes, with different sets of properties.
+ */
+ public String getName() {
+
+ return (this.name);
+
+ }
+
+
+ /**
+ * Return a property descriptor for the specified property, if it exists;
+ * otherwise, return <code>null</code>.
+ *
+ * @param name Name of the dynamic property for which a descriptor
+ * is requested
+ *
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public DynaProperty getDynaProperty(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException
+ ("No property name specified");
+ }
+ return ((DynaProperty) propertiesMap.get(name));
+
+ }
+
+
+ /**
+ * <p>Return an array of <code>ProperyDescriptors</code> for the properties
+ * currently defined in this DynaClass. If no properties are defined, a
+ * zero-length array will be returned.</p>
+ *
+ * <p><strong>FIXME</strong> - Should we really be implementing
+ * <code>getBeanInfo()</code> instead, which returns property descriptors
+ * and a bunch of other stuff?</p>
+ */
+ public DynaProperty[] getDynaProperties() {
+
+ return (properties);
+
+ }
+
+
+ /**
+ * Instantiate and return a new DynaBean instance, associated
+ * with this DynaClass.
+ *
+ * @exception IllegalAccessException if the Class or the appropriate
+ * constructor is not accessible
+ * @exception InstantiationException if this Class represents an abstract
+ * class, an array class, a primitive type, or void; or if instantiation
+ * fails for some other reason
+ */
+ public DynaBean newInstance()
+ throws IllegalAccessException, InstantiationException {
+
+ try {
+ // Refind the constructor after a deserialization (if needed)
+ if (constructor == null) {
+ setDynaBeanClass(this.dynaBeanClass);
+ }
+ // Invoke the constructor to create a new bean instance
+ return ((DynaBean) constructor.newInstance(constructorValues));
+ } catch (InvocationTargetException e) {
+ throw new InstantiationException
+ (e.getTargetException().getMessage());
+ }
+
+ }
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Return the Class object we will use to create new instances in the
+ * <code>newInstance()</code> method. This Class <strong>MUST</strong>
+ * implement the <code>DynaBean</code> interface.
+ */
+ public Class getDynaBeanClass() {
+
+ return (this.dynaBeanClass);
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * Set the Class object we will use to create new instances in the
+ * <code>newInstance()</code> method. This Class <strong>MUST</strong>
+ * implement the <code>DynaBean</code> interface.
+ *
+ * @param dynaBeanClass The new Class object
+ *
+ * @exception IllegalArgumentException if the specified Class does not
+ * implement the <code>DynaBean</code> interface
+ */
+ protected void setDynaBeanClass(Class dynaBeanClass) {
+
+ // Validate the argument type specified
+ if (dynaBeanClass.isInterface())
+ throw new IllegalArgumentException
+ ("Class " + dynaBeanClass.getName() +
+ " is an interface, not a class");
+ if (!DynaBean.class.isAssignableFrom(dynaBeanClass))
+ throw new IllegalArgumentException
+ ("Class " + dynaBeanClass.getName() +
+ " does not implement DynaBean");
+
+ // Identify the Constructor we will use in newInstance()
+ try {
+ this.constructor = dynaBeanClass.getConstructor(constructorTypes);
+ } catch (NoSuchMethodException e) {
+ throw new IllegalArgumentException
+ ("Class " + dynaBeanClass.getName() +
+ " does not have an appropriate constructor");
+ }
+ this.dynaBeanClass = dynaBeanClass;
+
+ }
+
+
+ /**
+ * Set the list of dynamic properties supported by this DynaClass.
+ *
+ * @param properties List of dynamic properties to be supported
+ */
+ protected void setProperties(DynaProperty properties[]) {
+
+ this.properties = properties;
+ propertiesMap.clear();
+ for (int i = 0; i < properties.length; i++) {
+ propertiesMap.put(properties[i].getName(), properties[i]);
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/BeanAccessLanguageException.java b/trunk/src/java/org/apache/commons/beanutils/BeanAccessLanguageException.java
new file mode 100644
index 0000000..9c508ab
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/BeanAccessLanguageException.java
@@ -0,0 +1,47 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+/**
+ * Thrown to indicate that the <em>Bean Access Language</em> cannot execute query
+ * against given bean. This is a runtime exception and access langauges are encouraged
+ * to subclass to create custom exceptions whenever appropriate.
+ *
+ * @author Robert Burrell Donkin
+ * @since 1.7
+ */
+
+public class BeanAccessLanguageException extends IllegalArgumentException {
+
+ // --------------------------------------------------------- Constuctors
+
+ /**
+ * Constructs a <code>BeanAccessLanguageException</code> without a detail message.
+ */
+ public BeanAccessLanguageException() {
+ super();
+ }
+
+ /**
+ * Constructs a <code>BeanAccessLanguageException</code> without a detail message.
+ *
+ * @param message the detail message explaining this exception
+ */
+ public BeanAccessLanguageException(String message) {
+ super(message);
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/BeanUtils.java b/trunk/src/java/org/apache/commons/beanutils/BeanUtils.java
new file mode 100644
index 0000000..ffcf3f0
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/BeanUtils.java
@@ -0,0 +1,315 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+import java.lang.reflect.InvocationTargetException;
+import java.util.Map;
+
+import org.apache.commons.collections.FastHashMap;
+
+
+/**
+ * <p>Utility methods for populating JavaBeans properties via reflection.</p>
+ *
+ * <p>The implementations are provided by {@link BeanUtilsBean}.
+ * These static utility methods use the default instance.
+ * More sophisticated behaviour can be provided by using a <code>BeanUtilsBean</code> instance.</p>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @author Rey Fran�ois
+ * @author Gregor Ra�man
+ * @version $Revision: 1.40 $ $Date: 2004/02/28 13:18:33 $
+ * @see BeanUtilsBean
+ */
+
+public class BeanUtils {
+
+
+ // ------------------------------------------------------ Private Variables
+
+
+ /**
+ * Dummy collection from the Commons Collections API, to force a
+ * ClassNotFoundException if commons-collections.jar is not present in the
+ * runtime classpath, and this class is the first one referenced.
+ * Otherwise, the ClassNotFoundException thrown by ConvertUtils or
+ * PropertyUtils can get masked.
+ */
+ private static FastHashMap dummy = new FastHashMap();
+
+ /**
+ * The debugging detail level for this component.
+ * @deprecated BeanUtils now uses commons-logging for all log messages.
+ * Use your favorite logging tool to configure logging for
+ * this class.
+ */
+ private static int debug = 0;
+
+ /**
+ * @deprecated BeanUtils now uses commons-logging for all log messages.
+ * Use your favorite logging tool to configure logging for
+ * this class.
+ */
+ public static int getDebug() {
+ return (debug);
+ }
+
+ /**
+ * @deprecated BeanUtils now uses commons-logging for all log messages.
+ * Use your favorite logging tool to configure logging for
+ * this class.
+ */
+ public static void setDebug(int newDebug) {
+ debug = newDebug;
+ }
+
+ // --------------------------------------------------------- Class Methods
+
+
+ /**
+ * <p>Clone a bean based on the available property getters and setters,
+ * even if the bean class itself does not implement Cloneable.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#cloneBean
+ */
+ public static Object cloneBean(Object bean)
+ throws IllegalAccessException, InstantiationException,
+ InvocationTargetException, NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().cloneBean(bean);
+
+ }
+
+
+ /**
+ * <p>Copy property values from the origin bean to the destination bean
+ * for all cases where the property names are the same.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#copyProperties
+ */
+ public static void copyProperties(Object dest, Object orig)
+ throws IllegalAccessException, InvocationTargetException {
+
+ BeanUtilsBean.getInstance().copyProperties(dest, orig);
+ }
+
+
+ /**
+ * <p>Copy the specified property value to the specified destination bean,
+ * performing any type conversion that is required.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#copyProperty
+ */
+ public static void copyProperty(Object bean, String name, Object value)
+ throws IllegalAccessException, InvocationTargetException {
+
+ BeanUtilsBean.getInstance().copyProperty(bean, name, value);
+ }
+
+
+ /**
+ * <p>Return the entire set of properties for which the specified bean
+ * provides a read method.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#describe
+ */
+ public static Map describe(Object bean)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().describe(bean);
+ }
+
+
+ /**
+ * <p>Return the value of the specified array property of the specified
+ * bean, as a String array.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#getArrayProperty
+ */
+ public static String[] getArrayProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().getArrayProperty(bean, name);
+ }
+
+
+ /**
+ * <p>Return the value of the specified indexed property of the specified
+ * bean, as a String.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#getIndexedProperty(Object, String)
+ */
+ public static String getIndexedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().getIndexedProperty(bean, name);
+
+ }
+
+
+ /**
+ * Return the value of the specified indexed property of the specified
+ * bean, as a String. The index is specified as a method parameter and
+ * must *not* be included in the property name expression
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#getIndexedProperty(Object, String, int)
+ */
+ public static String getIndexedProperty(Object bean,
+ String name, int index)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().getIndexedProperty(bean, name, index);
+
+ }
+
+
+ /**
+ * </p>Return the value of the specified indexed property of the specified
+ * bean, as a String.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#getMappedProperty(Object, String)
+ */
+ public static String getMappedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().getMappedProperty(bean, name);
+
+ }
+
+
+ /**
+ * </p>Return the value of the specified mapped property of the specified
+ * bean, as a String.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#getMappedProperty(Object, String, String)
+ */
+ public static String getMappedProperty(Object bean,
+ String name, String key)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().getMappedProperty(bean, name, key);
+
+ }
+
+
+ /**
+ * <p>Return the value of the (possibly nested) property of the specified
+ * name, for the specified bean, as a String.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#getNestedProperty
+ */
+ public static String getNestedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().getNestedProperty(bean, name);
+
+ }
+
+
+ /**
+ * <p>Return the value of the specified property of the specified bean,
+ * no matter which property reference format is used, as a String.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#getProperty
+ */
+ public static String getProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().getProperty(bean, name);
+
+ }
+
+
+ /**
+ * <p>Return the value of the specified simple property of the specified
+ * bean, converted to a String.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#getSimpleProperty
+ */
+ public static String getSimpleProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return BeanUtilsBean.getInstance().getSimpleProperty(bean, name);
+
+ }
+
+
+ /**
+ * <p>Populate the JavaBeans properties of the specified bean, based on
+ * the specified name/value pairs.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#populate
+ */
+ public static void populate(Object bean, Map properties)
+ throws IllegalAccessException, InvocationTargetException {
+
+ BeanUtilsBean.getInstance().populate(bean, properties);
+ }
+
+
+ /**
+ * <p>Set the specified property value, performing type conversions as
+ * required to conform to the type of the destination property.</p>
+ *
+ * <p>For more details see <code>BeanUtilsBean</code>.</p>
+ *
+ * @see BeanUtilsBean#setProperty
+ */
+ public static void setProperty(Object bean, String name, Object value)
+ throws IllegalAccessException, InvocationTargetException {
+
+ BeanUtilsBean.getInstance().setProperty(bean, name, value);
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/BeanUtilsBean.java b/trunk/src/java/org/apache/commons/beanutils/BeanUtilsBean.java
new file mode 100644
index 0000000..5026046
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/BeanUtilsBean.java
@@ -0,0 +1,1075 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.beans.IndexedPropertyDescriptor;
+import java.beans.PropertyDescriptor;
+import java.lang.reflect.Array;
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Method;
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.Map;
+import java.util.WeakHashMap;
+
+import org.apache.commons.collections.FastHashMap;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+
+/**
+ * <p>JavaBean property population methods.</p>
+ *
+ * <p>This class provides implementations for the utility methods in
+ * {@link BeanUtils}.
+ * Different instances can be used to isolate caches between classloaders
+ * and to vary the value converters registered.</p>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @author Rey Fran�ois
+ * @author Gregor Ra�man
+ * @version $Revision: 1.16 $ $Date: 2004/02/28 13:18:33 $
+ * @see BeanUtils
+ * @since 1.7
+ */
+
+public class BeanUtilsBean {
+
+
+ // ------------------------------------------------------ Private Class Variables
+
+ /**
+ * Contains <code>BeanUtilsBean</code> instances indexed by context classloader.
+ */
+ private static final ContextClassLoaderLocal
+ beansByClassLoader = new ContextClassLoaderLocal() {
+ // Creates the default instance used when the context classloader is unavailable
+ protected Object initialValue() {
+ return new BeanUtilsBean();
+ }
+ };
+
+ /**
+ * Gets the instance which provides the functionality for {@link BeanUtils}.
+ * This is a pseudo-singleton - an single instance is provided per (thread) context classloader.
+ * This mechanism provides isolation for web apps deployed in the same container.
+ */
+ public synchronized static BeanUtilsBean getInstance() {
+ return (BeanUtilsBean) beansByClassLoader.get();
+ }
+
+ /**
+ * Sets the instance which provides the functionality for {@link BeanUtils}.
+ * This is a pseudo-singleton - an single instance is provided per (thread) context classloader.
+ * This mechanism provides isolation for web apps deployed in the same container.
+ */
+ public synchronized static void setInstance(BeanUtilsBean newInstance) {
+ beansByClassLoader.set(newInstance);
+ }
+
+ // --------------------------------------------------------- Attributes
+
+ /**
+ * Logging for this instance
+ */
+ private Log log = LogFactory.getLog(BeanUtils.class);
+
+ /** Used to perform conversions between object types when setting properties */
+ private ConvertUtilsBean convertUtilsBean;
+
+ /** Used to access properties*/
+ private PropertyUtilsBean propertyUtilsBean;
+
+ // --------------------------------------------------------- Constuctors
+
+ /**
+ * <p>Constructs an instance using new property
+ * and conversion instances.</p>
+ */
+ public BeanUtilsBean() {
+ this(new ConvertUtilsBean(), new PropertyUtilsBean());
+ }
+
+ /**
+ * <p>Constructs an instance using given property and conversion instances.</p>
+ *
+ * @param convertUtilsBean use this <code>ConvertUtilsBean</code>
+ * to perform conversions from one object to another
+ * @param propertyUtilsBean use this <code>PropertyUtilsBean</code>
+ * to access properties
+ */
+ public BeanUtilsBean(
+ ConvertUtilsBean convertUtilsBean,
+ PropertyUtilsBean propertyUtilsBean) {
+
+ this.convertUtilsBean = convertUtilsBean;
+ this.propertyUtilsBean = propertyUtilsBean;
+ }
+
+ // --------------------------------------------------------- Public Methods
+
+ /**
+ * <p>Clone a bean based on the available property getters and setters,
+ * even if the bean class itself does not implement Cloneable.</p>
+ *
+ * <p>
+ * <strong>Note:</strong> this method creates a <strong>shallow</strong> clone.
+ * In other words, any objects referred to by the bean are shared with the clone
+ * rather than being cloned in turn.
+ * </p>
+ *
+ * @param bean Bean to be cloned
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InstantiationException if a new instance of the bean's
+ * class cannot be instantiated
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public Object cloneBean(Object bean)
+ throws IllegalAccessException, InstantiationException,
+ InvocationTargetException, NoSuchMethodException {
+
+ if (log.isDebugEnabled()) {
+ log.debug("Cloning bean: " + bean.getClass().getName());
+ }
+ Class clazz = bean.getClass();
+ Object newBean = null;
+ if (bean instanceof DynaBean) {
+ newBean = ((DynaBean) bean).getDynaClass().newInstance();
+ } else {
+ newBean = bean.getClass().newInstance();
+ }
+ getPropertyUtils().copyProperties(newBean, bean);
+ return (newBean);
+
+ }
+
+
+ /**
+ * <p>Copy property values from the origin bean to the destination bean
+ * for all cases where the property names are the same. For each
+ * property, a conversion is attempted as necessary. All combinations of
+ * standard JavaBeans and DynaBeans as origin and destination are
+ * supported. Properties that exist in the origin bean, but do not exist
+ * in the destination bean (or are read-only in the destination bean) are
+ * silently ignored.</p>
+ *
+ * <p>If the origin "bean" is actually a <code>Map</code>, it is assumed
+ * to contain String-valued <strong>simple</strong> property names as the keys, pointing at
+ * the corresponding property values that will be converted (if necessary)
+ * and set in the destination bean. <strong>Note</strong> that this method
+ * is intended to perform a "shallow copy" of the properties and so complex
+ * properties (for example, nested ones) will not be copied.</p>
+ *
+ * <p>This method differs from <code>populate()</code>, which
+ * was primarily designed for populating JavaBeans from the map of request
+ * parameters retrieved on an HTTP request, is that no scalar->indexed
+ * or indexed->scalar manipulations are performed. If the origin property
+ * is indexed, the destination property must be also.</p>
+ *
+ * <p>If you know that no type conversions are required, the
+ * <code>copyProperties()</code> method in {@link PropertyUtils} will
+ * execute faster than this method.</p>
+ *
+ * <p><strong>FIXME</strong> - Indexed and mapped properties that do not
+ * have getter and setter methods for the underlying array or Map are not
+ * copied by this method.</p>
+ *
+ * @param dest Destination bean whose properties are modified
+ * @param orig Origin bean whose properties are retrieved
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if the <code>dest</code> or
+ * <code>orig</code> argument is null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ public void copyProperties(Object dest, Object orig)
+ throws IllegalAccessException, InvocationTargetException {
+
+ // Validate existence of the specified beans
+ if (dest == null) {
+ throw new IllegalArgumentException
+ ("No destination bean specified");
+ }
+ if (orig == null) {
+ throw new IllegalArgumentException("No origin bean specified");
+ }
+ if (log.isDebugEnabled()) {
+ log.debug("BeanUtils.copyProperties(" + dest + ", " +
+ orig + ")");
+ }
+
+ // Copy the properties, converting as necessary
+ if (orig instanceof DynaBean) {
+ DynaProperty origDescriptors[] =
+ ((DynaBean) orig).getDynaClass().getDynaProperties();
+ for (int i = 0; i < origDescriptors.length; i++) {
+ String name = origDescriptors[i].getName();
+ if (getPropertyUtils().isWriteable(dest, name)) {
+ Object value = ((DynaBean) orig).get(name);
+ copyProperty(dest, name, value);
+ }
+ }
+ } else if (orig instanceof Map) {
+ Iterator names = ((Map) orig).keySet().iterator();
+ while (names.hasNext()) {
+ String name = (String) names.next();
+ if (getPropertyUtils().isWriteable(dest, name)) {
+ Object value = ((Map) orig).get(name);
+ copyProperty(dest, name, value);
+ }
+ }
+ } else /* if (orig is a standard JavaBean) */ {
+ PropertyDescriptor origDescriptors[] =
+ getPropertyUtils().getPropertyDescriptors(orig);
+ for (int i = 0; i < origDescriptors.length; i++) {
+ String name = origDescriptors[i].getName();
+ if ("class".equals(name)) {
+ continue; // No point in trying to set an object's class
+ }
+ if (getPropertyUtils().isReadable(orig, name) &&
+ getPropertyUtils().isWriteable(dest, name)) {
+ try {
+ Object value =
+ getPropertyUtils().getSimpleProperty(orig, name);
+ copyProperty(dest, name, value);
+ } catch (NoSuchMethodException e) {
+ ; // Should not happen
+ }
+ }
+ }
+ }
+
+ }
+
+
+ /**
+ * <p>Copy the specified property value to the specified destination bean,
+ * performing any type conversion that is required. If the specified
+ * bean does not have a property of the specified name, or the property
+ * is read only on the destination bean, return without
+ * doing anything. If you have custom destination property types, register
+ * {@link Converter}s for them by calling the <code>register()</code>
+ * method of {@link ConvertUtils}.</p>
+ *
+ * <p><strong>IMPLEMENTATION RESTRICTIONS</strong>:</p>
+ * <ul>
+ * <li>Does not support destination properties that are indexed,
+ * but only an indexed setter (as opposed to an array setter)
+ * is available.</li>
+ * <li>Does not support destination properties that are mapped,
+ * but only a keyed setter (as opposed to a Map setter)
+ * is available.</li>
+ * <li>The desired property type of a mapped setter cannot be
+ * determined (since Maps support any data type), so no conversion
+ * will be performed.</li>
+ * </ul>
+ *
+ * @param bean Bean on which setting is to be performed
+ * @param name Property name (can be nested/indexed/mapped/combo)
+ * @param value Value to be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ public void copyProperty(Object bean, String name, Object value)
+ throws IllegalAccessException, InvocationTargetException {
+
+ // Trace logging (if enabled)
+ if (log.isTraceEnabled()) {
+ StringBuffer sb = new StringBuffer(" copyProperty(");
+ sb.append(bean);
+ sb.append(", ");
+ sb.append(name);
+ sb.append(", ");
+ if (value == null) {
+ sb.append("<NULL>");
+ } else if (value instanceof String) {
+ sb.append((String) value);
+ } else if (value instanceof String[]) {
+ String values[] = (String[]) value;
+ sb.append('[');
+ for (int i = 0; i < values.length; i++) {
+ if (i > 0) {
+ sb.append(',');
+ }
+ sb.append(values[i]);
+ }
+ sb.append(']');
+ } else {
+ sb.append(value.toString());
+ }
+ sb.append(')');
+ log.trace(sb.toString());
+ }
+
+ // Resolve any nested expression to get the actual target bean
+ Object target = bean;
+ int delim = name.lastIndexOf(PropertyUtils.NESTED_DELIM);
+ if (delim >= 0) {
+ try {
+ target =
+ getPropertyUtils().getProperty(bean, name.substring(0, delim));
+ } catch (NoSuchMethodException e) {
+ return; // Skip this property setter
+ }
+ name = name.substring(delim + 1);
+ if (log.isTraceEnabled()) {
+ log.trace(" Target bean = " + target);
+ log.trace(" Target name = " + name);
+ }
+ }
+
+ // Declare local variables we will require
+ String propName = null; // Simple name of target property
+ Class type = null; // Java type of target property
+ int index = -1; // Indexed subscript value (if any)
+ String key = null; // Mapped key value (if any)
+
+ // Calculate the target property name, index, and key values
+ propName = name;
+ int i = propName.indexOf(PropertyUtils.INDEXED_DELIM);
+ if (i >= 0) {
+ int k = propName.indexOf(PropertyUtils.INDEXED_DELIM2);
+ try {
+ index =
+ Integer.parseInt(propName.substring(i + 1, k));
+ } catch (NumberFormatException e) {
+ ;
+ }
+ propName = propName.substring(0, i);
+ }
+ int j = propName.indexOf(PropertyUtils.MAPPED_DELIM);
+ if (j >= 0) {
+ int k = propName.indexOf(PropertyUtils.MAPPED_DELIM2);
+ try {
+ key = propName.substring(j + 1, k);
+ } catch (IndexOutOfBoundsException e) {
+ ;
+ }
+ propName = propName.substring(0, j);
+ }
+
+ // Calculate the target property type
+ if (target instanceof DynaBean) {
+ DynaClass dynaClass = ((DynaBean) target).getDynaClass();
+ DynaProperty dynaProperty = dynaClass.getDynaProperty(propName);
+ if (dynaProperty == null) {
+ return; // Skip this property setter
+ }
+ type = dynaProperty.getType();
+ } else {
+ PropertyDescriptor descriptor = null;
+ try {
+ descriptor =
+ getPropertyUtils().getPropertyDescriptor(target, name);
+ if (descriptor == null) {
+ return; // Skip this property setter
+ }
+ } catch (NoSuchMethodException e) {
+ return; // Skip this property setter
+ }
+ type = descriptor.getPropertyType();
+ if (type == null) {
+ // Most likely an indexed setter on a POJB only
+ if (log.isTraceEnabled()) {
+ log.trace(" target type for property '" +
+ propName + "' is null, so skipping ths setter");
+ }
+ return;
+ }
+ }
+ if (log.isTraceEnabled()) {
+ log.trace(" target propName=" + propName + ", type=" +
+ type + ", index=" + index + ", key=" + key);
+ }
+
+ // Convert the specified value to the required type and store it
+ if (index >= 0) { // Destination must be indexed
+ Converter converter = getConvertUtils().lookup(type.getComponentType());
+ if (converter != null) {
+ log.trace(" USING CONVERTER " + converter);
+ value = converter.convert(type, value);
+ }
+ try {
+ getPropertyUtils().setIndexedProperty(target, propName,
+ index, value);
+ } catch (NoSuchMethodException e) {
+ throw new InvocationTargetException
+ (e, "Cannot set " + propName);
+ }
+ } else if (key != null) { // Destination must be mapped
+ // Maps do not know what the preferred data type is,
+ // so perform no conversions at all
+ // FIXME - should we create or support a TypedMap?
+ try {
+ getPropertyUtils().setMappedProperty(target, propName,
+ key, value);
+ } catch (NoSuchMethodException e) {
+ throw new InvocationTargetException
+ (e, "Cannot set " + propName);
+ }
+ } else { // Destination must be simple
+ Converter converter = getConvertUtils().lookup(type);
+ if (converter != null) {
+ log.trace(" USING CONVERTER " + converter);
+ value = converter.convert(type, value);
+ }
+ try {
+ getPropertyUtils().setSimpleProperty(target, propName, value);
+ } catch (NoSuchMethodException e) {
+ throw new InvocationTargetException
+ (e, "Cannot set " + propName);
+ }
+ }
+
+ }
+
+
+ /**
+ * <p>Return the entire set of properties for which the specified bean
+ * provides a read method. This map contains the to <code>String</code>
+ * converted property values for all properties for which a read method
+ * is provided (i.e. where the getReadMethod() returns non-null).</p>
+ *
+ * <p>This map can be fed back to a call to
+ * <code>BeanUtils.populate()</code> to reconsitute the same set of
+ * properties, modulo differences for read-only and write-only
+ * properties, but only if there are no indexed properties.</p>
+ *
+ * @param bean Bean whose properties are to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public Map describe(Object bean)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ // return (Collections.EMPTY_MAP);
+ return (new java.util.HashMap());
+ }
+
+ if (log.isDebugEnabled()) {
+ log.debug("Describing bean: " + bean.getClass().getName());
+ }
+
+ Map description = new HashMap();
+ if (bean instanceof DynaBean) {
+ DynaProperty descriptors[] =
+ ((DynaBean) bean).getDynaClass().getDynaProperties();
+ for (int i = 0; i < descriptors.length; i++) {
+ String name = descriptors[i].getName();
+ description.put(name, getProperty(bean, name));
+ }
+ } else {
+ PropertyDescriptor descriptors[] =
+ getPropertyUtils().getPropertyDescriptors(bean);
+ for (int i = 0; i < descriptors.length; i++) {
+ String name = descriptors[i].getName();
+ if (descriptors[i].getReadMethod() != null)
+ description.put(name, getProperty(bean, name));
+ }
+ }
+ return (description);
+
+ }
+
+
+ /**
+ * Return the value of the specified array property of the specified
+ * bean, as a String array.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Name of the property to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public String[] getArrayProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getProperty(bean, name);
+ if (value == null) {
+ return (null);
+ } else if (value instanceof Collection) {
+ ArrayList values = new ArrayList();
+ Iterator items = ((Collection) value).iterator();
+ while (items.hasNext()) {
+ Object item = items.next();
+ if (item == null) {
+ values.add((String) null);
+ } else {
+ // convert to string using convert utils
+ values.add(getConvertUtils().convert(item));
+ }
+ }
+ return ((String[]) values.toArray(new String[values.size()]));
+ } else if (value.getClass().isArray()) {
+ int n = Array.getLength(value);
+ String results[] = new String[n];
+ for (int i = 0; i < n; i++) {
+ Object item = Array.get(value, i);
+ if (item == null) {
+ results[i] = null;
+ } else {
+ // convert to string using convert utils
+ results[i] = getConvertUtils().convert(item);
+ }
+ }
+ return (results);
+ } else {
+ String results[] = new String[1];
+ results[0] = value.toString();
+ return (results);
+ }
+
+ }
+
+
+ /**
+ * Return the value of the specified indexed property of the specified
+ * bean, as a String. The zero-relative index of the
+ * required value must be included (in square brackets) as a suffix to
+ * the property name, or <code>IllegalArgumentException</code> will be
+ * thrown.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name <code>propertyname[index]</code> of the property value
+ * to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public String getIndexedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getIndexedProperty(bean, name);
+ return (getConvertUtils().convert(value));
+
+ }
+
+
+ /**
+ * Return the value of the specified indexed property of the specified
+ * bean, as a String. The index is specified as a method parameter and
+ * must *not* be included in the property name expression
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Simple property name of the property value to be extracted
+ * @param index Index of the property value to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public String getIndexedProperty(Object bean,
+ String name, int index)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getIndexedProperty(bean, name, index);
+ return (getConvertUtils().convert(value));
+
+ }
+
+
+ /**
+ * Return the value of the specified indexed property of the specified
+ * bean, as a String. The String-valued key of the required value
+ * must be included (in parentheses) as a suffix to
+ * the property name, or <code>IllegalArgumentException</code> will be
+ * thrown.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name <code>propertyname(index)</code> of the property value
+ * to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public String getMappedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getMappedProperty(bean, name);
+ return (getConvertUtils().convert(value));
+
+ }
+
+
+ /**
+ * Return the value of the specified mapped property of the specified
+ * bean, as a String. The key is specified as a method parameter and
+ * must *not* be included in the property name expression
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Simple property name of the property value to be extracted
+ * @param key Lookup key of the property value to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public String getMappedProperty(Object bean,
+ String name, String key)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getMappedProperty(bean, name, key);
+ return (getConvertUtils().convert(value));
+
+ }
+
+
+ /**
+ * Return the value of the (possibly nested) property of the specified
+ * name, for the specified bean, as a String.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Possibly nested name of the property to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if a nested reference to a
+ * property returns null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public String getNestedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getNestedProperty(bean, name);
+ return (getConvertUtils().convert(value));
+
+ }
+
+
+ /**
+ * Return the value of the specified property of the specified bean,
+ * no matter which property reference format is used, as a String.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Possibly indexed and/or nested name of the property
+ * to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public String getProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return (getNestedProperty(bean, name));
+
+ }
+
+
+ /**
+ * Return the value of the specified simple property of the specified
+ * bean, converted to a String.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Name of the property to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * property cannot be found
+ */
+ public String getSimpleProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getSimpleProperty(bean, name);
+ return (getConvertUtils().convert(value));
+
+ }
+
+
+ /**
+ * <p>Populate the JavaBeans properties of the specified bean, based on
+ * the specified name/value pairs. This method uses Java reflection APIs
+ * to identify corresponding "property setter" method names, and deals
+ * with setter arguments of type <code>String</code>, <code>boolean</code>,
+ * <code>int</code>, <code>long</code>, <code>float</code>, and
+ * <code>double</code>. In addition, array setters for these types (or the
+ * corresponding primitive types) can also be identified.</p>
+ *
+ * <p>The particular setter method to be called for each property is
+ * determined using the usual JavaBeans introspection mechanisms. Thus,
+ * you may identify custom setter methods using a BeanInfo class that is
+ * associated with the class of the bean itself. If no such BeanInfo
+ * class is available, the standard method name conversion ("set" plus
+ * the capitalized name of the property in question) is used.</p>
+ *
+ * <p><strong>NOTE</strong>: It is contrary to the JavaBeans Specification
+ * to have more than one setter method (with different argument
+ * signatures) for the same property.</p>
+ *
+ * <p><strong>WARNING</strong> - The logic of this method is customized
+ * for extracting String-based request parameters from an HTTP request.
+ * It is probably not what you want for general property copying with
+ * type conversion. For that purpose, check out the
+ * <code>copyProperties()</code> method instead.</p>
+ *
+ * @param bean JavaBean whose properties are being populated
+ * @param properties Map keyed by property name, with the
+ * corresponding (String or String[]) value(s) to be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ public void populate(Object bean, Map properties)
+ throws IllegalAccessException, InvocationTargetException {
+
+ // Do nothing unless both arguments have been specified
+ if ((bean == null) || (properties == null)) {
+ return;
+ }
+ if (log.isDebugEnabled()) {
+ log.debug("BeanUtils.populate(" + bean + ", " +
+ properties + ")");
+ }
+
+ // Loop through the property name/value pairs to be set
+ Iterator names = properties.keySet().iterator();
+ while (names.hasNext()) {
+
+ // Identify the property name and value(s) to be assigned
+ String name = (String) names.next();
+ if (name == null) {
+ continue;
+ }
+ Object value = properties.get(name);
+
+ // Perform the assignment for this property
+ setProperty(bean, name, value);
+
+ }
+
+ }
+
+
+ /**
+ * <p>Set the specified property value, performing type conversions as
+ * required to conform to the type of the destination property.</p>
+ *
+ * <p>If the property is read only then the method returns
+ * without throwing an exception.</p>
+ *
+ * <p>If <code>null</code> is passed into a property expecting a primitive value,
+ * then this will be converted as if it were a <code>null</code> string.</p>
+ *
+ * <p><strong>WARNING</strong> - The logic of this method is customized
+ * to meet the needs of <code>populate()</code>, and is probably not what
+ * you want for general property copying with type conversion. For that
+ * purpose, check out the <code>copyProperty()</code> method instead.</p>
+ *
+ * <p><strong>WARNING</strong> - PLEASE do not modify the behavior of this
+ * method without consulting with the Struts developer community. There
+ * are some subtleties to its functionality that are not documented in the
+ * Javadoc description above, yet are vital to the way that Struts utilizes
+ * this method.</p>
+ *
+ * @param bean Bean on which setting is to be performed
+ * @param name Property name (can be nested/indexed/mapped/combo)
+ * @param value Value to be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ public void setProperty(Object bean, String name, Object value)
+ throws IllegalAccessException, InvocationTargetException {
+
+ // Trace logging (if enabled)
+ if (log.isTraceEnabled()) {
+ StringBuffer sb = new StringBuffer(" setProperty(");
+ sb.append(bean);
+ sb.append(", ");
+ sb.append(name);
+ sb.append(", ");
+ if (value == null) {
+ sb.append("<NULL>");
+ } else if (value instanceof String) {
+ sb.append((String) value);
+ } else if (value instanceof String[]) {
+ String values[] = (String[]) value;
+ sb.append('[');
+ for (int i = 0; i < values.length; i++) {
+ if (i > 0) {
+ sb.append(',');
+ }
+ sb.append(values[i]);
+ }
+ sb.append(']');
+ } else {
+ sb.append(value.toString());
+ }
+ sb.append(')');
+ log.trace(sb.toString());
+ }
+
+ // Resolve any nested expression to get the actual target bean
+ Object target = bean;
+ int delim = findLastNestedIndex(name);
+ if (delim >= 0) {
+ try {
+ target =
+ getPropertyUtils().getProperty(bean, name.substring(0, delim));
+ } catch (NoSuchMethodException e) {
+ return; // Skip this property setter
+ }
+ name = name.substring(delim + 1);
+ if (log.isTraceEnabled()) {
+ log.trace(" Target bean = " + target);
+ log.trace(" Target name = " + name);
+ }
+ }
+
+ // Declare local variables we will require
+ String propName = null; // Simple name of target property
+ Class type = null; // Java type of target property
+ int index = -1; // Indexed subscript value (if any)
+ String key = null; // Mapped key value (if any)
+
+ // Calculate the property name, index, and key values
+ propName = name;
+ int i = propName.indexOf(PropertyUtils.INDEXED_DELIM);
+ if (i >= 0) {
+ int k = propName.indexOf(PropertyUtils.INDEXED_DELIM2);
+ try {
+ index =
+ Integer.parseInt(propName.substring(i + 1, k));
+ } catch (NumberFormatException e) {
+ ;
+ }
+ propName = propName.substring(0, i);
+ }
+ int j = propName.indexOf(PropertyUtils.MAPPED_DELIM);
+ if (j >= 0) {
+ int k = propName.indexOf(PropertyUtils.MAPPED_DELIM2);
+ try {
+ key = propName.substring(j + 1, k);
+ } catch (IndexOutOfBoundsException e) {
+ ;
+ }
+ propName = propName.substring(0, j);
+ }
+
+ // Calculate the property type
+ if (target instanceof DynaBean) {
+ DynaClass dynaClass = ((DynaBean) target).getDynaClass();
+ DynaProperty dynaProperty = dynaClass.getDynaProperty(propName);
+ if (dynaProperty == null) {
+ return; // Skip this property setter
+ }
+ type = dynaProperty.getType();
+ } else {
+ PropertyDescriptor descriptor = null;
+ try {
+ descriptor =
+ getPropertyUtils().getPropertyDescriptor(target, name);
+ if (descriptor == null) {
+ return; // Skip this property setter
+ }
+ } catch (NoSuchMethodException e) {
+ return; // Skip this property setter
+ }
+ if (descriptor instanceof MappedPropertyDescriptor) {
+ if (((MappedPropertyDescriptor) descriptor).getMappedWriteMethod() == null) {
+ if (log.isDebugEnabled()) {
+ log.debug("Skipping read-only property");
+ }
+ return; // Read-only, skip this property setter
+ }
+ type = ((MappedPropertyDescriptor) descriptor).
+ getMappedPropertyType();
+ } else if (descriptor instanceof IndexedPropertyDescriptor) {
+ if (((IndexedPropertyDescriptor) descriptor).getIndexedWriteMethod() == null) {
+ if (log.isDebugEnabled()) {
+ log.debug("Skipping read-only property");
+ }
+ return; // Read-only, skip this property setter
+ }
+ type = ((IndexedPropertyDescriptor) descriptor).
+ getIndexedPropertyType();
+ } else {
+ if (descriptor.getWriteMethod() == null) {
+ if (log.isDebugEnabled()) {
+ log.debug("Skipping read-only property");
+ }
+ return; // Read-only, skip this property setter
+ }
+ type = descriptor.getPropertyType();
+ }
+ }
+
+ // Convert the specified value to the required type
+ Object newValue = null;
+ if (type.isArray() && (index < 0)) { // Scalar value into array
+ if (value == null) {
+ String values[] = new String[1];
+ values[0] = (String) value;
+ newValue = getConvertUtils().convert((String[]) values, type);
+ } else if (value instanceof String) {
+ String values[] = new String[1];
+ values[0] = (String) value;
+ newValue = getConvertUtils().convert((String[]) values, type);
+ } else if (value instanceof String[]) {
+ newValue = getConvertUtils().convert((String[]) value, type);
+ } else {
+ newValue = value;
+ }
+ } else if (type.isArray()) { // Indexed value into array
+ if (value instanceof String) {
+ newValue = getConvertUtils().convert((String) value,
+ type.getComponentType());
+ } else if (value instanceof String[]) {
+ newValue = getConvertUtils().convert(((String[]) value)[0],
+ type.getComponentType());
+ } else {
+ newValue = value;
+ }
+ } else { // Value into scalar
+ if ((value instanceof String) || (value == null)) {
+ newValue = getConvertUtils().convert((String) value, type);
+ } else if (value instanceof String[]) {
+ newValue = getConvertUtils().convert(((String[]) value)[0],
+ type);
+ } else if (getConvertUtils().lookup(value.getClass()) != null) {
+ newValue = getConvertUtils().convert(value.toString(), type);
+ } else {
+ newValue = value;
+ }
+ }
+
+ // Invoke the setter method
+ try {
+ if (index >= 0) {
+ getPropertyUtils().setIndexedProperty(target, propName,
+ index, newValue);
+ } else if (key != null) {
+ getPropertyUtils().setMappedProperty(target, propName,
+ key, newValue);
+ } else {
+ getPropertyUtils().setProperty(target, propName, newValue);
+ }
+ } catch (NoSuchMethodException e) {
+ throw new InvocationTargetException
+ (e, "Cannot set " + propName);
+ }
+
+ }
+
+ private int findLastNestedIndex(String expression)
+ {
+ // walk back from the end to the start
+ // and find the first index that
+ int bracketCount = 0;
+ for (int i = expression.length() - 1; i>=0 ; i--) {
+ char at = expression.charAt(i);
+ switch (at) {
+ case PropertyUtils.NESTED_DELIM:
+ if (bracketCount < 1) {
+ return i;
+ }
+ break;
+
+ case PropertyUtils.MAPPED_DELIM:
+ case PropertyUtils.INDEXED_DELIM:
+ // not bothered which
+ --bracketCount;
+ break;
+
+ case PropertyUtils.MAPPED_DELIM2:
+ case PropertyUtils.INDEXED_DELIM2:
+ // not bothered which
+ ++bracketCount;
+ break;
+ }
+ }
+ // can't find any
+ return -1;
+ }
+
+ /**
+ * Gets the <code>ConvertUtilsBean</code> instance used to perform the conversions.
+ */
+ public ConvertUtilsBean getConvertUtils() {
+ return convertUtilsBean;
+ }
+
+ /**
+ * Gets the <code>PropertyUtilsBean</code> instance used to access properties.
+ */
+ public PropertyUtilsBean getPropertyUtils() {
+ return propertyUtilsBean;
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/ConstructorUtils.java b/trunk/src/java/org/apache/commons/beanutils/ConstructorUtils.java
new file mode 100644
index 0000000..944d085
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/ConstructorUtils.java
@@ -0,0 +1,408 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import java.lang.reflect.Constructor;
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Modifier;
+
+/**
+ * <p> Utility reflection methods focussed on constructors, modelled after {@link MethodUtils}. </p>
+ *
+ * <h3>Known Limitations</h3>
+ * <h4>Accessing Public Constructors In A Default Access Superclass</h4>
+ * <p>There is an issue when invoking public constructors contained in a default access superclass.
+ * Reflection locates these constructors fine and correctly assigns them as public.
+ * However, an <code>IllegalAccessException</code> is thrown if the constructors is invoked.</p>
+ *
+ * <p><code>ConstructorUtils</code> contains a workaround for this situation.
+ * It will attempt to call <code>setAccessible</code> on this constructor.
+ * If this call succeeds, then the method can be invoked as normal.
+ * This call will only succeed when the application has sufficient security privilages.
+ * If this call fails then a warning will be logged and the method may fail.</p>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @author Rey Fran�ois
+ * @author Gregor Ra�man
+ * @author Jan Sorensen
+ * @author Robert Burrell Donkin
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:33 $
+ */
+public class ConstructorUtils {
+
+ // --------------------------------------------------------- Private Members
+ /** An empty class array */
+ private static final Class[] emptyClassArray = new Class[0];
+ /** An empty object array */
+ private static final Object[] emptyObjectArray = new Object[0];
+
+ // --------------------------------------------------------- Public Methods
+
+ /**
+ * <p>Convenience method returning new instance of <code>klazz</code> using a single argument constructor.
+ * The formal parameter type is inferred from the actual values of <code>arg</code>.
+ * See {@link #invokeExactConstructor(Class, Object[], Class[])} for more details.</p>
+ *
+ * <p>The signatures should be assignment compatible.</p>
+ *
+ * @param klass the class to be constructed.
+ * @param arg the actual argument
+ * @return new instance of <code>klazz</code>
+ *
+ * @see #invokeConstructor(java.lang.Class, java.lang.Object[], java.lang.Class[])
+ */
+ public static Object invokeConstructor(Class klass, Object arg)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException,
+ InstantiationException {
+
+ Object[] args = { arg };
+ return invokeConstructor(klass, args);
+
+ }
+
+ /**
+ * <p>Returns new instance of <code>klazz</code> created using the actual arguments <code>args</code>.
+ * The formal parameter types are inferred from the actual values of <code>args</code>.
+ * See {@link #invokeExactConstructor(Class, Object[], Class[])} for more details.</p>
+ *
+ * <p>The signatures should be assignment compatible.</p>
+ *
+ * @param klass the class to be constructed.
+ * @param args actual argument array
+ * @return new instance of <code>klazz</code>
+ *
+ * @see #invokeConstructor(java.lang.Class, java.lang.Object[], java.lang.Class[])
+ */
+ public static Object invokeConstructor(Class klass, Object[] args)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException,
+ InstantiationException {
+
+ if (null == args) {
+ args = emptyObjectArray;
+ }
+ int arguments = args.length;
+ Class parameterTypes[] = new Class[arguments];
+ for (int i = 0; i < arguments; i++) {
+ parameterTypes[i] = args[i].getClass();
+ }
+ return invokeConstructor(klass, args, parameterTypes);
+
+ }
+
+ /**
+ * <p>Returns new instance of <code>klazz</code> created using constructor
+ * with signature <code>parameterTypes</code> and actual arguments <code>args</code>.</p>
+ *
+ * <p>The signatures should be assignment compatible.</p>
+ *
+ * @param klass the class to be constructed.
+ * @param args actual argument array
+ * @param parameterTypes parameter types array
+ * @return new instance of <code>klazz</code>
+ *
+ * @throws NoSuchMethodException if matching constructor cannot be found
+ * @throws IllegalAccessException thrown on the constructor's invocation
+ * @throws InvocationTargetException thrown on the constructor's invocation
+ * @throws InstantiationException thrown on the constructor's invocation
+ * @see Constructor#newInstance
+ */
+ public static Object invokeConstructor(
+ Class klass,
+ Object[] args,
+ Class[] parameterTypes)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException,
+ InstantiationException {
+
+ if (parameterTypes == null) {
+ parameterTypes = emptyClassArray;
+ }
+ if (args == null) {
+ args = emptyObjectArray;
+ }
+
+ Constructor ctor =
+ getMatchingAccessibleConstructor(klass, parameterTypes);
+ if (null == ctor) {
+ throw new NoSuchMethodException(
+ "No such accessible constructor on object: " + klass.getName());
+ }
+ return ctor.newInstance(args);
+ }
+
+
+ /**
+ * <p>Convenience method returning new instance of <code>klazz</code> using a single argument constructor.
+ * The formal parameter type is inferred from the actual values of <code>arg</code>.
+ * See {@link #invokeExactConstructor(Class, Object[], Class[])} for more details.</p>
+ *
+ * <p>The signatures should match exactly.</p>
+ *
+ * @param klass the class to be constructed.
+ * @param arg the actual argument
+ * @return new instance of <code>klazz</code>
+ *
+ * @see #invokeExactConstructor(java.lang.Class, java.lang.Object[], java.lang.Class[])
+ */
+ public static Object invokeExactConstructor(Class klass, Object arg)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException,
+ InstantiationException {
+
+ Object[] args = { arg };
+ return invokeExactConstructor(klass, args);
+
+ }
+
+ /**
+ * <p>Returns new instance of <code>klazz</code> created using the actual arguments <code>args</code>.
+ * The formal parameter types are inferred from the actual values of <code>args</code>.
+ * See {@link #invokeExactConstructor(Class, Object[], Class[])} for more details.</p>
+ *
+ * <p>The signatures should match exactly.</p>
+ *
+ * @param klass the class to be constructed.
+ * @param args actual argument array
+ * @return new instance of <code>klazz</code>
+ *
+ * @see #invokeExactConstructor(java.lang.Class, java.lang.Object[], java.lang.Class[])
+ */
+ public static Object invokeExactConstructor(Class klass, Object[] args)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException,
+ InstantiationException {
+ if (null == args) {
+ args = emptyObjectArray;
+ }
+ int arguments = args.length;
+ Class parameterTypes[] = new Class[arguments];
+ for (int i = 0; i < arguments; i++) {
+ parameterTypes[i] = args[i].getClass();
+ }
+ return invokeExactConstructor(klass, args, parameterTypes);
+
+ }
+
+ /**
+ * <p>Returns new instance of <code>klazz</code> created using constructor
+ * with signature <code>parameterTypes</code> and actual arguments
+ * <code>args</code>.</p>
+ *
+ * <p>The signatures should match exactly.</p>
+ *
+ * @param klass the class to be constructed.
+ * @param args actual argument array
+ * @param parameterTypes parameter types array
+ * @return new instance of <code>klazz</code>
+ *
+ * @throws NoSuchMethodException if matching constructor cannot be found
+ * @throws IllegalAccessException thrown on the constructor's invocation
+ * @throws InvocationTargetException thrown on the constructor's invocation
+ * @throws InstantiationException thrown on the constructor's invocation
+ * @see Constructor#newInstance
+ */
+ public static Object invokeExactConstructor(
+ Class klass,
+ Object[] args,
+ Class[] parameterTypes)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException,
+ InstantiationException {
+
+ if (args == null) {
+ args = emptyObjectArray;
+ }
+
+ if (parameterTypes == null) {
+ parameterTypes = emptyClassArray;
+ }
+
+ Constructor ctor = getAccessibleConstructor(klass, parameterTypes);
+ if (null == ctor) {
+ throw new NoSuchMethodException(
+ "No such accessible constructor on object: " + klass.getName());
+ }
+ return ctor.newInstance(args);
+
+ }
+
+ /**
+ * Returns a constructor with single argument.
+ * @param klass the class to be constructed
+ * @return null if matching accessible constructor can not be found.
+ * @see Class#getConstructor
+ * @see #getAccessibleConstructor(java.lang.reflect.Constructor)
+ */
+ public static Constructor getAccessibleConstructor(
+ Class klass,
+ Class parameterType) {
+
+ Class[] parameterTypes = { parameterType };
+ return getAccessibleConstructor(klass, parameterTypes);
+
+ }
+
+ /**
+ * Returns a constructor given a class and signature.
+ * @param klass the class to be constructed
+ * @param parameterTypes the parameter array
+ * @return null if matching accessible constructor can not be found
+ * @see Class#getConstructor
+ * @see #getAccessibleConstructor(java.lang.reflect.Constructor)
+ */
+ public static Constructor getAccessibleConstructor(
+ Class klass,
+ Class[] parameterTypes) {
+
+ try {
+ return getAccessibleConstructor(
+ klass.getConstructor(parameterTypes));
+ } catch (NoSuchMethodException e) {
+ return (null);
+ }
+
+ }
+
+ /**
+ * Returns accessible version of the given constructor.
+ * @param ctor prototype constructor object.
+ * @return <code>null</code> if accessible constructor can not be found.
+ * @see java.lang.SecurityManager
+ */
+ public static Constructor getAccessibleConstructor(Constructor ctor) {
+
+ // Make sure we have a method to check
+ if (ctor == null) {
+ return (null);
+ }
+
+ // If the requested method is not public we cannot call it
+ if (!Modifier.isPublic(ctor.getModifiers())) {
+ return (null);
+ }
+
+ // If the declaring class is public, we are done
+ Class clazz = ctor.getDeclaringClass();
+ if (Modifier.isPublic(clazz.getModifiers())) {
+ return (ctor);
+ }
+
+ // what else can we do?
+ return null;
+
+ }
+
+ // -------------------------------------------------------- Private Methods
+ /**
+ * <p>Find an accessible constructor with compatible parameters.
+ * Compatible parameters mean that every method parameter is assignable from
+ * the given parameters. In other words, it finds constructor that will take
+ * the parameters given.</p>
+ *
+ * <p>First it checks if there is constructor matching the exact signature.
+ * If no such, all the constructors of the class are tested if their signatures
+ * are assignment compatible with the parameter types.
+ * The first matching constructor is returned.</p>
+ *
+ * @param clazz find constructor for this class
+ * @param parameterTypes find method with compatible parameters
+ * @return a valid Constructor object. If there's no matching constructor, returns <code>null</code>.
+ */
+ private static Constructor getMatchingAccessibleConstructor(
+ Class clazz,
+ Class[] parameterTypes) {
+ // see if we can find the method directly
+ // most of the time this works and it's much faster
+ try {
+ Constructor ctor = clazz.getConstructor(parameterTypes);
+ try {
+ //
+ // XXX Default access superclass workaround
+ //
+ // When a public class has a default access superclass
+ // with public methods, these methods are accessible.
+ // Calling them from compiled code works fine.
+ //
+ // Unfortunately, using reflection to invoke these methods
+ // seems to (wrongly) to prevent access even when the method
+ // modifer is public.
+ //
+ // The following workaround solves the problem but will only
+ // work from sufficiently privilages code.
+ //
+ // Better workarounds would be greatfully accepted.
+ //
+ ctor.setAccessible(true);
+ } catch (SecurityException se) {}
+ return ctor;
+
+ } catch (NoSuchMethodException e) { /* SWALLOW */
+ }
+
+ // search through all methods
+ int paramSize = parameterTypes.length;
+ Constructor[] ctors = clazz.getConstructors();
+ for (int i = 0, size = ctors.length; i < size; i++) {
+ // compare parameters
+ Class[] ctorParams = ctors[i].getParameterTypes();
+ int ctorParamSize = ctorParams.length;
+ if (ctorParamSize == paramSize) {
+ boolean match = true;
+ for (int n = 0; n < ctorParamSize; n++) {
+ if (!MethodUtils
+ .isAssignmentCompatible(
+ ctorParams[n],
+ parameterTypes[n])) {
+ match = false;
+ break;
+ }
+ }
+
+ if (match) {
+ // get accessible version of method
+ Constructor ctor = getAccessibleConstructor(ctors[i]);
+ if (ctor != null) {
+ try {
+ ctor.setAccessible(true);
+ } catch (SecurityException se) {}
+ return ctor;
+ }
+ }
+ }
+ }
+
+ return null;
+ }
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/ContextClassLoaderLocal.java b/trunk/src/java/org/apache/commons/beanutils/ContextClassLoaderLocal.java
new file mode 100644
index 0000000..23f2797
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/ContextClassLoaderLocal.java
@@ -0,0 +1,142 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import java.util.Map;
+import java.util.WeakHashMap;
+
+/**
+ * A value that is provided per (thread) context classloader.
+ * Patterned after ThreadLocal.
+ * There is a separate value used when Thread.getContextClassLoader() is null.
+ * This mechanism provides isolation for web apps deployed in the same container.
+ * <strong>Note:</strong> A WeakHashMap bug in several 1.3 JVMs results in a memory leak
+ * for those JVMs.
+ *
+ * @see java.lang.Thread#getContextClassLoader
+ * @author Eric Pabst
+ */
+public class ContextClassLoaderLocal {
+ private Map valueByClassLoader = new WeakHashMap();
+ private boolean globalValueInitialized = false;
+ private Object globalValue;
+
+ public ContextClassLoaderLocal() {
+ super();
+ }
+
+ /**
+ * Returns the initial value for this ContextClassLoaderLocal
+ * variable. This method will be called once per Context ClassLoader for
+ * each ContextClassLoaderLocal, the first time it is accessed
+ * with get or set. If the programmer desires ContextClassLoaderLocal variables
+ * to be initialized to some value other than null, ContextClassLoaderLocal must
+ * be subclassed, and this method overridden. Typically, an anonymous
+ * inner class will be used. Typical implementations of initialValue
+ * will call an appropriate constructor and return the newly constructed
+ * object.
+ *
+ * @return a new Object to be used as an initial value for this ContextClassLoaderLocal
+ */
+ protected Object initialValue() {
+ return null;
+ }
+
+ /**
+ * Gets the instance which provides the functionality for {@link BeanUtils}.
+ * This is a pseudo-singleton - an single instance is provided per (thread) context classloader.
+ * This mechanism provides isolation for web apps deployed in the same container.
+ * @return the object currently associated with the
+ */
+ public synchronized Object get() {
+ // synchronizing the whole method is a bit slower
+ // but guarentees no subtle threading problems, and there's no
+ // need to synchronize valueByClassLoader
+
+ // make sure that the map is given a change to purge itself
+ valueByClassLoader.isEmpty();
+ try {
+
+ ClassLoader contextClassLoader = Thread.currentThread().getContextClassLoader();
+ if (contextClassLoader != null) {
+
+ Object value = valueByClassLoader.get(contextClassLoader);
+ if ((value == null)
+ && !valueByClassLoader.containsKey(contextClassLoader)) {
+ value = initialValue();
+ valueByClassLoader.put(contextClassLoader, value);
+ }
+ return value;
+
+ }
+
+ } catch (SecurityException e) { /* SWALLOW - should we log this? */ }
+
+ // if none or exception, return the globalValue
+ if (!globalValueInitialized) {
+ globalValue = initialValue();
+ globalValueInitialized = true;
+ }//else already set
+ return globalValue;
+ }
+
+ /**
+ * Sets the value - a value is provided per (thread) context classloader.
+ * This mechanism provides isolation for web apps deployed in the same container.
+ *
+ * @param value the object to be associated with the entrant thread's context classloader
+ */
+ public synchronized void set(Object value) {
+ // synchronizing the whole method is a bit slower
+ // but guarentees no subtle threading problems
+
+ // make sure that the map is given a change to purge itself
+ valueByClassLoader.isEmpty();
+ try {
+
+ ClassLoader contextClassLoader = Thread.currentThread().getContextClassLoader();
+ if (contextClassLoader != null) {
+ valueByClassLoader.put(contextClassLoader, value);
+ return;
+ }
+
+ } catch (SecurityException e) { /* SWALLOW - should we log this? */ }
+
+ // if in doubt, set the global value
+ globalValue = value;
+ globalValueInitialized = true;
+ }
+
+ /**
+ * Unsets the value associated with the current thread's context classloader
+ */
+ public synchronized void unset() {
+ try {
+
+ ClassLoader contextClassLoader = Thread.currentThread().getContextClassLoader();
+ unset(contextClassLoader);
+
+ } catch (SecurityException e) { /* SWALLOW - should we log this? */ }
+ }
+
+ /**
+ * Unsets the value associated with the given classloader
+ */
+ public synchronized void unset(ClassLoader classLoader) {
+ valueByClassLoader.remove(classLoader);
+ }
+}
\ No newline at end of file
diff --git a/trunk/src/java/org/apache/commons/beanutils/ConversionException.java b/trunk/src/java/org/apache/commons/beanutils/ConversionException.java
new file mode 100644
index 0000000..3e88753
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/ConversionException.java
@@ -0,0 +1,89 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+/**
+ * <p>A <strong>ConversionException</strong> indicates that a call to
+ * <code>Converter.convert()</code> has failed to complete successfully.
+ *
+ * @author Craig McClanahan
+ * @author Paulo Gaspar
+ * @since 1.3
+ */
+
+public class ConversionException extends RuntimeException {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new exception with the specified message.
+ *
+ * @param message The message describing this exception
+ */
+ public ConversionException(String message) {
+
+ super(message);
+
+ }
+
+
+ /**
+ * Construct a new exception with the specified message and root cause.
+ *
+ * @param message The message describing this exception
+ * @param cause The root cause of this exception
+ */
+ public ConversionException(String message, Throwable cause) {
+
+ super(message);
+ this.cause = cause;
+
+ }
+
+
+ /**
+ * Construct a new exception with the specified root cause.
+ *
+ * @param cause The root cause of this exception
+ */
+ public ConversionException(Throwable cause) {
+
+ super(cause.getMessage());
+ this.cause = cause;
+
+ }
+
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ * The root cause of this <code>ConversionException</code>, compatible with
+ * JDK 1.4's extensions to <code>java.lang.Throwable</code>.
+ */
+ protected Throwable cause = null;
+
+ public Throwable getCause() {
+ return (this.cause);
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/ConvertUtils.java b/trunk/src/java/org/apache/commons/beanutils/ConvertUtils.java
new file mode 100644
index 0000000..a5707f7
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/ConvertUtils.java
@@ -0,0 +1,298 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+/**
+ * <p>Utility methods for converting String scalar values to objects of the
+ * specified Class, String arrays to arrays of the specified Class.</p>
+ *
+ * <p>For more details, see <code>ConvertUtilsBean</code> which provides the
+ * implementations for these methods.</p>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @version $Revision: 1.17 $ $Date: 2004/02/28 13:18:33 $
+ * @see ConvertUtilsBean
+ */
+
+public class ConvertUtils {
+
+
+ // ------------------------------------------------------ Static Properties
+
+ /**
+ * Gets the default value for Boolean conversions.
+ * @deprecated Register replacement converters for Boolean.TYPE and
+ * Boolean.class instead
+ */
+ public static boolean getDefaultBoolean() {
+ return (ConvertUtilsBean.getInstance().getDefaultBoolean());
+ }
+
+ /**
+ * Sets the default value for Boolean conversions.
+ * @deprecated Register replacement converters for Boolean.TYPE and
+ * Boolean.class instead
+ */
+ public static void setDefaultBoolean(boolean newDefaultBoolean) {
+ ConvertUtilsBean.getInstance().setDefaultBoolean(newDefaultBoolean);
+ }
+
+
+ /**
+ * Gets the default value for Byte conversions.
+ * @deprecated Register replacement converters for Byte.TYPE and
+ * Byte.class instead
+ */
+ public static byte getDefaultByte() {
+ return ConvertUtilsBean.getInstance().getDefaultByte();
+ }
+
+ /**
+ * Sets the default value for Byte conversions.
+ * @deprecated Register replacement converters for Byte.TYPE and
+ * Byte.class instead
+ */
+ public static void setDefaultByte(byte newDefaultByte) {
+ ConvertUtilsBean.getInstance().setDefaultByte(newDefaultByte);
+ }
+
+
+ /**
+ * Gets the default value for Character conversions.
+ * @deprecated Register replacement converters for Character.TYPE and
+ * Character.class instead
+ */
+ public static char getDefaultCharacter() {
+ return ConvertUtilsBean.getInstance().getDefaultCharacter();
+ }
+
+ /**
+ * Sets the default value for Character conversions.
+ * @deprecated Register replacement converters for Character.TYPE and
+ * Character.class instead
+ */
+ public static void setDefaultCharacter(char newDefaultCharacter) {
+ ConvertUtilsBean.getInstance().setDefaultCharacter(newDefaultCharacter);
+ }
+
+
+ /**
+ * Gets the default value for Double conversions.
+ * @deprecated Register replacement converters for Double.TYPE and
+ * Double.class instead
+ */
+ public static double getDefaultDouble() {
+ return ConvertUtilsBean.getInstance().getDefaultDouble();
+ }
+
+ /**
+ * Sets the default value for Double conversions.
+ * @deprecated Register replacement converters for Double.TYPE and
+ * Double.class instead
+ */
+ public static void setDefaultDouble(double newDefaultDouble) {
+ ConvertUtilsBean.getInstance().setDefaultDouble(newDefaultDouble);
+ }
+
+
+ /**
+ * Get the default value for Float conversions.
+ * @deprecated Register replacement converters for Float.TYPE and
+ * Float.class instead
+ */
+ public static float getDefaultFloat() {
+ return ConvertUtilsBean.getInstance().getDefaultFloat();
+ }
+
+ /**
+ * Sets the default value for Float conversions.
+ * @deprecated Register replacement converters for Float.TYPE and
+ * Float.class instead
+ */
+ public static void setDefaultFloat(float newDefaultFloat) {
+ ConvertUtilsBean.getInstance().setDefaultFloat(newDefaultFloat);
+ }
+
+
+ /**
+ * Gets the default value for Integer conversions.
+ * @deprecated Register replacement converters for Integer.TYPE and
+ * Integer.class instead
+ */
+ public static int getDefaultInteger() {
+ return ConvertUtilsBean.getInstance().getDefaultInteger();
+ }
+
+ /**
+ * Sets the default value for Integer conversions.
+ * @deprecated Register replacement converters for Integer.TYPE and
+ * Integer.class instead
+ */
+ public static void setDefaultInteger(int newDefaultInteger) {
+ ConvertUtilsBean.getInstance().setDefaultInteger(newDefaultInteger);
+ }
+
+
+ /**
+ * Gets the default value for Long conversions.
+ * @deprecated Register replacement converters for Long.TYPE and
+ * Long.class instead
+ */
+ public static long getDefaultLong() {
+ return (ConvertUtilsBean.getInstance().getDefaultLong());
+ }
+
+ /**
+ * Sets the default value for Long conversions.
+ * @deprecated Register replacement converters for Long.TYPE and
+ * Long.class instead
+ */
+ public static void setDefaultLong(long newDefaultLong) {
+ ConvertUtilsBean.getInstance().setDefaultLong(newDefaultLong);
+ }
+
+
+ /**
+ * Gets the default value for Short conversions.
+ * @deprecated Register replacement converters for Short.TYPE and
+ * Short.class instead
+ */
+ public static short getDefaultShort() {
+ return ConvertUtilsBean.getInstance().getDefaultShort();
+ }
+
+ /**
+ * Sets the default value for Short conversions.
+ * @deprecated Register replacement converters for Short.TYPE and
+ * Short.class instead
+ */
+ public static void setDefaultShort(short newDefaultShort) {
+ ConvertUtilsBean.getInstance().setDefaultShort(newDefaultShort);
+ }
+
+ // --------------------------------------------------------- Public Classes
+
+
+ /**
+ * <p>Convert the specified value into a String.</p>
+ *
+ * <p>For more details see <code>ConvertUtilsBean</code>.</p>
+ *
+ * @see ConvertUtilsBean#convert(Object)
+ */
+ public static String convert(Object value) {
+
+ return ConvertUtilsBean.getInstance().convert(value);
+
+ }
+
+
+ /**
+ * <p>Convert the specified value to an object of the specified class (if
+ * possible). Otherwise, return a String representation of the value.</p>
+ *
+ * <p>For more details see <code>ConvertUtilsBean</code>.</p>
+ *
+ * @see ConvertUtilsBean#convert(String, Class)
+ */
+ public static Object convert(String value, Class clazz) {
+
+ return ConvertUtilsBean.getInstance().convert(value, clazz);
+
+ }
+
+
+ /**
+ * <p>Convert an array of specified values to an array of objects of the
+ * specified class (if possible).</p>
+ *
+ * <p>For more details see <code>ConvertUtilsBean</code>.</p>
+ *
+ * @see ConvertUtilsBean#convert(String[], Class)
+ */
+ public static Object convert(String values[], Class clazz) {
+
+ return ConvertUtilsBean.getInstance().convert(values, clazz);
+
+ }
+
+
+ /**
+ * <p>Remove all registered {@link Converter}s, and re-establish the
+ * standard Converters.</p>
+ *
+ * <p>For more details see <code>ConvertUtilsBean</code>.</p>
+ *
+ * @see ConvertUtilsBean#deregister()
+ */
+ public static void deregister() {
+
+ ConvertUtilsBean.getInstance().deregister();
+
+ }
+
+
+ /**
+ * <p>Remove any registered {@link Converter} for the specified destination
+ * <code>Class</code>.</p>
+ *
+ * <p>For more details see <code>ConvertUtilsBean</code>.</p>
+ *
+ * @see ConvertUtilsBean#deregister(Class)
+ */
+ public static void deregister(Class clazz) {
+
+ ConvertUtilsBean.getInstance().deregister(clazz);
+
+ }
+
+
+ /**
+ * <p>Look up and return any registered {@link Converter} for the specified
+ * destination class; if there is no registered Converter, return
+ * <code>null</code>.</p>
+ *
+ * <p>For more details see <code>ConvertUtilsBean</code>.</p>
+ *
+ * @see ConvertUtilsBean#lookup(Class)
+ */
+ public static Converter lookup(Class clazz) {
+
+ return ConvertUtilsBean.getInstance().lookup(clazz);
+
+ }
+
+
+ /**
+ * <p>Register a custom {@link Converter} for the specified destination
+ * <code>Class</code>, replacing any previously registered Converter.</p>
+ *
+ * <p>For more details see <code>ConvertUtilsBean</code>.</p>
+ *
+ * @see ConvertUtilsBean#register(Converter, Class)
+ */
+ public static void register(Converter converter, Class clazz) {
+
+ ConvertUtilsBean.getInstance().register(converter, clazz);
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/ConvertUtilsBean.java b/trunk/src/java/org/apache/commons/beanutils/ConvertUtilsBean.java
new file mode 100644
index 0000000..fa73d09
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/ConvertUtilsBean.java
@@ -0,0 +1,581 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.io.File;
+import java.lang.reflect.Array;
+import java.math.BigDecimal;
+import java.math.BigInteger;
+import java.net.URL;
+import java.sql.Date;
+import java.sql.Time;
+import java.sql.Timestamp;
+import org.apache.commons.beanutils.converters.BigDecimalConverter;
+import org.apache.commons.beanutils.converters.BigIntegerConverter;
+import org.apache.commons.beanutils.converters.BooleanConverter;
+import org.apache.commons.beanutils.converters.BooleanArrayConverter;
+import org.apache.commons.beanutils.converters.ByteConverter;
+import org.apache.commons.beanutils.converters.ByteArrayConverter;
+import org.apache.commons.beanutils.converters.CharacterConverter;
+import org.apache.commons.beanutils.converters.CharacterArrayConverter;
+import org.apache.commons.beanutils.converters.ClassConverter;
+import org.apache.commons.beanutils.converters.DoubleConverter;
+import org.apache.commons.beanutils.converters.DoubleArrayConverter;
+import org.apache.commons.beanutils.converters.FileConverter;
+import org.apache.commons.beanutils.converters.FloatConverter;
+import org.apache.commons.beanutils.converters.FloatArrayConverter;
+import org.apache.commons.beanutils.converters.IntegerConverter;
+import org.apache.commons.beanutils.converters.IntegerArrayConverter;
+import org.apache.commons.beanutils.converters.LongConverter;
+import org.apache.commons.beanutils.converters.LongArrayConverter;
+import org.apache.commons.beanutils.converters.ShortConverter;
+import org.apache.commons.beanutils.converters.ShortArrayConverter;
+import org.apache.commons.beanutils.converters.SqlDateConverter;
+import org.apache.commons.beanutils.converters.SqlTimeConverter;
+import org.apache.commons.beanutils.converters.SqlTimestampConverter;
+import org.apache.commons.beanutils.converters.StringConverter;
+import org.apache.commons.beanutils.converters.StringArrayConverter;
+import org.apache.commons.beanutils.converters.URLConverter;
+import org.apache.commons.collections.FastHashMap;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+
+/**
+ * <p>Utility methods for converting String scalar values to objects of the
+ * specified Class, String arrays to arrays of the specified Class. The
+ * actual {@link Converter} instance to be used can be registered for each
+ * possible destination Class. Unless you override them, standard
+ * {@link Converter} instances are provided for all of the following
+ * destination Classes:</p>
+ * <ul>
+ * <li>java.lang.BigDecimal</li>
+ * <li>java.lang.BigInteger</li>
+ * <li>boolean and java.lang.Boolean</li>
+ * <li>byte and java.lang.Byte</li>
+ * <li>char and java.lang.Character</li>
+ * <li>java.lang.Class</li>
+ * <li>double and java.lang.Double</li>
+ * <li>float and java.lang.Float</li>
+ * <li>int and java.lang.Integer</li>
+ * <li>long and java.lang.Long</li>
+ * <li>short and java.lang.Short</li>
+ * <li>java.lang.String</li>
+ * <li>java.io.File</li>
+ * <li>java.net.URL</li>
+ * <li>java.sql.Date</li>
+ * <li>java.sql.Time</li>
+ * <li>java.sql.Timestamp</li>
+ * </ul>
+ *
+ * <p>For backwards compatibility, the standard Converters for primitive
+ * types (and the corresponding wrapper classes) return a defined
+ * default value when a conversion error occurs. If you prefer to have a
+ * {@link ConversionException} thrown instead, replace the standard Converter
+ * instances with instances created with the zero-arguments constructor. For
+ * example, to cause the Converters for integers to throw an exception on
+ * conversion errors, you could do this:</p>
+ * <pre>
+ * // No-args constructor gets the version that throws exceptions
+ * Converter myConverter =
+ * new org.apache.commons.beanutils.converter.IntegerConverter();
+ * ConvertUtils.register(myConverter, Integer.TYPE); // Native type
+ * ConvertUtils.register(myConverter, Integer.class); // Wrapper class
+ * </pre>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @author James Strachan
+ * @version $Revision: 1.12.2.1 $ $Date: 2004/07/27 21:44:26 $
+ * @since 1.7
+ */
+
+public class ConvertUtilsBean {
+
+ // ------------------------------------------------------- Class Methods
+ /** Get singleton instance */
+ protected static ConvertUtilsBean getInstance() {
+ return BeanUtilsBean.getInstance().getConvertUtils();
+ }
+
+ // ------------------------------------------------------- Variables
+
+
+ /**
+ * The set of {@link Converter}s that can be used to convert Strings
+ * into objects of a specified Class, keyed by the destination Class.
+ */
+ private FastHashMap converters = new FastHashMap();
+
+ /**
+ * The <code>Log</code> instance for this class.
+ */
+ private Log log = LogFactory.getLog(ConvertUtils.class);
+
+ // ------------------------------------------------------- Constructors
+
+ /** Construct a bean with standard converters registered */
+ public ConvertUtilsBean() {
+ converters.setFast(false);
+ deregister();
+ converters.setFast(true);
+ }
+
+ // --------------------------------------------------------- Public Methods
+
+ // ------------------------------------------------------ Static Properties
+
+
+ /**
+ * The default value for Boolean conversions.
+ * @deprecated Register replacement converters for Boolean.TYPE and
+ * Boolean.class instead
+ */
+ private Boolean defaultBoolean = Boolean.FALSE;
+
+ /**
+ * Gets the default value for Boolean conversions.
+ * @deprecated Register replacement converters for Boolean.TYPE and
+ * Boolean.class instead
+ */
+ public boolean getDefaultBoolean() {
+ return (defaultBoolean.booleanValue());
+ }
+
+ /**
+ * Sets the default value for Boolean conversions.
+ * @deprecated Register replacement converters for Boolean.TYPE and
+ * Boolean.class instead
+ */
+ public void setDefaultBoolean(boolean newDefaultBoolean) {
+ defaultBoolean = new Boolean(newDefaultBoolean);
+ register(new BooleanConverter(defaultBoolean), Boolean.TYPE);
+ register(new BooleanConverter(defaultBoolean), Boolean.class);
+ }
+
+
+ /**
+ * The default value for Byte conversions.
+ * @deprecated Register replacement converters for Byte.TYPE and
+ * Byte.class instead
+ */
+ private Byte defaultByte = new Byte((byte) 0);
+
+ /**
+ * Gets the default value for Byte conversions.
+ * @deprecated Register replacement converters for Byte.TYPE and
+ * Byte.class instead
+ */
+ public byte getDefaultByte() {
+ return (defaultByte.byteValue());
+ }
+
+ /**
+ * Sets the default value for Byte conversions.
+ * @deprecated Register replacement converters for Byte.TYPE and
+ * Byte.class instead
+ */
+ public void setDefaultByte(byte newDefaultByte) {
+ defaultByte = new Byte(newDefaultByte);
+ register(new ByteConverter(defaultByte), Byte.TYPE);
+ register(new ByteConverter(defaultByte), Byte.class);
+ }
+
+
+ /**
+ * The default value for Character conversions.
+ * @deprecated Register replacement converters for Character.TYPE and
+ * Character.class instead
+ */
+ private Character defaultCharacter = new Character(' ');
+
+ /**
+ * Gets the default value for Character conversions.
+ * @deprecated Register replacement converters for Character.TYPE and
+ * Character.class instead
+ */
+ public char getDefaultCharacter() {
+ return (defaultCharacter.charValue());
+ }
+
+ /**
+ * Sets the default value for Character conversions.
+ * @deprecated Register replacement converters for Character.TYPE and
+ * Character.class instead
+ */
+ public void setDefaultCharacter(char newDefaultCharacter) {
+ defaultCharacter = new Character(newDefaultCharacter);
+ register(new CharacterConverter(defaultCharacter),
+ Character.TYPE);
+ register(new CharacterConverter(defaultCharacter),
+ Character.class);
+ }
+
+
+ /**
+ * The default value for Double conversions.
+ * @deprecated Register replacement converters for Double.TYPE and
+ * Double.class instead
+ */
+ private Double defaultDouble = new Double((double) 0.0);
+
+ /**
+ * Gets the default value for Double conversions.
+ * @deprecated Register replacement converters for Double.TYPE and
+ * Double.class instead
+ */
+ public double getDefaultDouble() {
+ return (defaultDouble.doubleValue());
+ }
+
+ /**
+ * Sets the default value for Double conversions.
+ * @deprecated Register replacement converters for Double.TYPE and
+ * Double.class instead
+ */
+ public void setDefaultDouble(double newDefaultDouble) {
+ defaultDouble = new Double(newDefaultDouble);
+ register(new DoubleConverter(defaultDouble), Double.TYPE);
+ register(new DoubleConverter(defaultDouble), Double.class);
+ }
+
+
+ /**
+ * The default value for Float conversions.
+ * @deprecated Register replacement converters for Float.TYPE and
+ * Float.class instead
+ */
+ private Float defaultFloat = new Float((float) 0.0);
+
+ /**
+ * Gets the default value for Float conversions.
+ * @deprecated Register replacement converters for Float.TYPE and
+ * Float.class instead
+ */
+ public float getDefaultFloat() {
+ return (defaultFloat.floatValue());
+ }
+
+ /**
+ * Sets the default value for Float conversions.
+ * @deprecated Register replacement converters for Float.TYPE and
+ * Float.class instead
+ */
+ public void setDefaultFloat(float newDefaultFloat) {
+ defaultFloat = new Float(newDefaultFloat);
+ register(new FloatConverter(defaultFloat), Float.TYPE);
+ register(new FloatConverter(defaultFloat), Float.class);
+ }
+
+
+ /**
+ * The default value for Integer conversions.
+ * @deprecated Register replacement converters for Integer.TYPE and
+ * Integer.class instead
+ */
+ private Integer defaultInteger = new Integer(0);
+
+ /**
+ * Gets the default value for Integer conversions.
+ * @deprecated Register replacement converters for Integer.TYPE and
+ * Integer.class instead
+ */
+ public int getDefaultInteger() {
+ return (defaultInteger.intValue());
+ }
+
+ /**
+ * Sets the default value for Integer conversions.
+ * @deprecated Register replacement converters for Integer.TYPE and
+ * Integer.class instead
+ */
+ public void setDefaultInteger(int newDefaultInteger) {
+ defaultInteger = new Integer(newDefaultInteger);
+ register(new IntegerConverter(defaultInteger), Integer.TYPE);
+ register(new IntegerConverter(defaultInteger), Integer.class);
+ }
+
+
+ /**
+ * The default value for Long conversions.
+ * @deprecated Register replacement converters for Long.TYPE and
+ * Long.class instead
+ */
+ private Long defaultLong = new Long((long) 0);
+
+ /**
+ * Gets the default value for Long conversions.
+ * @deprecated Register replacement converters for Long.TYPE and
+ * Long.class instead
+ */
+ public long getDefaultLong() {
+ return (defaultLong.longValue());
+ }
+
+ /**
+ * Sets the default value for Long conversions.
+ * @deprecated Register replacement converters for Long.TYPE and
+ * Long.class instead
+ */
+ public void setDefaultLong(long newDefaultLong) {
+ defaultLong = new Long(newDefaultLong);
+ register(new LongConverter(defaultLong), Long.TYPE);
+ register(new LongConverter(defaultLong), Long.class);
+ }
+
+
+ /**
+ * The default value for Short conversions.
+ * @deprecated Register replacement converters for Short.TYPE and
+ * Short.class instead
+ */
+ private static Short defaultShort = new Short((short) 0);
+
+ /**
+ * Gets the default value for Short conversions.
+ * @deprecated Register replacement converters for Short.TYPE and
+ * Short.class instead
+ */
+ public short getDefaultShort() {
+ return (defaultShort.shortValue());
+ }
+
+ /**
+ * Sets the default value for Short conversions.
+ * @deprecated Register replacement converters for Short.TYPE and
+ * Short.class instead
+ */
+ public void setDefaultShort(short newDefaultShort) {
+ defaultShort = new Short(newDefaultShort);
+ register(new ShortConverter(defaultShort), Short.TYPE);
+ register(new ShortConverter(defaultShort), Short.class);
+ }
+
+
+
+ /**
+ * Convert the specified value into a String. If the specified value
+ * is an array, the first element (converted to a String) will be
+ * returned. The registered {@link Converter} for the
+ * <code>java.lang.String</code> class will be used, which allows
+ * applications to customize Object->String conversions (the default
+ * implementation simply uses toString()).
+ *
+ * @param value Value to be converted (may be null)
+ */
+ public String convert(Object value) {
+
+ if (value == null) {
+ return ((String) null);
+ } else if (value.getClass().isArray()) {
+ if (Array.getLength(value) < 1) {
+ return (null);
+ }
+ value = Array.get(value, 0);
+ if (value == null) {
+ return ((String) null);
+ } else {
+ Converter converter = lookup(String.class);
+ return ((String) converter.convert(String.class, value));
+ }
+ } else {
+ Converter converter = lookup(String.class);
+ return ((String) converter.convert(String.class, value));
+ }
+
+ }
+
+
+ /**
+ * Convert the specified value to an object of the specified class (if
+ * possible). Otherwise, return a String representation of the value.
+ *
+ * @param value Value to be converted (may be null)
+ * @param clazz Java class to be converted to
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public Object convert(String value, Class clazz) {
+
+ if (log.isDebugEnabled()) {
+ log.debug("Convert string '" + value + "' to class '" +
+ clazz.getName() + "'");
+ }
+ Converter converter = lookup(clazz);
+ if (converter == null) {
+ converter = lookup(String.class);
+ }
+ if (log.isTraceEnabled()) {
+ log.trace(" Using converter " + converter);
+ }
+ return (converter.convert(clazz, value));
+
+ }
+
+
+ /**
+ * Convert an array of specified values to an array of objects of the
+ * specified class (if possible). If the specified Java class is itself
+ * an array class, this class will be the type of the returned value.
+ * Otherwise, an array will be constructed whose component type is the
+ * specified class.
+ *
+ * @param values Values to be converted (may be null)
+ * @param clazz Java array or element class to be converted to
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public Object convert(String values[], Class clazz) {
+
+ Class type = clazz;
+ if (clazz.isArray()) {
+ type = clazz.getComponentType();
+ }
+ if (log.isDebugEnabled()) {
+ log.debug("Convert String[" + values.length + "] to class '" +
+ type.getName() + "[]'");
+ }
+ Converter converter = lookup(type);
+ if (converter == null) {
+ converter = lookup(String.class);
+ }
+ if (log.isTraceEnabled()) {
+ log.trace(" Using converter " + converter);
+ }
+ Object array = Array.newInstance(type, values.length);
+ for (int i = 0; i < values.length; i++) {
+ Array.set(array, i, converter.convert(type, values[i]));
+ }
+ return (array);
+
+ }
+
+
+ /**
+ * Remove all registered {@link Converter}s, and re-establish the
+ * standard Converters.
+ */
+ public void deregister() {
+
+ boolean booleanArray[] = new boolean[0];
+ byte byteArray[] = new byte[0];
+ char charArray[] = new char[0];
+ double doubleArray[] = new double[0];
+ float floatArray[] = new float[0];
+ int intArray[] = new int[0];
+ long longArray[] = new long[0];
+ short shortArray[] = new short[0];
+ String stringArray[] = new String[0];
+
+ converters.clear();
+ register(BigDecimal.class, new BigDecimalConverter());
+ register(BigInteger.class, new BigIntegerConverter());
+ register(Boolean.TYPE, new BooleanConverter(defaultBoolean));
+ register(Boolean.class, new BooleanConverter(defaultBoolean));
+ register(booleanArray.getClass(),
+ new BooleanArrayConverter(booleanArray));
+ register(Byte.TYPE, new ByteConverter(defaultByte));
+ register(Byte.class, new ByteConverter(defaultByte));
+ register(byteArray.getClass(),
+ new ByteArrayConverter(byteArray));
+ register(Character.TYPE,
+ new CharacterConverter(defaultCharacter));
+ register(Character.class,
+ new CharacterConverter(defaultCharacter));
+ register(charArray.getClass(),
+ new CharacterArrayConverter(charArray));
+ register(Class.class, new ClassConverter());
+ register(Double.TYPE, new DoubleConverter(defaultDouble));
+ register(Double.class, new DoubleConverter(defaultDouble));
+ register(doubleArray.getClass(),
+ new DoubleArrayConverter(doubleArray));
+ register(Float.TYPE, new FloatConverter(defaultFloat));
+ register(Float.class, new FloatConverter(defaultFloat));
+ register(floatArray.getClass(),
+ new FloatArrayConverter(floatArray));
+ register(Integer.TYPE, new IntegerConverter(defaultInteger));
+ register(Integer.class, new IntegerConverter(defaultInteger));
+ register(intArray.getClass(),
+ new IntegerArrayConverter(intArray));
+ register(Long.TYPE, new LongConverter(defaultLong));
+ register(Long.class, new LongConverter(defaultLong));
+ register(longArray.getClass(),
+ new LongArrayConverter(longArray));
+ register(Short.TYPE, new ShortConverter(defaultShort));
+ register(Short.class, new ShortConverter(defaultShort));
+ register(shortArray.getClass(),
+ new ShortArrayConverter(shortArray));
+ register(String.class, new StringConverter());
+ register(stringArray.getClass(),
+ new StringArrayConverter(stringArray));
+ register(Date.class, new SqlDateConverter());
+ register(Time.class, new SqlTimeConverter());
+ register(Timestamp.class, new SqlTimestampConverter());
+ register(File.class, new FileConverter());
+ register(URL.class, new URLConverter());
+
+ }
+
+ /** strictly for convenience since it has same parameter order as Map.put */
+ private void register(Class clazz, Converter converter) {
+ register(converter, clazz);
+ }
+
+ /**
+ * Remove any registered {@link Converter} for the specified destination
+ * <code>Class</code>.
+ *
+ * @param clazz Class for which to remove a registered Converter
+ */
+ public void deregister(Class clazz) {
+
+ converters.remove(clazz);
+
+ }
+
+
+ /**
+ * Look up and return any registered {@link Converter} for the specified
+ * destination class; if there is no registered Converter, return
+ * <code>null</code>.
+ *
+ * @param clazz Class for which to return a registered Converter
+ */
+ public Converter lookup(Class clazz) {
+
+ return ((Converter) converters.get(clazz));
+
+ }
+
+
+ /**
+ * Register a custom {@link Converter} for the specified destination
+ * <code>Class</code>, replacing any previously registered Converter.
+ *
+ * @param converter Converter to be registered
+ * @param clazz Destination class for conversions performed by this
+ * Converter
+ */
+ public void register(Converter converter, Class clazz) {
+
+ converters.put(clazz, converter);
+
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/Converter.java b/trunk/src/java/org/apache/commons/beanutils/Converter.java
new file mode 100644
index 0000000..3fecffe
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/Converter.java
@@ -0,0 +1,48 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+/**
+ * <p>General purpose data type converter that can be registered and used
+ * within the BeanUtils package to manage the conversion of objects from
+ * one type to another.
+ *
+ * @author Craig McClanahan
+ * @author Paulo Gaspar
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:33 $
+ * @since 1.3
+ */
+
+public interface Converter {
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value);
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/ConvertingWrapDynaBean.java b/trunk/src/java/org/apache/commons/beanutils/ConvertingWrapDynaBean.java
new file mode 100644
index 0000000..7931be5
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/ConvertingWrapDynaBean.java
@@ -0,0 +1,76 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+/**
+ * <p>Implementation of <code>DynaBean</code> that wraps a standard JavaBean
+ * instance, so that DynaBean APIs can be used to access its properties,
+ * though this implementation allows type conversion to occur when properties are set.
+ * This means that (say) Strings can be passed in as values in setter methods and
+ * this DynaBean will convert them to the correct primitive data types.</p>
+ *
+ * <p><strong>IMPLEMENTATION NOTE</strong> - This implementation does not
+ * support the <code>contains()</code> and <code>remove()</code> methods.</p>
+ *
+ * @author James Strachan
+ * @version $Revision: 1.3 $ $Date: 2002/01/23 22:35:58 $
+ */
+
+public class ConvertingWrapDynaBean extends WrapDynaBean {
+
+
+
+ /**
+ * Construct a new <code>DynaBean</code> associated with the specified
+ * JavaBean instance.
+ *
+ * @param instance JavaBean instance to be wrapped
+ */
+ public ConvertingWrapDynaBean(Object instance) {
+
+ super(instance);
+
+ }
+
+
+ /**
+ * Set the value of the property with the specified name
+ * performing any type conversions if necessary. So this method
+ * can accept String values for primitive numeric data types for example.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception NullPointerException if an attempt is made to set a
+ * primitive property to null
+ */
+ public void set(String name, Object value) {
+
+ try {
+ BeanUtils.copyProperty(instance, name, value);
+ } catch (Throwable t) {
+ throw new IllegalArgumentException
+ ("Property '" + name + "' has no write method");
+ }
+
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/DynaBean.java b/trunk/src/java/org/apache/commons/beanutils/DynaBean.java
new file mode 100644
index 0000000..fbda7ee
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/DynaBean.java
@@ -0,0 +1,166 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+/**
+ * <p>A <strong>DynaBean</strong> is a Java object that supports properties
+ * whose names and data types, as well as values, may be dynamically modified.
+ * To the maximum degree feasible, other components of the BeanUtils package
+ * will recognize such beans and treat them as standard JavaBeans for the
+ * purpose of retrieving and setting property values.</p>
+ *
+ * @author Craig McClanahan
+ * @author Paulo Gaspar
+ * @version $Revision: 1.9 $ $Date: 2004/02/28 13:18:33 $
+ */
+
+public interface DynaBean {
+
+
+ /**
+ * Does the specified mapped property contain a value for the specified
+ * key value?
+ *
+ * @param name Name of the property to check
+ * @param key Name of the key to check
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public boolean contains(String name, String key);
+
+
+ /**
+ * Return the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public Object get(String name);
+
+
+ /**
+ * Return the value of an indexed property with the specified name.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param index Index of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ * @exception NullPointerException if no array or List has been
+ * initialized for this property
+ */
+ public Object get(String name, int index);
+
+
+ /**
+ * Return the value of a mapped property with the specified name,
+ * or <code>null</code> if there is no value for the specified key.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param key Key of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public Object get(String name, String key);
+
+
+ /**
+ * Return the <code>DynaClass</code> instance that describes the set of
+ * properties available for this DynaBean.
+ */
+ public DynaClass getDynaClass();
+
+
+ /**
+ * Remove any existing value for the specified key on the
+ * specified mapped property.
+ *
+ * @param name Name of the property for which a value is to
+ * be removed
+ * @param key Key of the value to be removed
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public void remove(String name, String key);
+
+
+ /**
+ * Set the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception NullPointerException if an attempt is made to set a
+ * primitive property to null
+ */
+ public void set(String name, Object value);
+
+
+ /**
+ * Set the value of an indexed property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param index Index of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ */
+ public void set(String name, int index, Object value);
+
+
+ /**
+ * Set the value of a mapped property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param key Key of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public void set(String name, String key, Object value);
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/DynaClass.java b/trunk/src/java/org/apache/commons/beanutils/DynaClass.java
new file mode 100644
index 0000000..5b35a42
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/DynaClass.java
@@ -0,0 +1,87 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+
+
+
+/**
+ * <p>A <strong>DynaClass</strong> is a simulation of the functionality of
+ * <code>java.lang.Class</code> for classes implementing the
+ * <code>DynaBean</code> interface. DynaBean instances that share the same
+ * DynaClass all have the same set of available properties, along with any
+ * associated data types, read-only states, and write-only states.</p>
+ *
+ * @author Craig McClanahan
+ * @author Michael Smith
+ * @author Paulo Gaspar
+ * @version $Revision: 1.12 $ $Date: 2004/02/28 13:18:33 $
+ */
+
+public interface DynaClass {
+
+
+ /**
+ * Return the name of this DynaClass (analogous to the
+ * <code>getName()</code> method of <code>java.lang.Class</code), which
+ * allows the same <code>DynaClass</code> implementation class to support
+ * different dynamic classes, with different sets of properties.
+ */
+ public String getName();
+
+
+ /**
+ * Return a property descriptor for the specified property, if it exists;
+ * otherwise, return <code>null</code>.
+ *
+ * @param name Name of the dynamic property for which a descriptor
+ * is requested
+ *
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public DynaProperty getDynaProperty(String name);
+
+
+ /**
+ * <p>Return an array of <code>ProperyDescriptors</code> for the properties
+ * currently defined in this DynaClass. If no properties are defined, a
+ * zero-length array will be returned.</p>
+ *
+ * <p><strong>FIXME</strong> - Should we really be implementing
+ * <code>getBeanInfo()</code> instead, which returns property descriptors
+ * and a bunch of other stuff?</p>
+ */
+ public DynaProperty[] getDynaProperties();
+
+
+ /**
+ * Instantiate and return a new DynaBean instance, associated
+ * with this DynaClass.
+ *
+ * @exception IllegalAccessException if the Class or the appropriate
+ * constructor is not accessible
+ * @exception InstantiationException if this Class represents an abstract
+ * class, an array class, a primitive type, or void; or if instantiation
+ * fails for some other reason
+ */
+ public DynaBean newInstance()
+ throws IllegalAccessException, InstantiationException;
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/DynaProperty.java b/trunk/src/java/org/apache/commons/beanutils/DynaProperty.java
new file mode 100644
index 0000000..8f53fb7
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/DynaProperty.java
@@ -0,0 +1,315 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.io.IOException;
+import java.io.Serializable;
+import java.io.ObjectOutputStream;
+import java.io.ObjectInputStream;
+import java.io.StreamCorruptedException;
+import java.util.List;
+import java.util.Map;
+
+
+/**
+ * <p>The metadata describing an individual property of a DynaBean.</p>
+ *
+ * <p>The meta contains an <em>optional</em> content type property ({@link #getContentType})
+ * for use by mapped and iterated properties.
+ * A mapped or iterated property may choose to indicate the type it expects.
+ * The DynaBean implementation may choose to enforce this type on its entries.
+ * Alternatively, an implementatin may choose to ignore this property.
+ * All keys for maps must be of type String so no meta data is needed for map keys.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.13 $ $Date: 2004/02/28 13:18:33 $
+ */
+
+public class DynaProperty implements Serializable {
+
+ // ----------------------------------------------------------- Constants
+
+ /*
+ * There are issues with serializing primitive class types on certain JVM versions
+ * (including java 1.3).
+ * This class uses a custom serialization implementation that writes an integer
+ * for these primitive class.
+ * This list of constants are the ones used in serialization.
+ * If these values are changed, then older versions will no longer be read correctly
+ */
+ private static final int BOOLEAN_TYPE = 1;
+ private static final int BYTE_TYPE = 2;
+ private static final int CHAR_TYPE = 3;
+ private static final int DOUBLE_TYPE = 4;
+ private static final int FLOAT_TYPE = 5;
+ private static final int INT_TYPE = 6;
+ private static final int LONG_TYPE = 7;
+ private static final int SHORT_TYPE = 8;
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a property that accepts any data type.
+ *
+ * @param name Name of the property being described
+ */
+ public DynaProperty(String name) {
+
+ this(name, Object.class);
+
+ }
+
+
+ /**
+ * Construct a property of the specified data type.
+ *
+ * @param name Name of the property being described
+ * @param type Java class representing the property data type
+ */
+ public DynaProperty(String name, Class type) {
+
+ super();
+ this.name = name;
+ this.type = type;
+
+ }
+
+ /**
+ * Construct an indexed or mapped <code>DynaProperty</code> that supports (pseudo)-introspection
+ * of the content type.
+ *
+ * @param name Name of the property being described
+ * @param type Java class representing the property data type
+ * @param contentType Class that all indexed or mapped elements are instances of
+ */
+ public DynaProperty(String name, Class type, Class contentType) {
+
+ super();
+ this.name = name;
+ this.type = type;
+ this.contentType = contentType;
+
+ }
+
+ // ------------------------------------------------------------- Properties
+
+ /** Property name */
+ protected String name = null;
+ /**
+ * Get the name of this property.
+ */
+ public String getName() {
+ return (this.name);
+ }
+
+ /** Property type */
+ protected transient Class type = null;
+ /**
+ * <p>Gets the Java class representing the data type of the underlying property
+ * values.</p>
+ *
+ * <p>There are issues with serializing primitive class types on certain JVM versions
+ * (including java 1.3).
+ * Therefore, this field <strong>must not be serialized using the standard methods</strong>.</p>
+ *
+ * <p><strong>Please leave this field as <code>transient</code></strong></p>
+ */
+ public Class getType() {
+ return (this.type);
+ }
+
+
+ /** The <em>(optional)</em> type of content elements for indexed <code>DynaProperty</code> */
+ protected transient Class contentType;
+ /**
+ * Gets the <em>(optional)</em> type of the indexed content for <code>DynaProperty</code>'s
+ * that support this feature.
+ *
+ * <p>There are issues with serializing primitive class types on certain JVM versions
+ * (including java 1.3).
+ * Therefore, this field <strong>must not be serialized using the standard methods</strong>.</p>
+ *
+ * @return the Class for the content type if this is an indexed <code>DynaProperty</code>
+ * and this feature is supported. Otherwise null.
+ */
+ public Class getContentType() {
+ return contentType;
+ }
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Does this property represent an indexed value (ie an array or List)?
+ */
+ public boolean isIndexed() {
+
+ if (type == null) {
+ return (false);
+ } else if (type.isArray()) {
+ return (true);
+ } else if (List.class.isAssignableFrom(type)) {
+ return (true);
+ } else {
+ return (false);
+ }
+
+ }
+
+
+ /**
+ * Does this property represent a mapped value (ie a Map)?
+ */
+ public boolean isMapped() {
+
+ if (type == null) {
+ return (false);
+ } else {
+ return (Map.class.isAssignableFrom(type));
+ }
+
+ }
+
+
+ /**
+ * Return a String representation of this Object.
+ */
+ public String toString() {
+
+ StringBuffer sb = new StringBuffer("DynaProperty[name=");
+ sb.append(this.name);
+ sb.append(",type=");
+ sb.append(this.type);
+ if (isMapped() || isIndexed()) {
+ sb.append(" <").append(this.contentType).append(">");
+ }
+ sb.append("]");
+ return (sb.toString());
+
+ }
+
+ // --------------------------------------------------------- Serialization helper methods
+
+ /**
+ * Writes this object safely.
+ * There are issues with serializing primitive class types on certain JVM versions
+ * (including java 1.3).
+ * This method provides a workaround.
+ */
+ private void writeObject(ObjectOutputStream out) throws IOException {
+
+ writeAnyClass(this.type,out);
+
+ if (isMapped() || isIndexed()) {
+ writeAnyClass(this.contentType,out);
+ }
+
+ // write out other values
+ out.defaultWriteObject();
+ }
+
+ /**
+ * Write a class using safe encoding to workaround java 1.3 serialization bug.
+ */
+ private void writeAnyClass(Class clazz, ObjectOutputStream out) throws IOException {
+ // safely write out any class
+ int primitiveType = 0;
+ if (Boolean.TYPE.equals(clazz)) {
+ primitiveType = BOOLEAN_TYPE;
+ } else if (Byte.TYPE.equals(clazz)) {
+ primitiveType = BYTE_TYPE;
+ } else if (Character.TYPE.equals(clazz)) {
+ primitiveType = CHAR_TYPE;
+ } else if (Double.TYPE.equals(clazz)) {
+ primitiveType = DOUBLE_TYPE;
+ } else if (Float.TYPE.equals(clazz)) {
+ primitiveType = FLOAT_TYPE;
+ } else if (Integer.TYPE.equals(clazz)) {
+ primitiveType = INT_TYPE;
+ } else if (Long.TYPE.equals(clazz)) {
+ primitiveType = LONG_TYPE;
+ } else if (Short.TYPE.equals(clazz)) {
+ primitiveType = SHORT_TYPE;
+ }
+
+ if (primitiveType == 0) {
+ // then it's not a primitive type
+ out.writeBoolean(false);
+ out.writeObject(clazz);
+ } else {
+ // we'll write out a constant instead
+ out.writeBoolean(true);
+ out.writeInt(primitiveType);
+ }
+ }
+
+ /**
+ * Reads field values for this object safely.
+ * There are issues with serializing primitive class types on certain JVM versions
+ * (including java 1.3).
+ * This method provides a workaround.
+ *
+ * @throws StreamCorruptedException when the stream data values are outside expected range
+ */
+ private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException {
+
+ this.type = readAnyClass(in);
+
+ if (isMapped() || isIndexed()) {
+ this.contentType = readAnyClass(in);
+ }
+
+ // read other values
+ in.defaultReadObject();
+ }
+
+
+ /**
+ * Reads a class using safe encoding to workaround java 1.3 serialization bug.
+ */
+ private Class readAnyClass(ObjectInputStream in) throws IOException, ClassNotFoundException {
+ // read back type class safely
+ if (in.readBoolean()) {
+ // it's a type constant
+ switch (in.readInt()) {
+
+ case BOOLEAN_TYPE: return Boolean.TYPE;
+ case BYTE_TYPE: return Byte.TYPE;
+ case CHAR_TYPE: return Character.TYPE;
+ case DOUBLE_TYPE: return Double.TYPE;
+ case FLOAT_TYPE: return Float.TYPE;
+ case INT_TYPE: return Integer.TYPE;
+ case LONG_TYPE: return Long.TYPE;
+ case SHORT_TYPE: return Short.TYPE;
+ default:
+ // something's gone wrong
+ throw new StreamCorruptedException(
+ "Invalid primitive type. "
+ + "Check version of beanutils used to serialize is compatible.");
+
+ }
+
+ } else {
+ // it's another class
+ return ((Class) in.readObject());
+ }
+ }
+}
\ No newline at end of file
diff --git a/trunk/src/java/org/apache/commons/beanutils/JDBCDynaClass.java b/trunk/src/java/org/apache/commons/beanutils/JDBCDynaClass.java
new file mode 100644
index 0000000..3f55e3e
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/JDBCDynaClass.java
@@ -0,0 +1,214 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import java.io.Serializable;
+import java.sql.ResultSet;
+import java.sql.ResultSetMetaData;
+import java.sql.SQLException;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.Map;
+
+/**
+ * <p>Provides common logic for JDBC implementations of {@link DynaClass}.</p>
+ *
+ * @author Craig R. McClanahan
+ * @author George Franciscus
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:33 $
+ */
+
+abstract class JDBCDynaClass implements DynaClass, Serializable {
+
+ // ----------------------------------------------------- Instance Variables
+
+ /**
+ * <p>Flag defining whether column names should be lower cased when
+ * converted to property names.</p>
+ */
+ protected boolean lowerCase = true;
+
+ /**
+ * <p>The set of dynamic properties that are part of this
+ * {@link DynaClass}.</p>
+ */
+ protected DynaProperty properties[] = null;
+
+ /**
+ * <p>The set of dynamic properties that are part of this
+ * {@link DynaClass}, keyed by the property name. Individual descriptor
+ * instances will be the same instances as those in the
+ * <code>properties</code> list.</p>
+ */
+ protected Map propertiesMap = new HashMap();
+
+ // ------------------------------------------------------ DynaClass Methods
+
+ /**
+ * <p>Return the name of this DynaClass (analogous to the
+ * <code>getName()</code> method of <code>java.lang.Class</code), which
+ * allows the same <code>DynaClass</code> implementation class to support
+ * different dynamic classes, with different sets of properties.</p>
+ */
+ public String getName() {
+
+ return (this.getClass().getName());
+
+ }
+
+ /**
+ * <p>Return a property descriptor for the specified property, if it
+ * exists; otherwise, return <code>null</code>.</p>
+ *
+ * @param name Name of the dynamic property for which a descriptor
+ * is requested
+ *
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public DynaProperty getDynaProperty(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("No property name specified");
+ }
+ return ((DynaProperty) propertiesMap.get(name));
+
+ }
+
+ /**
+ * <p>Return an array of <code>ProperyDescriptors</code> for the properties
+ * currently defined in this DynaClass. If no properties are defined, a
+ * zero-length array will be returned.</p>
+ */
+ public DynaProperty[] getDynaProperties() {
+
+ return (properties);
+
+ }
+
+ /**
+ * <p>Instantiate and return a new DynaBean instance, associated
+ * with this DynaClass. <strong>NOTE</strong> - This operation is not
+ * supported, and throws an exception.</p>
+ *
+ * @exception IllegalAccessException if the Class or the appropriate
+ * constructor is not accessible
+ * @exception InstantiationException if this Class represents an abstract
+ * class, an array class, a primitive type, or void; or if instantiation
+ * fails for some other reason
+ */
+ public DynaBean newInstance()
+ throws IllegalAccessException, InstantiationException {
+
+ throw new UnsupportedOperationException("newInstance() not supported");
+
+ }
+
+ /**
+ * <p>Loads and returns the <code>Class</code> of the given name.
+ * By default, a load from the thread context class loader is attempted.
+ * If there is no such class loader, the class loader used to load this
+ * class will be utilized.</p>
+ *
+ * @exception SQLException if an exception was thrown trying to load
+ * the specified class
+ */
+ protected Class loadClass(String className) throws SQLException {
+
+ try {
+ ClassLoader cl = Thread.currentThread().getContextClassLoader();
+ if (cl == null) {
+ cl = this.getClass().getClassLoader();
+ }
+ return (cl.loadClass(className));
+ } catch (Exception e) {
+ throw new SQLException(
+ "Cannot load column class '" + className + "': " + e);
+ }
+
+ }
+
+ /**
+ * <p>Factory method to create a new DynaProperty for the given index
+ * into the result set metadata.</p>
+ *
+ * @param metadata is the result set metadata
+ * @param i is the column index in the metadata
+ * @return the newly created DynaProperty instance
+ */
+ protected DynaProperty createDynaProperty(
+ ResultSetMetaData metadata,
+ int i)
+ throws SQLException {
+
+ String name = null;
+ if (lowerCase) {
+ name = metadata.getColumnName(i).toLowerCase();
+ } else {
+ name = metadata.getColumnName(i);
+ }
+ String className = null;
+ try {
+ className = metadata.getColumnClassName(i);
+ } catch (SQLException e) {
+ // this is a patch for HsqlDb to ignore exceptions
+ // thrown by its metadata implementation
+ }
+
+ // Default to Object type if no class name could be retrieved
+ // from the metadata
+ Class clazz = Object.class;
+ if (className != null) {
+ clazz = loadClass(className);
+ }
+ return new DynaProperty(name, clazz);
+
+ }
+
+ /**
+ * <p>Introspect the metadata associated with our result set, and populate
+ * the <code>properties</code> and <code>propertiesMap</code> instance
+ * variables.</p>
+ *
+ * @param resultSet The <code>resultSet</code> whose metadata is to
+ * be introspected
+ *
+ * @exception SQLException if an error is encountered processing the
+ * result set metadata
+ */
+ protected void introspect(ResultSet resultSet) throws SQLException {
+
+ // Accumulate an ordered list of DynaProperties
+ ArrayList list = new ArrayList();
+ ResultSetMetaData metadata = resultSet.getMetaData();
+ int n = metadata.getColumnCount();
+ for (int i = 1; i <= n; i++) { // JDBC is one-relative!
+ DynaProperty dynaProperty = createDynaProperty(metadata, i);
+ if (dynaProperty != null) {
+ list.add(dynaProperty);
+ }
+ }
+
+ // Convert this list into the internal data structures we need
+ properties =
+ (DynaProperty[]) list.toArray(new DynaProperty[list.size()]);
+ for (int i = 0; i < properties.length; i++) {
+ propertiesMap.put(properties[i].getName(), properties[i]);
+ }
+
+ }
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/LazyDynaBean.java b/trunk/src/java/org/apache/commons/beanutils/LazyDynaBean.java
new file mode 100644
index 0000000..791a903
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/LazyDynaBean.java
@@ -0,0 +1,836 @@
+/*
+ * Copyright 2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.beanutils;
+
+import java.util.List;
+import java.util.ArrayList;
+import java.util.Map;
+import java.util.HashMap;
+import java.util.Date;
+import java.lang.reflect.Array;
+import java.math.BigDecimal;
+import java.math.BigInteger;
+import java.io.Serializable;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+/**
+ * <p>DynaBean which automatically adds properties to the <code>DynaClass</code>
+ * and provides <i>Lazy List</i> and <i>Lazy Map</i> features.</p>
+ *
+ * <p>DynaBeans deal with three types of properties - <i>simple</i>, <i>indexed</i> and <i>mapped</i> and
+ * have the following <code>get()</code> and <code>set()</code> methods for
+ * each of these types:</p>
+ * <ul>
+ * <li><i>Simple</i> property methods - <code>get(name)</code> and <code>set(name, value)</code></li>
+ * <li><i>Indexed</i> property methods - <code>get(name, index)</code> and <code>set(name, index, value)</code></li>
+ * <li><i>Mapped</i> property methods - <code>get(name, key)</code> and <code>set(name, key, value)</code></li>
+ * </ul>
+ *
+ * <p><b><u>Getting Property Values</u></b></p>
+ * <p>Calling any of the <code>get()</code> methods, for a property which
+ * doesn't exist, returns <code>null</code> in this implementation.</p>
+ *
+ * <p><b><u>Setting Simple Properties</u></b></p>
+ * <p>The <code>LazyDynaBean</code> will automatically add a property to the <code>DynaClass</code>
+ * if it doesn't exist when the <code>set(name, value)</code> method is called.</p>
+ *
+ * <code>DynaBean myBean = new LazyDynaBean();</code></br>
+ * <code>myBean.set("myProperty", "myValue");</code></br>
+ *
+ * <p><b><u>Setting Indexed Properties</u></b></p>
+ * <p>If the property <b>doesn't</b> exist, the <code>LazyDynaBean</code> will automatically add
+ * a property with an <code>ArrayList</code> type to the <code>DynaClass</code> when
+ * the <code>set(name, index, value)</code> method is called.
+ * It will also instantiate a new <code>ArrayList</code> and automatically <i>grow</i>
+ * the <code>List</code> so that it is big enough to accomodate the index being set.
+ * <code>ArrayList</code> is the default indexed property that LazyDynaBean uses but
+ * this can be easily changed by overriding the <code>newIndexedProperty(name)</code>
+ * method.</p>
+ *
+ * <code>DynaBean myBean = new LazyDynaBean();</code></br>
+ * <code>myBean.set("myIndexedProperty", 0, "myValue1");</code></br>
+ * <code>myBean.set("myIndexedProperty", 1, "myValue2");</code></br>
+ *
+ * <p>If the indexed property <b>does</b> exist in the <code>DynaClass</code> but is set to
+ * <code>null</code> in the <code>LazyDynaBean</code>, then it will instantiate a
+ * new <code>List</code> or <code>Array</code> as specified by the property's type
+ * in the <code>DynaClass</code> and automatically <i>grow</i> the <code>List</code>
+ * or <code>Array</code> so that it is big enough to accomodate the index being set.</p>
+ *
+ * <code>DynaBean myBean = new LazyDynaBean();</code></br>
+ * <code>MutableDynaClass myClass = (MutableDynaClass)myBean.getDynaClass();</code></br>
+ * <code>myClass.add("myIndexedProperty", int[].class);</code></br>
+ * <code>myBean.set("myIndexedProperty", 0, new Integer(10));</code></br>
+ * <code>myBean.set("myIndexedProperty", 1, new Integer(20));</code></br>
+ *
+ * <p><b><u>Setting Mapped Properties</u></b></p>
+ * <p>If the property <b>doesn't</b> exist, the <code>LazyDynaBean</code> will automatically add
+ * a property with a <code>HashMap</code> type to the <code>DynaClass</code> and
+ * instantiate a new <code>HashMap</code> in the DynaBean when the
+ * <code>set(name, key, value)</code> method is called. <code>HashMap</code> is the default
+ * mapped property that LazyDynaBean uses but this can be easily changed by overriding
+ * the <code>newMappedProperty(name)</code> method.</p>
+ *
+ * <code>DynaBean myBean = new LazyDynaBean();</code></br>
+ * <code>myBean.set("myMappedProperty", "myKey", "myValue");</code></br>
+ *
+ * <p>If the mapped property <b>does</b> exist in the <code>DynaClass</code> but is set to
+ * <code>null</code> in the <code>LazyDynaBean</code>, then it will instantiate a
+ * new <code>Map</code> as specified by the property's type in the <code>DynaClass</code>.</p>
+ *
+ * <code>DynaBean myBean = new LazyDynaBean();</code></br>
+ * <code>MutableDynaClass myClass = (MutableDynaClass)myBean.getDynaClass();</code></br>
+ * <code>myClass.add("myMappedProperty", TreeMap.class);</code></br>
+ * <code>myBean.set("myMappedProperty", "myKey", "myValue");</code></br>
+ *
+ * <p><b><u><i>Restricted</i> DynaClass</u></b></p>
+ * <p><code>MutableDynaClass</code> have a facility to <i>restrict</i> the <code>DynaClass</code>
+ * so that its properties cannot be modified. If the <code>MutableDynaClass</code> is
+ * restricted then calling any of the <code>set()</code> methods for a property which
+ * doesn't exist will result in a <code>IllegalArgumentException</code> being thrown.</p>
+ *
+ * @see LazyDynaClass
+ * @author Niall Pemberton
+ */
+public class LazyDynaBean implements DynaBean, Serializable {
+
+
+ /**
+ * Commons Logging
+ */
+ private static Log logger = LogFactory.getLog(LazyDynaBean.class);
+
+ protected static final BigInteger BigInteger_ZERO = new BigInteger("0");
+ protected static final BigDecimal BigDecimal_ZERO = new BigDecimal("0");
+ protected static final Character Character_SPACE = new Character(' ');
+ protected static final Byte Byte_ZERO = new Byte((byte)0);
+ protected static final Short Short_ZERO = new Short((short)0);
+ protected static final Integer Integer_ZERO = new Integer(0);
+ protected static final Long Long_ZERO = new Long((long)0);
+ protected static final Float Float_ZERO = new Float((byte)0);
+ protected static final Double Double_ZERO = new Double((byte)0);
+
+ /**
+ * The <code>MutableDynaClass</code> "base class" that this DynaBean
+ * is associated with.
+ */
+ protected Map values;
+
+ /**
+ * The <code>MutableDynaClass</code> "base class" that this DynaBean
+ * is associated with.
+ */
+ protected MutableDynaClass dynaClass;
+
+
+ // ------------------- Constructors ----------------------------------
+
+ /**
+ * Construct a new <code>LazyDynaBean</code> with a <code>LazyDynaClass</code> instance.
+ */
+ public LazyDynaBean() {
+ this(new LazyDynaClass());
+ }
+
+ /**
+ * Construct a new <code>LazyDynaBean</code> with a <code>LazyDynaClass</code> instance.
+ *
+ * @param name Name of this DynaBean class
+ */
+ public LazyDynaBean(String name) {
+ this(new LazyDynaClass(name));
+ }
+
+ /**
+ * Construct a new <code>DynaBean</code> associated with the specified
+ * <code>DynaClass</code> instance - if its not a <code>MutableDynaClass</code>
+ * then a new <code>LazyDynaClass</code> is created and the properties copied.
+ *
+ * @param dynaClass The DynaClass we are associated with
+ */
+ public LazyDynaBean(DynaClass dynaClass) {
+
+ values = newMap();
+
+ if (dynaClass instanceof MutableDynaClass) {
+ this.dynaClass = (MutableDynaClass)dynaClass;
+ } else {
+ this.dynaClass = new LazyDynaClass(dynaClass.getName(), dynaClass.getDynaProperties());
+ }
+
+ }
+
+
+ // ------------------- Public Methods ----------------------------------
+
+ /**
+ * Return the Map backing this <code>DynaBean</code>
+ */
+ public Map getMap() {
+ return values;
+ }
+
+ /**
+ * <p>Return the size of an indexed or mapped property.</p>
+ *
+ * @param name Name of the property
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public int size(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("No property name specified");
+ }
+
+ Object value = values.get(name);
+ if (value == null) {
+ return 0;
+ }
+
+ if (value instanceof Map) {
+ return ((Map)value).size();
+ }
+
+ if (value instanceof List) {
+ return ((List)value).size();
+ }
+
+ if ((value.getClass().isArray())) {
+ return Array.getLength(value);
+ }
+
+ return 0;
+
+ }
+
+ // ------------------- DynaBean Methods ----------------------------------
+
+ /**
+ * Does the specified mapped property contain a value for the specified
+ * key value?
+ *
+ * @param name Name of the property to check
+ * @param key Name of the key to check
+ *
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public boolean contains(String name, String key) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("No property name specified");
+ }
+
+ Object value = values.get(name);
+ if (value == null) {
+ return false;
+ }
+
+ if (value instanceof Map) {
+ return (((Map) value).containsKey(key));
+ }
+
+ return false;
+
+ }
+
+ /**
+ * <p>Return the value of a simple property with the specified name.</p>
+ *
+ * <p><strong>N.B.</strong> Returns <code>null</code> if there is no property
+ * of the specified name.</p>
+ *
+ * @param name Name of the property whose value is to be retrieved.
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public Object get(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("No property name specified");
+ }
+
+ // Value found
+ Object value = values.get(name);
+ if (value != null) {
+ return value;
+ }
+
+ // Property doesn't exist
+ if (!isDynaProperty(name)) {
+ return null;
+ }
+
+ // Property doesn't exist
+ value = createProperty(name, dynaClass.getDynaProperty(name).getType());
+
+ if (value != null) {
+ set(name, value);
+ }
+
+ return value;
+
+ }
+
+ /**
+ * <p>Return the value of an indexed property with the specified name.</p>
+ *
+ * <p><strong>N.B.</strong> Returns <code>null</code> if there is no 'indexed'
+ * property of the specified name.</p>
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param index Index of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ */
+ public Object get(String name, int index) {
+
+ // If its not a property, then create default indexed property
+ if (!isDynaProperty(name)) {
+ set(name, defaultIndexedProperty(name));
+ }
+
+ // Get the indexed property
+ Object indexedProperty = get(name);
+
+ // Check that the property is indexed
+ if (!dynaClass.getDynaProperty(name).isIndexed()) {
+ throw new IllegalArgumentException
+ ("Non-indexed property for '" + name + "[" + index + "]' "
+ + dynaClass.getDynaProperty(name).getName());
+ }
+
+ // Grow indexed property to appropriate size
+ indexedProperty = growIndexedProperty(name, indexedProperty, index);
+
+ // Return the indexed value
+ if (indexedProperty.getClass().isArray()) {
+ return Array.get(indexedProperty, index);
+ } else if (indexedProperty instanceof List) {
+ return ((List)indexedProperty).get(index);
+ } else {
+ throw new IllegalArgumentException
+ ("Non-indexed property for '" + name + "[" + index + "]' "
+ + indexedProperty.getClass().getName());
+ }
+
+ }
+
+ /**
+ * <p>Return the value of a mapped property with the specified name.</p>
+ *
+ * <p><strong>N.B.</strong> Returns <code>null</code> if there is no 'mapped'
+ * property of the specified name.</p>
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param key Key of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public Object get(String name, String key) {
+
+ // If its not a property, then create default mapped property
+ if (!isDynaProperty(name)) {
+ set(name, defaultMappedProperty(name));
+ }
+
+ // Get the mapped property
+ Object mappedProperty = get(name);
+
+ // Check that the property is mapped
+ if (!dynaClass.getDynaProperty(name).isMapped()) {
+ throw new IllegalArgumentException
+ ("Non-mapped property for '" + name + "(" + key + ")' "
+ + dynaClass.getDynaProperty(name).getType().getName());
+ }
+
+ // Get the value from the Map
+ if (mappedProperty instanceof Map) {
+ return (((Map) mappedProperty).get(key));
+ } else {
+ throw new IllegalArgumentException
+ ("Non-mapped property for '" + name + "(" + key + ")'"
+ + mappedProperty.getClass().getName());
+ }
+
+ }
+
+
+ /**
+ * Return the <code>DynaClass</code> instance that describes the set of
+ * properties available for this DynaBean.
+ */
+ public DynaClass getDynaClass() {
+ return (DynaClass)dynaClass;
+ }
+
+ /**
+ * Remove any existing value for the specified key on the
+ * specified mapped property.
+ *
+ * @param name Name of the property for which a value is to
+ * be removed
+ * @param key Key of the value to be removed
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public void remove(String name, String key) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("No property name specified");
+ }
+
+ Object value = values.get(name);
+ if (value == null) {
+ return;
+ }
+
+ if (value instanceof Map) {
+ ((Map) value).remove(key);
+ } else {
+ throw new IllegalArgumentException
+ ("Non-mapped property for '" + name + "(" + key + ")'"
+ + value.getClass().getName());
+ }
+
+ }
+
+ /**
+ * Set the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception IllegalArgumentException if this is not an existing property
+ * name for our DynaClass and the MutableDynaClass is restricted
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception NullPointerException if an attempt is made to set a
+ * primitive property to null
+ */
+ public void set(String name, Object value) {
+
+ // If the property doesn't exist, then add it
+ if (!isDynaProperty(name)) {
+
+ if (dynaClass.isRestricted()) {
+ throw new IllegalArgumentException
+ ("Invalid property name '" + name + "' (DynaClass is restricted)");
+ }
+ if (value == null) {
+ dynaClass.add(name);
+ } else {
+ dynaClass.add(name, value.getClass());
+ }
+
+ }
+
+ DynaProperty descriptor = dynaClass.getDynaProperty(name);
+
+ if (value == null) {
+ if (descriptor.getType().isPrimitive()) {
+ throw new NullPointerException
+ ("Primitive value for '" + name + "'");
+ }
+ } else if (!isAssignable(descriptor.getType(), value.getClass())) {
+ throw new ConversionException
+ ("Cannot assign value of type '" +
+ value.getClass().getName() +
+ "' to property '" + name + "' of type '" +
+ descriptor.getType().getName() + "'");
+ }
+
+ // Set the property's value
+ values.put(name, value);
+
+ }
+
+ /**
+ * Set the value of an indexed property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param index Index of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ */
+ public void set(String name, int index, Object value) {
+
+ // If its not a property, then create default indexed property
+ if (!isDynaProperty(name)) {
+ set(name, defaultIndexedProperty(name));
+ }
+
+ // Get the indexed property
+ Object indexedProperty = get(name);
+
+ // Check that the property is indexed
+ if (!dynaClass.getDynaProperty(name).isIndexed()) {
+ throw new IllegalArgumentException
+ ("Non-indexed property for '" + name + "[" + index + "]'"
+ + dynaClass.getDynaProperty(name).getType().getName());
+ }
+
+ // Grow indexed property to appropriate size
+ indexedProperty = growIndexedProperty(name, indexedProperty, index);
+
+ // Set the value in an array
+ if (indexedProperty.getClass().isArray()) {
+ Array.set(indexedProperty, index, value);
+ } else if (indexedProperty instanceof List) {
+ ((List)indexedProperty).set(index, value);
+ } else {
+ throw new IllegalArgumentException
+ ("Non-indexed property for '" + name + "[" + index + "]' "
+ + indexedProperty.getClass().getName());
+ }
+
+ }
+
+ /**
+ * Set the value of a mapped property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param key Key of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public void set(String name, String key, Object value) {
+
+ // If the 'mapped' property doesn't exist, then add it
+ if (!isDynaProperty(name)) {
+ set(name, defaultMappedProperty(name));
+ }
+
+ // Get the mapped property
+ Object mappedProperty = get(name);
+
+ // Check that the property is mapped
+ if (!dynaClass.getDynaProperty(name).isMapped()) {
+ throw new IllegalArgumentException
+ ("Non-mapped property for '" + name + "(" + key + ")'"
+ + dynaClass.getDynaProperty(name).getType().getName());
+ }
+
+ // Set the value in the Map
+ ((Map)mappedProperty).put(key, value);
+
+ }
+
+ // ------------------- protected Methods ----------------------------------
+
+ protected Object growIndexedProperty(String name, Object indexedProperty, int index) {
+
+ // Grow a List to the appropriate size
+ if (indexedProperty instanceof List) {
+
+ List list = (List)indexedProperty;
+ while (index >= list.size()) {
+ list.add(null);
+ }
+
+ }
+
+ // Grow an Array to the appropriate size
+ if ((indexedProperty.getClass().isArray())) {
+
+ int length = Array.getLength(indexedProperty);
+ if (index >= length) {
+ Class componentType = indexedProperty.getClass().getComponentType();
+ Object newArray = Array.newInstance(componentType, (index + 1));
+ System.arraycopy(indexedProperty, 0, newArray, 0, length);
+ indexedProperty = newArray;
+ set(name, indexedProperty);
+ int newLength = Array.getLength(indexedProperty);
+ for (int i = length; i < newLength; i++) {
+ Array.set(indexedProperty, i, createProperty(name+"["+i+"]", componentType));
+ }
+ }
+ }
+
+ return indexedProperty;
+
+ }
+
+ /**
+ * Create a new Instance of a Property
+ */
+ protected Object createProperty(String name, Class type) {
+
+ // Create Lists, arrays or DynaBeans
+ if (type.isArray() || List.class.isAssignableFrom(type)) {
+ return createIndexedProperty(name, type);
+ }
+
+ if (Map.class.isAssignableFrom(type)) {
+ return createMappedProperty(name, type);
+ }
+
+ if (DynaBean.class.isAssignableFrom(type)) {
+ return createDynaBeanProperty(name, type);
+ }
+
+ if (type.isPrimitive()) {
+ return createPrimitiveProperty(name, type);
+ }
+
+ if (Number.class.isAssignableFrom(type)) {
+ return createNumberProperty(name, type);
+ }
+
+ return createOtherProperty(name, type);
+
+ }
+
+ /**
+ * Create a new Instance of an 'Indexed' Property
+ */
+ protected Object createIndexedProperty(String name, Class type) {
+
+ // Create the indexed object
+ Object indexedProperty = null;
+
+ if (type == null) {
+
+ indexedProperty = defaultIndexedProperty(name);
+
+ } else if (type.isArray()) {
+
+ indexedProperty = Array.newInstance(type.getComponentType(), 0);
+
+ } else if (List.class.isAssignableFrom(type)) {
+ if (type.isInterface()) {
+ indexedProperty = defaultIndexedProperty(name);
+ } else {
+ try {
+ indexedProperty = type.newInstance();
+ }
+ catch (Exception ex) {
+ throw new IllegalArgumentException
+ ("Error instantiating indexed property of type '" +
+ type.getName() + "' for '" + name + "' " + ex);
+ }
+ }
+ } else {
+
+ throw new IllegalArgumentException
+ ("Non-indexed property of type '" + type.getName() + "' for '" + name + "'");
+ }
+
+ return indexedProperty;
+
+ }
+
+ /**
+ * Create a new Instance of a 'Mapped' Property
+ */
+ protected Object createMappedProperty(String name, Class type) {
+
+ // Create the mapped object
+ Object mappedProperty = null;
+
+ if (type == null) {
+
+ mappedProperty = defaultMappedProperty(name);
+
+ } else if (type.isInterface()) {
+
+ mappedProperty = defaultMappedProperty(name);
+
+ } else if (Map.class.isAssignableFrom(type)) {
+ try {
+ mappedProperty = type.newInstance();
+ }
+ catch (Exception ex) {
+ throw new IllegalArgumentException
+ ("Error instantiating mapped property of type '" + type.getName() + "' for '" + name + "' " + ex);
+ }
+ } else {
+
+ throw new IllegalArgumentException
+ ("Non-mapped property of type '" + type.getName() + "' for '" + name + "'");
+ }
+
+ return mappedProperty;
+
+ }
+
+ /**
+ * Create a new Instance of a 'Mapped' Property
+ */
+ protected Object createDynaBeanProperty(String name, Class type) {
+ try {
+ return type.newInstance();
+ }
+ catch (Exception ex) {
+ if (logger.isWarnEnabled()) {
+ logger.warn("Error instantiating DynaBean property of type '" + type.getName() + "' for '" + name + "' " + ex);
+ }
+ return null;
+ }
+ }
+
+ /**
+ * Create a new Instance of a 'Primitive' Property
+ */
+ protected Object createPrimitiveProperty(String name, Class type) {
+
+ if (type == Boolean.TYPE) {
+ return Boolean.FALSE;
+ } else if (type == Integer.TYPE) {
+ return Integer_ZERO;
+ } else if (type == Long.TYPE) {
+ return Long_ZERO;
+ } else if (type == Double.TYPE) {
+ return Double_ZERO;
+ } else if (type == Float.TYPE) {
+ return Float_ZERO;
+ } else if (type == Byte.TYPE) {
+ return Byte_ZERO;
+ } else if (type == Short.TYPE) {
+ return Short_ZERO;
+ } else if (type == Character.TYPE) {
+ return Character_SPACE;
+ } else {
+ return null;
+ }
+
+ }
+
+ /**
+ * Create a new Instance of a 'Primitive' Property
+ */
+ protected Object createNumberProperty(String name, Class type) {
+
+ return null;
+
+ }
+
+ /**
+ * Create a new Instance of a 'Mapped' Property
+ */
+ protected Object createOtherProperty(String name, Class type) {
+
+ if (type == String.class || type == Boolean.class ||
+ type == Character.class || Date.class.isAssignableFrom(type)) {
+ return null;
+ }
+
+ try {
+ return type.newInstance();
+ }
+ catch (Exception ex) {
+ if (logger.isWarnEnabled()) {
+ logger.warn("Error instantiating property of type '" + type.getName() + "' for '" + name + "' " + ex);
+ }
+ return null;
+ }
+ }
+
+ /**
+ * <p>Creates a new <code>ArrayList</code> for an 'indexed' property
+ * which doesn't exist.</p>
+ *
+ * <p>This method shouls be overriden if an alternative <code>List</code>
+ * or <code>Array</code> implementation is required for 'indexed' properties.</p>
+ *
+ * @param name Name of the 'indexed property.
+ */
+ protected Object defaultIndexedProperty(String name) {
+ return new ArrayList();
+ }
+
+ /**
+ * <p>Creates a new <code>HashMap</code> for a 'mapped' property
+ * which doesn't exist.</p>
+ *
+ * <p>This method can be overriden if an alternative <code>Map</code>
+ * implementation is required for 'mapped' properties.</p>
+ *
+ * @param name Name of the 'mapped property.
+ */
+ protected Map defaultMappedProperty(String name) {
+ return new HashMap();
+ }
+
+ /**
+ * Indicates if there is a property with the specified name.
+ */
+ protected boolean isDynaProperty(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("No property name specified");
+ }
+
+ // Handle LazyDynaClasses
+ if (dynaClass instanceof LazyDynaClass) {
+ return ((LazyDynaClass)dynaClass).isDynaProperty(name);
+ }
+
+ // Handle other MutableDynaClass
+ return dynaClass.getDynaProperty(name) == null ? false : true;
+
+ }
+
+ /**
+ * Is an object of the source class assignable to the destination class?
+ *
+ * @param dest Destination class
+ * @param source Source class
+ */
+ protected boolean isAssignable(Class dest, Class source) {
+
+ if (dest.isAssignableFrom(source) ||
+ ((dest == Boolean.TYPE) && (source == Boolean.class)) ||
+ ((dest == Byte.TYPE) && (source == Byte.class)) ||
+ ((dest == Character.TYPE) && (source == Character.class)) ||
+ ((dest == Double.TYPE) && (source == Double.class)) ||
+ ((dest == Float.TYPE) && (source == Float.class)) ||
+ ((dest == Integer.TYPE) && (source == Integer.class)) ||
+ ((dest == Long.TYPE) && (source == Long.class)) ||
+ ((dest == Short.TYPE) && (source == Short.class))) {
+ return (true);
+ } else {
+ return (false);
+ }
+
+ }
+
+ /**
+ * <p>Creates a new instance of the <code>Map</code>.</p>
+ */
+ protected Map newMap() {
+ return new HashMap();
+ }
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/LazyDynaClass.java b/trunk/src/java/org/apache/commons/beanutils/LazyDynaClass.java
new file mode 100644
index 0000000..99fc35f
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/LazyDynaClass.java
@@ -0,0 +1,337 @@
+/*
+ * Copyright 2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.beanutils;
+
+/**
+ * <p>DynaClass which implements the <code>MutableDynaClass</code> interface.</p>
+ *
+ * <p>A <code>MutableDynaClass</code> is a specialized extension to <code>DynaClass</code>
+ * that allows properties to be added or removed dynamically.</p>
+ *
+ * <p>This implementation has one slightly unusual default behaviour - calling
+ * the <code>getDynaProperty(name)</code> method for a property which doesn't
+ * exist returns a <code>DynaProperty</code> rather than <code>null</code>. The
+ * reason for this is that <code>BeanUtils</code> calls this method to check if
+ * a property exists before trying to set the value. This would defeat the object
+ * of the <code>LazyDynaBean</code> which automatically adds missing properties
+ * when any of its <code>set()</code> methods are called. For this reason the
+ * <code>isDynaProperty(name)</code> method has been added to this implementation
+ * in order to determine if a property actually exists. If the more <i>normal</i>
+ * behaviour of returning <code>null</code> is required, then this can be achieved
+ * by calling the <code>setReturnNull(true)</code>.</p>
+ *
+ * <p>The <code>add(name, type, readable, writable)</code> method is not implemented
+ * and always throws an <code>UnsupportedOperationException</code>. I believe
+ * this attributes need to be added to the <code>DynaProperty</code> class
+ * in order to control read/write facilities.</p>
+ *
+ * @see LazyDynaBean
+ * @author Niall Pemberton
+ */
+public class LazyDynaClass extends BasicDynaClass implements MutableDynaClass {
+
+ /**
+ * Controls whether changes to this DynaClass's properties are allowed.
+ */
+ protected boolean restricted;
+
+ /**
+ * <p>Controls whether the <code>getDynaProperty()</code> method returns
+ * null if a property doesn't exist - or creates a new one.</p>
+ *
+ * <p>Default is <code>false</code>.
+ */
+ protected boolean returnNull = false;
+
+ /**
+ * Construct a new LazyDynaClass with default parameters.
+ */
+ public LazyDynaClass() {
+ this(null, (DynaProperty[])null);
+ }
+
+ /**
+ * Construct a new LazyDynaClass with the specified name.
+ *
+ * @param name Name of this DynaBean class
+ */
+ public LazyDynaClass(String name) {
+ this(name, (DynaProperty[])null);
+ }
+
+ /**
+ * Construct a new LazyDynaClass with the specified name and DynaBean class.
+ *
+ * @param name Name of this DynaBean class
+ * @param dynaBeanClass The implementation class for new instances
+ */
+ public LazyDynaClass(String name, Class dynaBeanClass) {
+ this(name, dynaBeanClass, null);
+ }
+
+ /**
+ * Construct a new LazyDynaClass with the specified name and properties.
+ *
+ * @param name Name of this DynaBean class
+ * @param properties Property descriptors for the supported properties
+ */
+ public LazyDynaClass(String name, DynaProperty[] properties) {
+ this(name, LazyDynaBean.class, properties);
+ }
+
+ /**
+ * Construct a new LazyDynaClass with the specified name, DynaBean class and properties.
+ *
+ * @param name Name of this DynaBean class
+ * @param dynaBeanClass The implementation class for new intances
+ * @param properties Property descriptors for the supported properties
+ */
+ public LazyDynaClass(String name, Class dynaBeanClass, DynaProperty properties[]) {
+ super(name, dynaBeanClass, properties);
+ }
+
+ /**
+ * <p>Is this DynaClass currently restricted.</p>
+ * <p>If restricted, no changes to the existing registration of
+ * property names, data types, readability, or writeability are allowed.</p>
+ */
+ public boolean isRestricted() {
+ return restricted;
+ }
+
+ /**
+ * <p>Set whether this DynaClass is currently restricted.</p>
+ * <p>If restricted, no changes to the existing registration of
+ * property names, data types, readability, or writeability are allowed.</p>
+ */
+ public void setRestricted(boolean restricted) {
+ this.restricted = restricted;
+ }
+
+ /**
+ * Should this DynaClass return a <code>null</code> from
+ * the <code>getDynaProperty(name)</code> method if the property
+ * doesn't exist.
+ */
+ public boolean isReturnNull() {
+ return returnNull;
+ }
+
+ /**
+ * Set whether this DynaClass should return a <code>null</code> from
+ * the <code>getDynaProperty(name)</code> method if the property
+ * doesn't exist.
+ */
+ public void setReturnNull(boolean returnNull) {
+ this.returnNull = returnNull;
+ }
+
+ /**
+ * Add a new dynamic property with no restrictions on data type,
+ * readability, or writeability.
+ *
+ * @param name Name of the new dynamic property
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no new properties can be added
+ */
+ public void add(String name) {
+ add(new DynaProperty(name));
+ }
+
+ /**
+ * Add a new dynamic property with the specified data type, but with
+ * no restrictions on readability or writeability.
+ *
+ * @param name Name of the new dynamic property
+ * @param type Data type of the new dynamic property (null for no
+ * restrictions)
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no new properties can be added
+ */
+ public void add(String name, Class type) {
+ add(new DynaProperty(name, type));
+ }
+
+ /**
+ * <p>Add a new dynamic property with the specified data type, readability,
+ * and writeability.</p>
+ *
+ * <p><strong>N.B.</strong>Support for readable/writeable properties has not been implemented
+ * and this method always throws a <code>UnsupportedOperationException</code>.</p>
+ *
+ * <p>I'm not sure the intention of the original authors for this method, but it seems to
+ * me that readable/writable should be attributes of the <code>DynaProperty</code> class
+ * (which they are not) and is the reason this method has not been implemented.</p>
+ *
+ * @param name Name of the new dynamic property
+ * @param type Data type of the new dynamic property (null for no
+ * restrictions)
+ * @param readable Set to <code>true</code> if this property value
+ * should be readable
+ * @param writeable Set to <code>true</code> if this property value
+ * should be writeable
+ *
+ * @exception UnsupportedOperationException anytime this method is called
+ */
+ public void add(String name, Class type, boolean readable, boolean writeable) {
+ throw new java.lang.UnsupportedOperationException("readable/writable properties not supported");
+ }
+
+ /**
+ * Add a new dynamic property.
+ *
+ * @param property Property the new dynamic property to add.
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no new properties can be added
+ */
+ protected void add(DynaProperty property) {
+
+ if (property.getName() == null) {
+ throw new IllegalArgumentException("Property name is missing.");
+ }
+
+ if (isRestricted()) {
+ throw new IllegalStateException("DynaClass is currently restricted. No new properties can be added.");
+ }
+
+ // Check if property already exists
+ if (propertiesMap.get(property.getName()) != null) {
+ return;
+ }
+
+ // Create a new property array with the specified property
+ DynaProperty[] oldProperties = getDynaProperties();
+ DynaProperty[] newProperties = new DynaProperty[oldProperties.length+1];
+ System.arraycopy(oldProperties, 0, newProperties, 0, oldProperties.length);
+ newProperties[oldProperties.length] = property;
+
+ // Update the properties
+ setProperties(newProperties);
+
+ }
+
+ /**
+ * Remove the specified dynamic property, and any associated data type,
+ * readability, and writeability, from this dynamic class.
+ * <strong>NOTE</strong> - This does <strong>NOT</strong> cause any
+ * corresponding property values to be removed from DynaBean instances
+ * associated with this DynaClass.
+ *
+ * @param name Name of the dynamic property to remove
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no properties can be removed
+ */
+ public void remove(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("Property name is missing.");
+ }
+
+ if (isRestricted()) {
+ throw new IllegalStateException("DynaClass is currently restricted. No properties can be removed.");
+ }
+
+ // Ignore if property doesn't exist
+ if (propertiesMap.get(name) == null) {
+ return;
+ }
+
+
+ // Create a new property array of without the specified property
+ DynaProperty[] oldProperties = getDynaProperties();
+ DynaProperty[] newProperties = new DynaProperty[oldProperties.length-1];
+ int j = 0;
+ for (int i = 0; i < oldProperties.length; i++) {
+ if (!(name.equals(oldProperties[i].getName()))) {
+ newProperties[j] = oldProperties[i];
+ j++;
+ }
+ }
+
+ // Update the properties
+ setProperties(newProperties);
+
+ }
+
+ /**
+ * <p>Return a property descriptor for the specified property.</p>
+ *
+ * <p>If the property is not found and the <code>returnNull</code> indicator is
+ * <code>true</code>, this method always returns <code>null</code>.</p>
+ *
+ * <p>If the property is not found and the <code>returnNull</code> indicator is
+ * <code>false</code> a new property descriptor is created and returned (although
+ * its not actually added to the DynaClass's properties). This is the default
+ * beahviour.</p>
+ *
+ * <p>The reason for not returning a <code>null</code> property descriptor is that
+ * <code>BeanUtils</code> uses this method to check if a property exists
+ * before trying to set it - since these <i>Lazy</i> implementations automatically
+ * add any new properties when they are set, returning <code>null</code> from
+ * this method would defeat their purpose.</p>
+ *
+ * @param name Name of the dynamic property for which a descriptor
+ * is requested
+ *
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public DynaProperty getDynaProperty(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("Property name is missing.");
+ }
+
+ DynaProperty dynaProperty = (DynaProperty)propertiesMap.get(name);
+
+ // If it doesn't exist and returnNull is false
+ // create a new DynaProperty
+ if (dynaProperty == null && !isReturnNull() && !isRestricted()) {
+ dynaProperty = new DynaProperty(name);
+ }
+
+ return dynaProperty;
+
+ }
+
+ /**
+ * <p>Indicate whether a property actually exists.</p>
+ *
+ * <p><strong>N.B.</strong> Using <code>getDynaProperty(name) == null</code>
+ * doesn't work in this implementation because that method might
+ * return a DynaProperty if it doesn't exist (depending on the
+ * <code>returnNull</code> indicator).</p>
+ *
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public boolean isDynaProperty(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("Property name is missing.");
+ }
+
+ return propertiesMap.get(name) == null ? false : true;
+
+ }
+
+}
\ No newline at end of file
diff --git a/trunk/src/java/org/apache/commons/beanutils/LazyDynaMap.java b/trunk/src/java/org/apache/commons/beanutils/LazyDynaMap.java
new file mode 100644
index 0000000..3e8cca3
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/LazyDynaMap.java
@@ -0,0 +1,428 @@
+/*
+ * Copyright 2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.beanutils;
+
+import java.util.Map;
+import java.util.Iterator;
+
+/**
+ * <p>Provides a <i>light weight</i> <code>DynaBean</code> facade to a <code>Map</code> with <i>lazy</i> map/list processing.</p>
+ *
+ * <p>Its a <i>light weight</i> <code>DynaBean</code> implementation because there is no
+ * actual <code>DynaClass</code> associated with this <code>DynaBean</code> - in fact
+ * it implements the <code>DynaClass</code> interface itself providing <i>pseudo</i> DynaClass
+ * behaviour from the actual values stored in the <code>Map</code>.</p>
+ *
+ * <p>As well providing rhe standard <code>DynaBean</code> access to the <code>Map</code>'s properties
+ * this class also provides the usual <i>Lazy</i> behaviour:</p>
+ * <ul>
+ * <li>Properties don't need to be pre-defined in a <code>DynaClass</code></li>
+ * <li>Indexed properties (<code>Lists</code> or <code>Arrays</code>) are automatically instantiated
+ * and <i>grown</i> so that they are large enough to cater for the index being set.</li>
+ * <li>Mapped properties are automatically instantiated.</li>
+ * </ul>
+ *
+ * <p><b><u><i>Restricted</i> DynaClass</u></b></p>
+ * <p>This class implements the <code>MutableDynaClass</code> interface.
+ * <code>MutableDynaClass</code> have a facility to <i>restrict</i> the <code>DynaClass</code>
+ * so that its properties cannot be modified. If the <code>MutableDynaClass</code> is
+ * restricted then calling any of the <code>set()</code> methods for a property which
+ * doesn't exist will result in a <code>IllegalArgumentException</code> being thrown.</p>
+ *
+ * @author Niall Pemberton
+ */
+public class LazyDynaMap extends LazyDynaBean implements MutableDynaClass {
+
+ /**
+ * The name of this DynaClass (analogous to the
+ * <code>getName()</code> method of <code>java.lang.Class</code>).
+ */
+ protected String name;
+
+ /**
+ * Controls whether changes to this DynaClass's properties are allowed.
+ */
+ protected boolean restricted;
+
+ /**
+ * <p>Controls whether the <code>getDynaProperty()</code> method returns
+ * null if a property doesn't exist - or creates a new one.</p>
+ *
+ * <p>Default is <code>false</code>.
+ */
+ protected boolean returnNull = false;
+
+
+ // ------------------- Constructors ----------------------------------
+
+ /**
+ * Default Constructor.
+ */
+ public LazyDynaMap() {
+ this(null, (Map)null);
+ }
+
+ /**
+ * Construct a new <code>LazyDynaMap</code> with the specified name.
+ *
+ * @param name Name of this DynaBean class
+ */
+ public LazyDynaMap(String name) {
+ this(name, (Map)null);
+ }
+
+ /**
+ * Construct a new <code>LazyDynaMap</code> with the specified <code>Map</code>.
+ *
+ * @param values The Map backing this <code>LazyDynaMap</code>
+ */
+ public LazyDynaMap(Map values) {
+ this(null, values);
+ }
+
+ /**
+ * Construct a new <code>LazyDynaMap</code> with the specified name and <code>Map</code>.
+ *
+ * @param name Name of this DynaBean class
+ * @param values The Map backing this <code>LazyDynaMap</code>
+ */
+ public LazyDynaMap(String name, Map values) {
+ this.name = name == null ? "LazyDynaMap" : name;
+ this.values = values == null ? newMap() : values;
+ this.dynaClass = this;
+ }
+
+ /**
+ * Construct a new <code>LazyDynaMap</code> with the specified properties.
+ *
+ * @param properties Property descriptors for the supported properties
+ */
+ public LazyDynaMap(DynaProperty[] properties) {
+ this(null, properties);
+ }
+
+ /**
+ * Construct a new <code>LazyDynaMap</code> with the specified name and properties.
+ *
+ * @param name Name of this DynaBean class
+ * @param properties Property descriptors for the supported properties
+ */
+ public LazyDynaMap(String name, DynaProperty[] properties) {
+ this(name, (Map)null);
+ if (properties != null) {
+ for (int i = 0; i < properties.length; i++) {
+ add(properties[i]);
+ }
+ }
+ }
+
+ /**
+ * Construct a new <code>LazyDynaMap</code> based on an exisiting DynaClass
+ *
+ * @param dynaClass DynaClass to copy the name and properties from
+ */
+ public LazyDynaMap(DynaClass dynaClass) {
+ this(dynaClass.getName(), dynaClass.getDynaProperties());
+ }
+
+ // ------------------- Public Methods ----------------------------------
+
+ /**
+ * Set the Map backing this <code>DynaBean</code>
+ */
+ public void setMap(Map values) {
+ this.values = values;
+ }
+
+ // ------------------- DynaBean Methods ----------------------------------
+
+ /**
+ * Set the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param value Value to which this property is to be set
+ */
+ public void set(String name, Object value) {
+
+ if (isRestricted() && !values.containsKey(name)) {
+ throw new IllegalArgumentException
+ ("Invalid property name '" + name + "' (DynaClass is restricted)");
+ }
+
+ values.put(name, value);
+
+ }
+
+ // ------------------- DynaClass Methods ----------------------------------
+
+ /**
+ * Return the name of this DynaClass (analogous to the
+ * <code>getName()</code> method of <code>java.lang.Class</code)
+ */
+ public String getName() {
+ return this.name;
+ }
+
+ /**
+ * <p>Return a property descriptor for the specified property.</p>
+ *
+ * <p>If the property is not found and the <code>returnNull</code> indicator is
+ * <code>true</code>, this method always returns <code>null</code>.</p>
+ *
+ * <p>If the property is not found and the <code>returnNull</code> indicator is
+ * <code>false</code> a new property descriptor is created and returned (although
+ * its not actually added to the DynaClass's properties). This is the default
+ * beahviour.</p>
+ *
+ * <p>The reason for not returning a <code>null</code> property descriptor is that
+ * <code>BeanUtils</code> uses this method to check if a property exists
+ * before trying to set it - since these <i>Map</i> implementations automatically
+ * add any new properties when they are set, returning <code>null</code> from
+ * this method would defeat their purpose.</p>
+ *
+ * @param name Name of the dynamic property for which a descriptor
+ * is requested
+ *
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public DynaProperty getDynaProperty(String name) {
+
+ if (name == null)
+ throw new IllegalArgumentException("Property name is missing.");
+
+ // If it doesn't exist and returnNull is false
+ // create a new DynaProperty
+ if (!values.containsKey(name) && isReturnNull()) {
+ return null;
+ }
+
+ Object value = values.get(name);
+
+ if (value == null) {
+ return new DynaProperty(name);
+ } else {
+ return new DynaProperty(name, value.getClass());
+ }
+
+ }
+
+ /**
+ * <p>Return an array of <code>ProperyDescriptors</code> for the properties
+ * currently defined in this DynaClass. If no properties are defined, a
+ * zero-length array will be returned.</p>
+ *
+ * <p><strong>FIXME</strong> - Should we really be implementing
+ * <code>getBeanInfo()</code> instead, which returns property descriptors
+ * and a bunch of other stuff?</p>
+ */
+ public DynaProperty[] getDynaProperties() {
+
+ int i = 0;
+ DynaProperty[] properties = new DynaProperty[values.size()];
+ Iterator iterator = values.keySet().iterator();
+
+ while (iterator.hasNext()) {
+ String name = (String)iterator.next();
+ Object value = values.get(name);
+ properties[i++] = new DynaProperty(name, value == null ? null : value.getClass());
+ }
+
+ return properties;
+
+ }
+
+ /**
+ * Instantiate and return a new DynaBean instance, associated
+ * with this DynaClass.
+ */
+ public DynaBean newInstance() {
+ return new LazyDynaMap(this);
+ }
+
+
+ // ------------------- MutableDynaClass Methods ----------------------------------
+
+ /**
+ * <p>Is this DynaClass currently restricted.</p>
+ * <p>If restricted, no changes to the existing registration of
+ * property names, data types, readability, or writeability are allowed.</p>
+ */
+ public boolean isRestricted() {
+ return restricted;
+ }
+
+ /**
+ * <p>Set whether this DynaClass is currently restricted.</p>
+ * <p>If restricted, no changes to the existing registration of
+ * property names, data types, readability, or writeability are allowed.</p>
+ */
+ public void setRestricted(boolean restricted) {
+ this.restricted = restricted;
+ }
+
+ /**
+ * Add a new dynamic property with no restrictions on data type,
+ * readability, or writeability.
+ *
+ * @param name Name of the new dynamic property
+ *
+ * @exception IllegalArgumentException if name is null
+ */
+ public void add(String name) {
+ add(name, null);
+ }
+
+ /**
+ * Add a new dynamic property with the specified data type, but with
+ * no restrictions on readability or writeability.
+ *
+ * @param name Name of the new dynamic property
+ * @param type Data type of the new dynamic property (null for no
+ * restrictions)
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no new properties can be added
+ */
+ public void add(String name, Class type) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("Property name is missing.");
+ }
+
+ if (isRestricted())
+ throw new IllegalStateException("DynaClass is currently restricted. No new properties can be added.");
+
+ Object value = values.get(name);
+
+ // Check if the property already exists
+ if (value == null) {
+ values.put(name, type == null ? null : createProperty(name, type));
+ }
+
+ }
+
+ /**
+ * <p>Add a new dynamic property with the specified data type, readability,
+ * and writeability.</p>
+ *
+ * <p><strong>N.B.</strong>Support for readable/writeable properties has not been implemented
+ * and this method always throws a <code>UnsupportedOperationException</code>.</p>
+ *
+ * <p>I'm not sure the intention of the original authors for this method, but it seems to
+ * me that readable/writable should be attributes of the <code>DynaProperty</code> class
+ * (which they are not) and is the reason this method has not been implemented.</p>
+ *
+ * @param name Name of the new dynamic property
+ * @param type Data type of the new dynamic property (null for no
+ * restrictions)
+ * @param readable Set to <code>true</code> if this property value
+ * should be readable
+ * @param writeable Set to <code>true</code> if this property value
+ * should be writeable
+ *
+ * @exception UnsupportedOperationException anytime this method is called
+ */
+ public void add(String name, Class type, boolean readable, boolean writeable) {
+ throw new java.lang.UnsupportedOperationException("readable/writable properties not supported");
+ }
+
+ /**
+ * Add a new dynamic property.
+ *
+ * @param property Property the new dynamic property to add.
+ *
+ * @exception IllegalArgumentException if name is null
+ */
+ protected void add(DynaProperty property) {
+ add(property.getName(), property.getType());
+ }
+
+ /**
+ * Remove the specified dynamic property, and any associated data type,
+ * readability, and writeability, from this dynamic class.
+ * <strong>NOTE</strong> - This does <strong>NOT</strong> cause any
+ * corresponding property values to be removed from DynaBean instances
+ * associated with this DynaClass.
+ *
+ * @param name Name of the dynamic property to remove
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no properties can be removed
+ */
+ public void remove(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("Property name is missing.");
+ }
+
+ if (isRestricted()) {
+ throw new IllegalStateException("DynaClass is currently restricted. No properties can be removed.");
+ }
+
+ // Remove, if property doesn't exist
+ if (values.containsKey(name)) {
+ values.remove(name);
+ }
+
+ }
+
+
+ // ------------------- Additional Public Methods ----------------------------------
+
+ /**
+ * Should this DynaClass return a <code>null</code> from
+ * the <code>getDynaProperty(name)</code> method if the property
+ * doesn't exist.
+ */
+ public boolean isReturnNull() {
+ return returnNull;
+ }
+
+ /**
+ * Set whether this DynaClass should return a <code>null</code> from
+ * the <code>getDynaProperty(name)</code> method if the property
+ * doesn't exist.
+ */
+ public void setReturnNull(boolean returnNull) {
+ this.returnNull = returnNull;
+ }
+
+
+ // ------------------- Protected Methods ----------------------------------
+
+ /**
+ * <p>Indicate whether a property actually exists.</p>
+ *
+ * <p><strong>N.B.</strong> Using <code>getDynaProperty(name) == null</code>
+ * doesn't work in this implementation because that method might
+ * return a DynaProperty if it doesn't exist (depending on the
+ * <code>returnNull</code> indicator).</p>
+ *
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ protected boolean isDynaProperty(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException("Property name is missing.");
+ }
+
+ return values.containsKey(name);
+
+ }
+
+}
\ No newline at end of file
diff --git a/trunk/src/java/org/apache/commons/beanutils/MappedPropertyDescriptor.java b/trunk/src/java/org/apache/commons/beanutils/MappedPropertyDescriptor.java
new file mode 100644
index 0000000..a33ed74
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/MappedPropertyDescriptor.java
@@ -0,0 +1,562 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.beans.IntrospectionException;
+import java.beans.PropertyDescriptor;
+import java.lang.reflect.Method;
+import java.lang.reflect.Modifier;
+import java.security.AccessController;
+import java.security.PrivilegedAction;
+
+
+/**
+ * A MappedPropertyDescriptor describes one mapped property.
+ * Mapped properties are multivalued properties like indexed properties
+ * but that are accessed with a String key instead of an index.
+ * Such property values are typically stored in a Map collection.
+ * For this class to work properly, a mapped value must have
+ * getter and setter methods of the form
+ * <p><code>get<strong>Property</strong>(String key)<code> and
+ * <p><code>set<Property>(String key, Object value)<code>,
+ * <p>where <code><strong>Property</strong></code> must be replaced
+ * by the name of the property.
+ * @see java.beans.PropertyDescriptor
+ *
+ * @author Rey Fran�ois
+ * @author Gregor Ra�man
+ * @version $Revision: 1.18.2.1 $ $Date: 2004/07/27 21:44:26 $
+ */
+
+
+public class MappedPropertyDescriptor extends PropertyDescriptor {
+ // ----------------------------------------------------- Instance Variables
+
+ /**
+ * The underlying data type of the property we are describing.
+ */
+ private Class mappedPropertyType;
+
+ /**
+ * The reader method for this property (if any).
+ */
+ private Method mappedReadMethod;
+
+ /**
+ * The writer method for this property (if any).
+ */
+ private Method mappedWriteMethod;
+
+ /**
+ * The parameter types array for the reader method signature.
+ */
+ private static final Class[] stringClassArray = new Class[]{String.class};
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Constructs a MappedPropertyDescriptor for a property that follows
+ * the standard Java convention by having getFoo and setFoo
+ * accessor methods, with the addition of a String parameter (the key).
+ * Thus if the argument name is "fred", it will
+ * assume that the writer method is "setFred" and the reader method
+ * is "getFred". Note that the property name should start with a lower
+ * case character, which will be capitalized in the method names.
+ *
+ * @param propertyName The programmatic name of the property.
+ * @param beanClass The Class object for the target bean. For
+ * example sun.beans.OurButton.class.
+ *
+ * @exception IntrospectionException if an exception occurs during
+ * introspection.
+ */
+ public MappedPropertyDescriptor(String propertyName, Class beanClass)
+ throws IntrospectionException {
+
+ super(propertyName, null, null);
+
+ if (propertyName == null || propertyName.length() == 0) {
+ throw new IntrospectionException("bad property name: " +
+ propertyName + " on class: " + beanClass.getClass().getName());
+ }
+
+ setName(propertyName);
+ String base = capitalizePropertyName(propertyName);
+
+ // Look for mapped read method and matching write method
+ try {
+ mappedReadMethod = findMethod(beanClass, "get" + base, 1,
+ stringClassArray);
+ Class params[] = { String.class, mappedReadMethod.getReturnType() };
+ mappedWriteMethod = findMethod(beanClass, "set" + base, 2, params);
+ } catch (IntrospectionException e) {
+ ;
+ }
+
+ // If there's no read method, then look for just a write method
+ if (mappedReadMethod == null) {
+ mappedWriteMethod = findMethod(beanClass, "set" + base, 2);
+ }
+
+ if ((mappedReadMethod == null) && (mappedWriteMethod == null)) {
+ throw new IntrospectionException("Property '" + propertyName +
+ "' not found on " +
+ beanClass.getName());
+ }
+
+ findMappedPropertyType();
+ }
+
+
+ /**
+ * This constructor takes the name of a mapped property, and method
+ * names for reading and writing the property.
+ *
+ * @param propertyName The programmatic name of the property.
+ * @param beanClass The Class object for the target bean. For
+ * example sun.beans.OurButton.class.
+ * @param mappedGetterName The name of the method used for
+ * reading one of the property values. May be null if the
+ * property is write-only.
+ * @param mappedSetterName The name of the method used for writing
+ * one of the property values. May be null if the property is
+ * read-only.
+ *
+ * @exception IntrospectionException if an exception occurs during
+ * introspection.
+ */
+ public MappedPropertyDescriptor(String propertyName, Class beanClass,
+ String mappedGetterName, String mappedSetterName)
+ throws IntrospectionException {
+
+ super(propertyName, null, null);
+
+ if (propertyName == null || propertyName.length() == 0) {
+ throw new IntrospectionException("bad property name: " +
+ propertyName);
+ }
+ setName(propertyName);
+
+ // search the mapped get and set methods
+ mappedReadMethod =
+ findMethod(beanClass, mappedGetterName, 1, stringClassArray);
+
+ if (mappedReadMethod != null) {
+ Class params[] = { String.class, mappedReadMethod.getReturnType() };
+ mappedWriteMethod =
+ findMethod(beanClass, mappedSetterName, 2, params);
+ } else {
+ mappedWriteMethod =
+ findMethod(beanClass, mappedSetterName, 2);
+ }
+
+ findMappedPropertyType();
+ }
+
+ /**
+ * This constructor takes the name of a mapped property, and Method
+ * objects for reading and writing the property.
+ *
+ * @param propertyName The programmatic name of the property.
+ * @param mappedGetter The method used for reading one of
+ * the property values. May be be null if the property
+ * is write-only.
+ * @param mappedSetter The method used for writing one the
+ * property values. May be null if the property is read-only.
+ *
+ * @exception IntrospectionException if an exception occurs during
+ * introspection.
+ */
+ public MappedPropertyDescriptor(String propertyName,
+ Method mappedGetter, Method mappedSetter)
+ throws IntrospectionException {
+
+ super(propertyName, mappedGetter, mappedSetter);
+
+ if (propertyName == null || propertyName.length() == 0) {
+ throw new IntrospectionException("bad property name: " +
+ propertyName);
+ }
+
+ setName(propertyName);
+ mappedReadMethod = mappedGetter;
+ mappedWriteMethod = mappedSetter;
+ findMappedPropertyType();
+ }
+
+ // -------------------------------------------------------- Public Methods
+
+ /**
+ * Gets the Class object for the property values.
+ *
+ * @return The Java type info for the property values. Note that
+ * the "Class" object may describe a built-in Java type such as "int".
+ * The result may be "null" if this is a mapped property that
+ * does not support non-keyed access.
+ * <p>
+ * This is the type that will be returned by the mappedReadMethod.
+ */
+ public Class getMappedPropertyType() {
+ return mappedPropertyType;
+ }
+
+ /**
+ * Gets the method that should be used to read one of the property value.
+ *
+ * @return The method that should be used to read the property value.
+ * May return null if the property can't be read.
+ */
+ public Method getMappedReadMethod() {
+ return mappedReadMethod;
+ }
+
+ /**
+ * Sets the method that should be used to read one of the property value.
+ *
+ * @param mappedGetter The new getter method.
+ */
+ public void setMappedReadMethod(Method mappedGetter)
+ throws IntrospectionException {
+ mappedReadMethod = mappedGetter;
+ findMappedPropertyType();
+ }
+
+ /**
+ * Gets the method that should be used to write one of the property value.
+ *
+ * @return The method that should be used to write one of the property value.
+ * May return null if the property can't be written.
+ */
+ public Method getMappedWriteMethod() {
+ return mappedWriteMethod;
+ }
+
+ /**
+ * Sets the method that should be used to write the property value.
+ *
+ * @param mappedSetter The new setter method.
+ */
+ public void setMappedWriteMethod(Method mappedSetter)
+ throws IntrospectionException {
+ mappedWriteMethod = mappedSetter;
+ findMappedPropertyType();
+ }
+
+ // ------------------------------------------------------- Private Methods
+
+ /**
+ * Introspect our bean class to identify the corresponding getter
+ * and setter methods.
+ */
+ private void findMappedPropertyType() throws IntrospectionException {
+ try {
+ mappedPropertyType = null;
+ if (mappedReadMethod != null) {
+ if (mappedReadMethod.getParameterTypes().length != 1) {
+ throw new IntrospectionException
+ ("bad mapped read method arg count");
+ }
+ mappedPropertyType = mappedReadMethod.getReturnType();
+ if (mappedPropertyType == Void.TYPE) {
+ throw new IntrospectionException
+ ("mapped read method " +
+ mappedReadMethod.getName() + " returns void");
+ }
+ }
+
+ if (mappedWriteMethod != null) {
+ Class params[] = mappedWriteMethod.getParameterTypes();
+ if (params.length != 2) {
+ throw new IntrospectionException
+ ("bad mapped write method arg count");
+ }
+ if (mappedPropertyType != null &&
+ mappedPropertyType != params[1]) {
+ throw new IntrospectionException
+ ("type mismatch between mapped read and write methods");
+ }
+ mappedPropertyType = params[1];
+ }
+ } catch (IntrospectionException ex) {
+ throw ex;
+ }
+ }
+
+
+ /**
+ * Return a capitalized version of the specified property name.
+ *
+ * @param s The property name
+ */
+ private static String capitalizePropertyName(String s) {
+ if (s.length() == 0) {
+ return s;
+ }
+
+ char chars[] = s.toCharArray();
+ chars[0] = Character.toUpperCase(chars[0]);
+ return new String(chars);
+ }
+
+ //======================================================================
+ // Package private support methods (copied from java.beans.Introspector).
+ //======================================================================
+
+ // Cache of Class.getDeclaredMethods:
+ private static java.util.Hashtable
+ declaredMethodCache = new java.util.Hashtable();
+
+ /*
+ * Internal method to return *public* methods within a class.
+ */
+ private static synchronized Method[] getPublicDeclaredMethods(Class clz) {
+ // Looking up Class.getDeclaredMethods is relatively expensive,
+ // so we cache the results.
+ final Class fclz = clz;
+ Method[] result = (Method[]) declaredMethodCache.get(fclz);
+ if (result != null) {
+ return result;
+ }
+
+ // We have to raise privilege for getDeclaredMethods
+ result = (Method[])
+ AccessController.doPrivileged(new PrivilegedAction() {
+ public Object run() {
+ try{
+
+ return fclz.getDeclaredMethods();
+
+ } catch (SecurityException ex) {
+ // this means we're in a limited security environment
+ // so let's try going through the public methods
+ // and null those those that are not from the declaring
+ // class
+ Method[] methods = fclz.getMethods();
+ for (int i = 0, size = methods.length; i < size; i++) {
+ Method method = methods[i];
+ if (!(fclz.equals(method.getDeclaringClass()))) {
+ methods[i] = null;
+ }
+ }
+ return methods;
+ }
+ }
+ });
+
+ // Null out any non-public methods.
+ for (int i = 0; i < result.length; i++) {
+ Method method = result[i];
+ if (method != null) {
+ int mods = method.getModifiers();
+ if (!Modifier.isPublic(mods)) {
+ result[i] = null;
+ }
+ }
+ }
+
+ // Add it to the cache.
+ declaredMethodCache.put(clz, result);
+ return result;
+ }
+
+ /**
+ * Internal support for finding a target methodName on a given class.
+ */
+ private static Method internalFindMethod(Class start, String methodName,
+ int argCount) {
+ // For overridden methods we need to find the most derived version.
+ // So we start with the given class and walk up the superclass chain.
+ for (Class cl = start; cl != null; cl = cl.getSuperclass()) {
+ Method methods[] = getPublicDeclaredMethods(cl);
+ for (int i = 0; i < methods.length; i++) {
+ Method method = methods[i];
+ if (method == null) {
+ continue;
+ }
+ // skip static methods.
+ int mods = method.getModifiers();
+ if (Modifier.isStatic(mods)) {
+ continue;
+ }
+ if (method.getName().equals(methodName) &&
+ method.getParameterTypes().length == argCount) {
+ return method;
+ }
+ }
+ }
+
+ // Now check any inherited interfaces. This is necessary both when
+ // the argument class is itself an interface, and when the argument
+ // class is an abstract class.
+ Class ifcs[] = start.getInterfaces();
+ for (int i = 0; i < ifcs.length; i++) {
+ Method m = internalFindMethod(ifcs[i], methodName, argCount);
+ if (m != null) {
+ return m;
+ }
+ }
+
+ return null;
+ }
+
+ /**
+ * Internal support for finding a target methodName with a given
+ * parameter list on a given class.
+ */
+ private static Method internalFindMethod(Class start, String methodName,
+ int argCount, Class args[]) {
+ // For overriden methods we need to find the most derived version.
+ // So we start with the given class and walk up the superclass chain.
+ for (Class cl = start; cl != null; cl = cl.getSuperclass()) {
+ Method methods[] = getPublicDeclaredMethods(cl);
+ for (int i = 0; i < methods.length; i++) {
+ Method method = methods[i];
+ if (method == null) {
+ continue;
+ }
+ // skip static methods.
+ int mods = method.getModifiers();
+ if (Modifier.isStatic(mods)) {
+ continue;
+ }
+ // make sure method signature matches.
+ Class params[] = method.getParameterTypes();
+ if (method.getName().equals(methodName) &&
+ params.length == argCount) {
+ boolean different = false;
+ if (argCount > 0) {
+ for (int j = 0; j < argCount; j++) {
+ if (params[j] != args[j]) {
+ different = true;
+ continue;
+ }
+ }
+ if (different) {
+ continue;
+ }
+ }
+ return method;
+ }
+ }
+ }
+
+ // Now check any inherited interfaces. This is necessary both when
+ // the argument class is itself an interface, and when the argument
+ // class is an abstract class.
+ Class ifcs[] = start.getInterfaces();
+ for (int i = 0; i < ifcs.length; i++) {
+ Method m = internalFindMethod(ifcs[i], methodName, argCount);
+ if (m != null) {
+ return m;
+ }
+ }
+
+ return null;
+ }
+
+ /**
+ * Find a target methodName on a given class.
+ */
+ static Method findMethod(Class cls, String methodName, int argCount)
+ throws IntrospectionException {
+ if (methodName == null) {
+ return null;
+ }
+
+ Method m = internalFindMethod(cls, methodName, argCount);
+ if (m != null) {
+ return m;
+ }
+
+ // We failed to find a suitable method
+ throw new IntrospectionException("No method \"" + methodName +
+ "\" with " + argCount + " arg(s)");
+ }
+
+ /**
+ * Find a target methodName with specific parameter list on a given class.
+ */
+ static Method findMethod(Class cls, String methodName, int argCount,
+ Class args[]) throws IntrospectionException {
+ if (methodName == null) {
+ return null;
+ }
+
+ Method m = internalFindMethod(cls, methodName, argCount, args);
+ if (m != null) {
+ return m;
+ }
+
+ // We failed to find a suitable method
+ throw new IntrospectionException("No method \"" + methodName +
+ "\" with " + argCount + " arg(s) of matching types.");
+ }
+
+ /**
+ * Return true if class a is either equivalent to class b, or
+ * if class a is a subclass of class b, ie if a either "extends"
+ * or "implements" b.
+ * Note tht either or both "Class" objects may represent interfaces.
+ */
+ static boolean isSubclass(Class a, Class b) {
+ // We rely on the fact that for any given java class or
+ // primtitive type there is a unqiue Class object, so
+ // we can use object equivalence in the comparisons.
+ if (a == b) {
+ return true;
+ }
+
+ if (a == null || b == null) {
+ return false;
+ }
+
+ for (Class x = a; x != null; x = x.getSuperclass()) {
+ if (x == b) {
+ return true;
+ }
+
+ if (b.isInterface()) {
+ Class interfaces[] = x.getInterfaces();
+ for (int i = 0; i < interfaces.length; i++) {
+ if (isSubclass(interfaces[i], b)) {
+ return true;
+ }
+ }
+ }
+ }
+
+ return false;
+ }
+
+
+ /**
+ * Return true iff the given method throws the given exception.
+ */
+
+ private boolean throwsException(Method method, Class exception) {
+
+ Class exs[] = method.getExceptionTypes();
+ for (int i = 0; i < exs.length; i++) {
+ if (exs[i] == exception) {
+ return true;
+ }
+ }
+
+ return false;
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/MethodUtils.java b/trunk/src/java/org/apache/commons/beanutils/MethodUtils.java
new file mode 100644
index 0000000..df3a1d0
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/MethodUtils.java
@@ -0,0 +1,851 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Method;
+import java.lang.reflect.Modifier;
+
+import java.util.WeakHashMap;
+
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+
+
+/**
+ * <p> Utility reflection methods focussed on methods in general rather than properties in particular. </p>
+ *
+ * <h3>Known Limitations</h3>
+ * <h4>Accessing Public Methods In A Default Access Superclass</h4>
+ * <p>There is an issue when invoking public methods contained in a default access superclass.
+ * Reflection locates these methods fine and correctly assigns them as public.
+ * However, an <code>IllegalAccessException</code> is thrown if the method is invoked.</p>
+ *
+ * <p><code>MethodUtils</code> contains a workaround for this situation.
+ * It will attempt to call <code>setAccessible</code> on this method.
+ * If this call succeeds, then the method can be invoked as normal.
+ * This call will only succeed when the application has sufficient security privilages.
+ * If this call fails then a warning will be logged and the method may fail.</p>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @author Rey Fran�ois
+ * @author Gregor Ra�man
+ * @author Jan Sorensen
+ * @author Robert Burrell Donkin
+ */
+
+public class MethodUtils {
+
+ // --------------------------------------------------------- Private Methods
+
+ /**
+ * All logging goes through this logger
+ */
+ private static Log log = LogFactory.getLog(MethodUtils.class);
+ /** Only log warning about accessibility work around once */
+ private static boolean loggedAccessibleWarning = false;
+
+ /** An empty class array */
+ private static final Class[] emptyClassArray = new Class[0];
+ /** An empty object array */
+ private static final Object[] emptyObjectArray = new Object[0];
+
+ /**
+ * Stores a cache of Methods against MethodDescriptors, in a WeakHashMap.
+ */
+ private static WeakHashMap cache = new WeakHashMap();
+
+ // --------------------------------------------------------- Public Methods
+
+ /**
+ * <p>Invoke a named method whose parameter type matches the object type.</p>
+ *
+ * <p>The behaviour of this method is less deterministic
+ * than {@link #invokeExactMethod}.
+ * It loops through all methods with names that match
+ * and then executes the first it finds with compatable parameters.</p>
+ *
+ * <p>This method supports calls to methods taking primitive parameters
+ * via passing in wrapping classes. So, for example, a <code>Boolean</code> class
+ * would match a <code>boolean</code> primitive.</p>
+ *
+ * <p> This is a convenient wrapper for
+ * {@link #invokeMethod(Object object,String methodName,Object [] args)}.
+ * </p>
+ *
+ * @param object invoke method on this object
+ * @param methodName get method with this name
+ * @param arg use this argument
+ *
+ * @throws NoSuchMethodException if there is no such accessible method
+ * @throws InvocationTargetException wraps an exception thrown by the
+ * method invoked
+ * @throws IllegalAccessException if the requested method is not accessible
+ * via reflection
+ */
+ public static Object invokeMethod(
+ Object object,
+ String methodName,
+ Object arg)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException {
+
+ Object[] args = {arg};
+ return invokeMethod(object, methodName, args);
+
+ }
+
+
+ /**
+ * <p>Invoke a named method whose parameter type matches the object type.</p>
+ *
+ * <p>The behaviour of this method is less deterministic
+ * than {@link #invokeExactMethod(Object object,String methodName,Object [] args)}.
+ * It loops through all methods with names that match
+ * and then executes the first it finds with compatable parameters.</p>
+ *
+ * <p>This method supports calls to methods taking primitive parameters
+ * via passing in wrapping classes. So, for example, a <code>Boolean</code> class
+ * would match a <code>boolean</code> primitive.</p>
+ *
+ * <p> This is a convenient wrapper for
+ * {@link #invokeMethod(Object object,String methodName,Object [] args,Class[] parameterTypes)}.
+ * </p>
+ *
+ * @param object invoke method on this object
+ * @param methodName get method with this name
+ * @param args use these arguments - treat null as empty array
+ *
+ * @throws NoSuchMethodException if there is no such accessible method
+ * @throws InvocationTargetException wraps an exception thrown by the
+ * method invoked
+ * @throws IllegalAccessException if the requested method is not accessible
+ * via reflection
+ */
+ public static Object invokeMethod(
+ Object object,
+ String methodName,
+ Object[] args)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException {
+
+ if (args == null) {
+ args = emptyObjectArray;
+ }
+ int arguments = args.length;
+ Class parameterTypes [] = new Class[arguments];
+ for (int i = 0; i < arguments; i++) {
+ parameterTypes[i] = args[i].getClass();
+ }
+ return invokeMethod(object, methodName, args, parameterTypes);
+
+ }
+
+
+ /**
+ * <p>Invoke a named method whose parameter type matches the object type.</p>
+ *
+ * <p>The behaviour of this method is less deterministic
+ * than {@link
+ * #invokeExactMethod(Object object,String methodName,Object [] args,Class[] parameterTypes)}.
+ * It loops through all methods with names that match
+ * and then executes the first it finds with compatable parameters.</p>
+ *
+ * <p>This method supports calls to methods taking primitive parameters
+ * via passing in wrapping classes. So, for example, a <code>Boolean</code> class
+ * would match a <code>boolean</code> primitive.</p>
+ *
+ *
+ * @param object invoke method on this object
+ * @param methodName get method with this name
+ * @param args use these arguments - treat null as empty array
+ * @param parameterTypes match these parameters - treat null as empty array
+ *
+ * @throws NoSuchMethodException if there is no such accessible method
+ * @throws InvocationTargetException wraps an exception thrown by the
+ * method invoked
+ * @throws IllegalAccessException if the requested method is not accessible
+ * via reflection
+ */
+ public static Object invokeMethod(
+ Object object,
+ String methodName,
+ Object[] args,
+ Class[] parameterTypes)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException {
+
+ if (parameterTypes == null) {
+ parameterTypes = emptyClassArray;
+ }
+ if (args == null) {
+ args = emptyObjectArray;
+ }
+
+ Method method = getMatchingAccessibleMethod(
+ object.getClass(),
+ methodName,
+ parameterTypes);
+ if (method == null)
+ throw new NoSuchMethodException("No such accessible method: " +
+ methodName + "() on object: " + object.getClass().getName());
+ return method.invoke(object, args);
+ }
+
+
+ /**
+ * <p>Invoke a method whose parameter type matches exactly the object
+ * type.</p>
+ *
+ * <p> This is a convenient wrapper for
+ * {@link #invokeExactMethod(Object object,String methodName,Object [] args)}.
+ * </p>
+ *
+ * @param object invoke method on this object
+ * @param methodName get method with this name
+ * @param arg use this argument
+ *
+ * @throws NoSuchMethodException if there is no such accessible method
+ * @throws InvocationTargetException wraps an exception thrown by the
+ * method invoked
+ * @throws IllegalAccessException if the requested method is not accessible
+ * via reflection
+ */
+ public static Object invokeExactMethod(
+ Object object,
+ String methodName,
+ Object arg)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException {
+
+ Object[] args = {arg};
+ return invokeExactMethod(object, methodName, args);
+
+ }
+
+
+ /**
+ * <p>Invoke a method whose parameter types match exactly the object
+ * types.</p>
+ *
+ * <p> This uses reflection to invoke the method obtained from a call to
+ * {@link #getAccessibleMethod}.</p>
+ *
+ * @param object invoke method on this object
+ * @param methodName get method with this name
+ * @param args use these arguments - treat null as empty array
+ *
+ * @throws NoSuchMethodException if there is no such accessible method
+ * @throws InvocationTargetException wraps an exception thrown by the
+ * method invoked
+ * @throws IllegalAccessException if the requested method is not accessible
+ * via reflection
+ */
+ public static Object invokeExactMethod(
+ Object object,
+ String methodName,
+ Object[] args)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException {
+ if (args == null) {
+ args = emptyObjectArray;
+ }
+ int arguments = args.length;
+ Class parameterTypes [] = new Class[arguments];
+ for (int i = 0; i < arguments; i++) {
+ parameterTypes[i] = args[i].getClass();
+ }
+ return invokeExactMethod(object, methodName, args, parameterTypes);
+
+ }
+
+
+ /**
+ * <p>Invoke a method whose parameter types match exactly the parameter
+ * types given.</p>
+ *
+ * <p>This uses reflection to invoke the method obtained from a call to
+ * {@link #getAccessibleMethod}.</p>
+ *
+ * @param object invoke method on this object
+ * @param methodName get method with this name
+ * @param args use these arguments - treat null as empty array
+ * @param parameterTypes match these parameters - treat null as empty array
+ *
+ * @throws NoSuchMethodException if there is no such accessible method
+ * @throws InvocationTargetException wraps an exception thrown by the
+ * method invoked
+ * @throws IllegalAccessException if the requested method is not accessible
+ * via reflection
+ */
+ public static Object invokeExactMethod(
+ Object object,
+ String methodName,
+ Object[] args,
+ Class[] parameterTypes)
+ throws
+ NoSuchMethodException,
+ IllegalAccessException,
+ InvocationTargetException {
+
+ if (args == null) {
+ args = emptyObjectArray;
+ }
+
+ if (parameterTypes == null) {
+ parameterTypes = emptyClassArray;
+ }
+
+ Method method = getAccessibleMethod(
+ object.getClass(),
+ methodName,
+ parameterTypes);
+ if (method == null)
+ throw new NoSuchMethodException("No such accessible method: " +
+ methodName + "() on object: " + object.getClass().getName());
+ return method.invoke(object, args);
+
+ }
+
+
+ /**
+ * <p>Return an accessible method (that is, one that can be invoked via
+ * reflection) with given name and a single parameter. If no such method
+ * can be found, return <code>null</code>.
+ * Basically, a convenience wrapper that constructs a <code>Class</code>
+ * array for you.</p>
+ *
+ * @param clazz get method from this class
+ * @param methodName get method with this name
+ * @param parameterType taking this type of parameter
+ */
+ public static Method getAccessibleMethod(
+ Class clazz,
+ String methodName,
+ Class parameterType) {
+
+ Class[] parameterTypes = {parameterType};
+ return getAccessibleMethod(clazz, methodName, parameterTypes);
+
+ }
+
+
+ /**
+ * <p>Return an accessible method (that is, one that can be invoked via
+ * reflection) with given name and parameters. If no such method
+ * can be found, return <code>null</code>.
+ * This is just a convenient wrapper for
+ * {@link #getAccessibleMethod(Method method)}.</p>
+ *
+ * @param clazz get method from this class
+ * @param methodName get method with this name
+ * @param parameterTypes with these parameters types
+ */
+ public static Method getAccessibleMethod(
+ Class clazz,
+ String methodName,
+ Class[] parameterTypes) {
+
+ try {
+ MethodDescriptor md = new MethodDescriptor(clazz, methodName, parameterTypes, true);
+ // Check the cache first
+ Method method = (Method)cache.get(md);
+ if (method != null) {
+ return method;
+ }
+
+ method = getAccessibleMethod
+ (clazz.getMethod(methodName, parameterTypes));
+ cache.put(md, method);
+ return method;
+ } catch (NoSuchMethodException e) {
+ return (null);
+ }
+
+ }
+
+
+ /**
+ * <p>Return an accessible method (that is, one that can be invoked via
+ * reflection) that implements the specified Method. If no such method
+ * can be found, return <code>null</code>.</p>
+ *
+ * @param method The method that we wish to call
+ */
+ public static Method getAccessibleMethod(Method method) {
+
+ // Make sure we have a method to check
+ if (method == null) {
+ return (null);
+ }
+
+ // If the requested method is not public we cannot call it
+ if (!Modifier.isPublic(method.getModifiers())) {
+ return (null);
+ }
+
+ // If the declaring class is public, we are done
+ Class clazz = method.getDeclaringClass();
+ if (Modifier.isPublic(clazz.getModifiers())) {
+ return (method);
+ }
+
+ // Check the implemented interfaces and subinterfaces
+ method =
+ getAccessibleMethodFromInterfaceNest(clazz,
+ method.getName(),
+ method.getParameterTypes());
+ return (method);
+
+ }
+
+
+ // -------------------------------------------------------- Private Methods
+
+ /**
+ * <p>Return an accessible method (that is, one that can be invoked via
+ * reflection) that implements the specified method, by scanning through
+ * all implemented interfaces and subinterfaces. If no such method
+ * can be found, return <code>null</code>.</p>
+ *
+ * <p> There isn't any good reason why this method must be private.
+ * It is because there doesn't seem any reason why other classes should
+ * call this rather than the higher level methods.</p>
+ *
+ * @param clazz Parent class for the interfaces to be checked
+ * @param methodName Method name of the method we wish to call
+ * @param parameterTypes The parameter type signatures
+ */
+ private static Method getAccessibleMethodFromInterfaceNest
+ (Class clazz, String methodName, Class parameterTypes[]) {
+
+ Method method = null;
+
+ // Search up the superclass chain
+ for (; clazz != null; clazz = clazz.getSuperclass()) {
+
+ // Check the implemented interfaces of the parent class
+ Class interfaces[] = clazz.getInterfaces();
+ for (int i = 0; i < interfaces.length; i++) {
+
+ // Is this interface public?
+ if (!Modifier.isPublic(interfaces[i].getModifiers()))
+ continue;
+
+ // Does the method exist on this interface?
+ try {
+ method = interfaces[i].getDeclaredMethod(methodName,
+ parameterTypes);
+ } catch (NoSuchMethodException e) {
+ ;
+ }
+ if (method != null)
+ break;
+
+ // Recursively check our parent interfaces
+ method =
+ getAccessibleMethodFromInterfaceNest(interfaces[i],
+ methodName,
+ parameterTypes);
+ if (method != null)
+ break;
+
+ }
+
+ }
+
+ // If we found a method return it
+ if (method != null)
+ return (method);
+
+ // We did not find anything
+ return (null);
+
+ }
+
+ /**
+ * <p>Find an accessible method that matches the given name and has compatible parameters.
+ * Compatible parameters mean that every method parameter is assignable from
+ * the given parameters.
+ * In other words, it finds a method with the given name
+ * that will take the parameters given.<p>
+ *
+ * <p>This method is slightly undeterminstic since it loops
+ * through methods names and return the first matching method.</p>
+ *
+ * <p>This method is used by
+ * {@link
+ * #invokeMethod(Object object,String methodName,Object [] args,Class[] parameterTypes)}.
+ *
+ * <p>This method can match primitive parameter by passing in wrapper classes.
+ * For example, a <code>Boolean</code> will match a primitive <code>boolean</code>
+ * parameter.
+ *
+ * @param clazz find method in this class
+ * @param methodName find method with this name
+ * @param parameterTypes find method with compatible parameters
+ */
+ public static Method getMatchingAccessibleMethod(
+ Class clazz,
+ String methodName,
+ Class[] parameterTypes) {
+ // trace logging
+ if (log.isTraceEnabled()) {
+ log.trace("Matching name=" + methodName + " on " + clazz);
+ }
+ MethodDescriptor md = new MethodDescriptor(clazz, methodName, parameterTypes, false);
+
+ // see if we can find the method directly
+ // most of the time this works and it's much faster
+ try {
+ // Check the cache first
+ Method method = (Method)cache.get(md);
+ if (method != null) {
+ return method;
+ }
+
+ method = clazz.getMethod(methodName, parameterTypes);
+ if (log.isTraceEnabled()) {
+ log.trace("Found straight match: " + method);
+ log.trace("isPublic:" + Modifier.isPublic(method.getModifiers()));
+ }
+
+ try {
+ //
+ // XXX Default access superclass workaround
+ //
+ // When a public class has a default access superclass
+ // with public methods, these methods are accessible.
+ // Calling them from compiled code works fine.
+ //
+ // Unfortunately, using reflection to invoke these methods
+ // seems to (wrongly) to prevent access even when the method
+ // modifer is public.
+ //
+ // The following workaround solves the problem but will only
+ // work from sufficiently privilages code.
+ //
+ // Better workarounds would be greatfully accepted.
+ //
+ method.setAccessible(true);
+
+ } catch (SecurityException se) {
+ // log but continue just in case the method.invoke works anyway
+ if (!loggedAccessibleWarning) {
+ boolean vunerableJVM = false;
+ try {
+ String specVersion = System.getProperty("java.specification.version");
+ if (specVersion.charAt(0) == '1' &&
+ (specVersion.charAt(0) == '0' ||
+ specVersion.charAt(0) == '1' ||
+ specVersion.charAt(0) == '2' ||
+ specVersion.charAt(0) == '3')) {
+
+ vunerableJVM = true;
+ }
+ } catch (SecurityException e) {
+ // don't know - so display warning
+ vunerableJVM = true;
+ }
+ if (vunerableJVM) {
+ log.warn(
+ "Current Security Manager restricts use of workarounds for reflection bugs "
+ + " in pre-1.4 JVMs.");
+ }
+ loggedAccessibleWarning = true;
+ }
+ log.debug(
+ "Cannot setAccessible on method. Therefore cannot use jvm access bug workaround.",
+ se);
+ }
+ cache.put(md, method);
+ return method;
+
+ } catch (NoSuchMethodException e) { /* SWALLOW */ }
+
+ // search through all methods
+ int paramSize = parameterTypes.length;
+ Method[] methods = clazz.getMethods();
+ for (int i = 0, size = methods.length; i < size ; i++) {
+ if (methods[i].getName().equals(methodName)) {
+ // log some trace information
+ if (log.isTraceEnabled()) {
+ log.trace("Found matching name:");
+ log.trace(methods[i]);
+ }
+
+ // compare parameters
+ Class[] methodsParams = methods[i].getParameterTypes();
+ int methodParamSize = methodsParams.length;
+ if (methodParamSize == paramSize) {
+ boolean match = true;
+ for (int n = 0 ; n < methodParamSize; n++) {
+ if (log.isTraceEnabled()) {
+ log.trace("Param=" + parameterTypes[n].getName());
+ log.trace("Method=" + methodsParams[n].getName());
+ }
+ if (!isAssignmentCompatible(methodsParams[n], parameterTypes[n])) {
+ if (log.isTraceEnabled()) {
+ log.trace(methodsParams[n] + " is not assignable from "
+ + parameterTypes[n]);
+ }
+ match = false;
+ break;
+ }
+ }
+
+ if (match) {
+ // get accessible version of method
+ Method method = getAccessibleMethod(methods[i]);
+ if (method != null) {
+ if (log.isTraceEnabled()) {
+ log.trace(method + " accessible version of "
+ + methods[i]);
+ }
+ try {
+ //
+ // XXX Default access superclass workaround
+ // (See above for more details.)
+ //
+ method.setAccessible(true);
+
+ } catch (SecurityException se) {
+ // log but continue just in case the method.invoke works anyway
+ if (!loggedAccessibleWarning) {
+ log.warn(
+ "Cannot use JVM pre-1.4 access bug workaround due to restrictive security manager.");
+ loggedAccessibleWarning = true;
+ }
+ log.debug(
+ "Cannot setAccessible on method. Therefore cannot use jvm access bug workaround.",
+ se);
+ }
+ cache.put(md, method);
+ return method;
+ }
+
+ log.trace("Couldn't find accessible method.");
+ }
+ }
+ }
+ }
+
+ // didn't find a match
+ log.trace("No match found.");
+ return null;
+ }
+
+ /**
+ * <p>Determine whether a type can be used as a parameter in a method invocation.
+ * This method handles primitive conversions correctly.</p>
+ *
+ * <p>In order words, it will match a <code>Boolean</code> to a <code>boolean</code>,
+ * a <code>Long</code> to a <code>long</code>,
+ * a <code>Float</code> to a <code>float</code>,
+ * a <code>Integer</code> to a <code>int</code>,
+ * and a <code>Double</code> to a <code>double</code>.
+ * Now logic widening matches are allowed.
+ * For example, a <code>Long</code> will not match a <code>int</code>.
+ *
+ * @param parameterType the type of parameter accepted by the method
+ * @param parameterization the type of parameter being tested
+ *
+ * @return true if the assignement is compatible.
+ */
+ public static final boolean isAssignmentCompatible(Class parameterType, Class parameterization) {
+ // try plain assignment
+ if (parameterType.isAssignableFrom(parameterization)) {
+ return true;
+ }
+
+ if (parameterType.isPrimitive()) {
+ // this method does *not* do widening - you must specify exactly
+ // is this the right behaviour?
+ Class parameterWrapperClazz = getPrimitiveWrapper(parameterType);
+ if (parameterWrapperClazz != null) {
+ return parameterWrapperClazz.equals(parameterization);
+ }
+ }
+
+ return false;
+ }
+
+ /**
+ * Gets the wrapper object class for the given primitive type class.
+ * For example, passing <code>boolean.class</code> returns <code>Boolean.class</code>
+ * @param primitiveType the primitive type class for which a match is to be found
+ * @return the wrapper type associated with the given primitive
+ * or null if no match is found
+ */
+ public static Class getPrimitiveWrapper(Class primitiveType) {
+ // does anyone know a better strategy than comparing names?
+ if (boolean.class.equals(primitiveType)) {
+ return Boolean.class;
+ } else if (float.class.equals(primitiveType)) {
+ return Float.class;
+ } else if (long.class.equals(primitiveType)) {
+ return Long.class;
+ } else if (int.class.equals(primitiveType)) {
+ return Integer.class;
+ } else if (short.class.equals(primitiveType)) {
+ return Short.class;
+ } else if (byte.class.equals(primitiveType)) {
+ return Byte.class;
+ } else if (double.class.equals(primitiveType)) {
+ return Double.class;
+ } else if (char.class.equals(primitiveType)) {
+ return Character.class;
+ } else {
+
+ return null;
+ }
+ }
+
+ /**
+ * Gets the class for the primitive type corresponding to the primitive wrapper class given.
+ * For example, an instance of <code>Boolean.class</code> returns a <code>boolean.class</code>.
+ * @param wrapperType the
+ * @return the primitive type class corresponding to the given wrapper class,
+ * null if no match is found
+ */
+ public static Class getPrimitiveType(Class wrapperType) {
+ // does anyone know a better strategy than comparing names?
+ if (Boolean.class.equals(wrapperType)) {
+ return boolean.class;
+ } else if (Float.class.equals(wrapperType)) {
+ return float.class;
+ } else if (Long.class.equals(wrapperType)) {
+ return long.class;
+ } else if (Integer.class.equals(wrapperType)) {
+ return int.class;
+ } else if (Short.class.equals(wrapperType)) {
+ return short.class;
+ } else if (Byte.class.equals(wrapperType)) {
+ return byte.class;
+ } else if (Double.class.equals(wrapperType)) {
+ return double.class;
+ } else if (Character.class.equals(wrapperType)) {
+ return char.class;
+ } else {
+ if (log.isDebugEnabled()) {
+ log.debug("Not a known primitive wrapper class: " + wrapperType);
+ }
+ return null;
+ }
+ }
+
+ /**
+ * Find a non primitive representation for given primitive class.
+ *
+ * @param clazz the class to find a representation for, not null
+ * @return the original class if it not a primitive. Otherwise the wrapper class. Not null
+ */
+ public static Class toNonPrimitiveClass(Class clazz) {
+ if (clazz.isPrimitive()) {
+ Class primitiveClazz = MethodUtils.getPrimitiveWrapper(clazz);
+ // the above method returns
+ if (primitiveClazz != null) {
+ return primitiveClazz;
+ } else {
+ return clazz;
+ }
+ } else {
+ return clazz;
+ }
+ }
+
+
+ /**
+ * Represents the key to looking up a Method by reflection.
+ */
+ private static class MethodDescriptor {
+ private Class cls;
+ private String methodName;
+ private Class[] paramTypes;
+ private boolean exact;
+ private int hashCode;
+
+ /**
+ * The sole constructor.
+ *
+ * @param cls the class to reflect, must not be null
+ * @param methodName the method name to obtain
+ * @param paramTypes the array of classes representing the paramater types
+ * @param exact whether the match has to be exact.
+ */
+ public MethodDescriptor(Class cls, String methodName, Class[] paramTypes, boolean exact) {
+ if (cls == null) {
+ throw new IllegalArgumentException("Class cannot be null");
+ }
+ if (methodName == null) {
+ throw new IllegalArgumentException("Method Name cannot be null");
+ }
+ if (paramTypes == null) {
+ paramTypes = emptyClassArray;
+ }
+
+ this.cls = cls;
+ this.methodName = methodName;
+ this.paramTypes = paramTypes;
+ this.exact= exact;
+
+ this.hashCode = methodName.length();
+ }
+ /**
+ * Checks for equality.
+ * @param obj object to be tested for equality
+ * @return true, if the object describes the same Method.
+ */
+ public boolean equals(Object obj) {
+ if (!(obj instanceof MethodDescriptor)) {
+ return false;
+ }
+ MethodDescriptor md = (MethodDescriptor)obj;
+
+ return (
+ exact == md.exact &&
+ methodName.equals(md.methodName) &&
+ cls.equals(md.cls) &&
+ java.util.Arrays.equals(paramTypes, md.paramTypes)
+ );
+ }
+ /**
+ * Returns the string length of method name. I.e. if the
+ * hashcodes are different, the objects are different. If the
+ * hashcodes are the same, need to use the equals method to
+ * determine equality.
+ * @return the string length of method name.
+ */
+ public int hashCode() {
+ return hashCode;
+ }
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/MutableDynaClass.java b/trunk/src/java/org/apache/commons/beanutils/MutableDynaClass.java
new file mode 100644
index 0000000..a3ca9bb
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/MutableDynaClass.java
@@ -0,0 +1,120 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+
+
+
+/**
+ * <p>A specialized extension to <code>DynaClass</code> that allows properties
+ * to be added or removed dynamically.</p>
+ *
+ * <p><strong>WARNING</strong> - No guarantees that this will be in the final
+ * APIs ... it's here primarily to preserve some concepts that were in the
+ * original proposal for further discussion.</p>
+ *
+ * @author Craig McClanahan
+ * @author Michael Smith
+ * @author Paulo Gaspar
+ * @version $Revision: 1.8 $ $Date: 2004/02/28 13:18:33 $
+ */
+
+public interface MutableDynaClass extends DynaClass {
+
+
+ /**
+ * Add a new dynamic property with no restrictions on data type,
+ * readability, or writeability.
+ *
+ * @param name Name of the new dynamic property
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no new properties can be added
+ */
+ public void add(String name);
+
+
+ /**
+ * Add a new dynamic property with the specified data type, but with
+ * no restrictions on readability or writeability.
+ *
+ * @param name Name of the new dynamic property
+ * @param type Data type of the new dynamic property (null for no
+ * restrictions)
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no new properties can be added
+ */
+ public void add(String name, Class type);
+
+
+ /**
+ * Add a new dynamic property with the specified data type, readability,
+ * and writeability.
+ *
+ * @param name Name of the new dynamic property
+ * @param type Data type of the new dynamic property (null for no
+ * restrictions)
+ * @param readable Set to <code>true</code> if this property value
+ * should be readable
+ * @param writeable Set to <code>true</code> if this property value
+ * should be writeable
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no new properties can be added
+ */
+ public void add(String name, Class type, boolean readable,
+ boolean writeable);
+
+
+ /**
+ * Is this DynaClass currently restricted, if so, no changes to the
+ * existing registration of property names, data types, readability, or
+ * writeability are allowed.
+ */
+ public boolean isRestricted();
+
+
+ /**
+ * Remove the specified dynamic property, and any associated data type,
+ * readability, and writeability, from this dynamic class.
+ * <strong>NOTE</strong> - This does <strong>NOT</strong> cause any
+ * corresponding property values to be removed from DynaBean instances
+ * associated with this DynaClass.
+ *
+ * @param name Name of the dynamic property to remove
+ *
+ * @exception IllegalArgumentException if name is null
+ * @exception IllegalStateException if this DynaClass is currently
+ * restricted, so no properties can be removed
+ */
+ public void remove(String name);
+
+
+ /**
+ * Set the restricted state of this DynaClass to the specified value.
+ *
+ * @param restricted The new restricted state
+ */
+ public void setRestricted(boolean restricted);
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/NestedNullException.java b/trunk/src/java/org/apache/commons/beanutils/NestedNullException.java
new file mode 100644
index 0000000..07f4366
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/NestedNullException.java
@@ -0,0 +1,46 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+/**
+ * Thrown to indicate that the <em>Bean Access Language</em> cannot execute query
+ * against given bean since a nested bean referenced is null.
+ *
+ * @author Robert Burrell Donkin
+ * @since 1.7
+ */
+
+public class NestedNullException extends BeanAccessLanguageException {
+
+ // --------------------------------------------------------- Constuctors
+
+ /**
+ * Constructs a <code>NestedNullException</code> without a detail message.
+ */
+ public NestedNullException() {
+ super();
+ }
+
+ /**
+ * Constructs a <code>NestedNullException</code> without a detail message.
+ *
+ * @param message the detail message explaining this exception
+ */
+ public NestedNullException(String message) {
+ super(message);
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/PropertyUtils.java b/trunk/src/java/org/apache/commons/beanutils/PropertyUtils.java
new file mode 100644
index 0000000..d3b2d6a
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/PropertyUtils.java
@@ -0,0 +1,581 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+import java.beans.PropertyDescriptor;
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Method;
+import java.util.Map;
+
+import org.apache.commons.collections.FastHashMap;
+
+
+/**
+ * <p>Utility methods for using Java Reflection APIs to facilitate generic
+ * property getter and setter operations on Java objects.</p>
+ *
+ * <p>The implementations for these methods are provided by <code>PropertyUtilsBean</code>.
+ * For more details see {@link PropertyUtilsBean}.</p>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @author Rey Fran�ois
+ * @author Gregor Ra�man
+ * @author Jan Sorensen
+ * @author Scott Sanders
+ * @version $Revision: 1.42.2.1 $ $Date: 2004/07/27 21:31:00 $
+ * @see PropertyUtilsBean
+ */
+
+public class PropertyUtils {
+
+
+ // ----------------------------------------------------- Manifest Constants
+
+
+ /**
+ * The delimiter that preceeds the zero-relative subscript for an
+ * indexed reference.
+ */
+ public static final char INDEXED_DELIM = '[';
+
+
+ /**
+ * The delimiter that follows the zero-relative subscript for an
+ * indexed reference.
+ */
+ public static final char INDEXED_DELIM2 = ']';
+
+
+ /**
+ * The delimiter that preceeds the key of a mapped property.
+ */
+ public static final char MAPPED_DELIM = '(';
+
+
+ /**
+ * The delimiter that follows the key of a mapped property.
+ */
+ public static final char MAPPED_DELIM2 = ')';
+
+
+ /**
+ * The delimiter that separates the components of a nested reference.
+ */
+ public static final char NESTED_DELIM = '.';
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * The debugging detail level for this component.
+ * @deprecated The <code>debug</code> static property is no longer used
+ */
+ private static int debug = 0;
+
+ /**
+ * @deprecated The <code>debug</code> static property is no longer used
+ */
+ public static int getDebug() {
+ return (debug);
+ }
+
+ /**
+ * @deprecated The <code>debug</code> static property is no longer used
+ */
+ public static void setDebug(int newDebug) {
+ debug = newDebug;
+ }
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Clear any cached property descriptors information for all classes
+ * loaded by any class loaders. This is useful in cases where class
+ * loaders are thrown away to implement class reloading.
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#clearDescriptors
+ */
+ public static void clearDescriptors() {
+
+ PropertyUtilsBean.getInstance().clearDescriptors();
+
+ }
+
+
+ /**
+ * <p>Copy property values from the "origin" bean to the "destination" bean
+ * for all cases where the property names are the same (even though the
+ * actual getter and setter methods might have been customized via
+ * <code>BeanInfo</code> classes).</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#copyProperties
+ */
+ public static void copyProperties(Object dest, Object orig)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ PropertyUtilsBean.getInstance().copyProperties(dest, orig);
+ }
+
+
+ /**
+ * <p>Return the entire set of properties for which the specified bean
+ * provides a read method.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#describe
+ */
+ public static Map describe(Object bean)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return (PropertyUtilsBean.getInstance().describe(bean));
+
+ }
+
+
+ /**
+ * <p>Return the value of the specified indexed property of the specified
+ * bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getIndexedProperty(Object,String)
+ */
+ public static Object getIndexedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return (PropertyUtilsBean.getInstance().getIndexedProperty(bean, name));
+
+ }
+
+
+ /**
+ * <p>Return the value of the specified indexed property of the specified
+ * bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getIndexedProperty(Object,String, int)
+ */
+ public static Object getIndexedProperty(Object bean,
+ String name, int index)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return (PropertyUtilsBean.getInstance().getIndexedProperty(bean, name, index));
+ }
+
+
+ /**
+ * <p>Return the value of the specified mapped property of the
+ * specified bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getMappedProperty(Object,String)
+ */
+ public static Object getMappedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return (PropertyUtilsBean.getInstance().getMappedProperty(bean, name));
+
+ }
+
+
+ /**
+ * <p>Return the value of the specified mapped property of the specified
+ * bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getMappedProperty(Object,String, String)
+ */
+ public static Object getMappedProperty(Object bean,
+ String name, String key)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return PropertyUtilsBean.getInstance().getMappedProperty(bean, name, key);
+
+ }
+
+
+ /**
+ * <p>Return the mapped property descriptors for this bean class.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getMappedPropertyDescriptors(Class)
+ * @deprecated This method should not be exposed
+ */
+ public static FastHashMap getMappedPropertyDescriptors(Class beanClass) {
+
+ return PropertyUtilsBean.getInstance().getMappedPropertyDescriptors(beanClass);
+
+ }
+
+
+ /**
+ * <p>Return the mapped property descriptors for this bean.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getMappedPropertyDescriptors(Object)
+ * @deprecated This method should not be exposed
+ */
+ public static FastHashMap getMappedPropertyDescriptors(Object bean) {
+
+ return PropertyUtilsBean.getInstance().getMappedPropertyDescriptors(bean);
+
+ }
+
+
+ /**
+ * <p>Return the value of the (possibly nested) property of the specified
+ * name, for the specified bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getNestedProperty
+ */
+ public static Object getNestedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return PropertyUtilsBean.getInstance().getNestedProperty(bean, name);
+
+ }
+
+
+ /**
+ * <p>Return the value of the specified property of the specified bean,
+ * no matter which property reference format is used, with no
+ * type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getProperty
+ */
+ public static Object getProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return (PropertyUtilsBean.getInstance().getProperty(bean, name));
+
+ }
+
+
+ /**
+ * <p>Retrieve the property descriptor for the specified property of the
+ * specified bean, or return <code>null</code> if there is no such
+ * descriptor.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getPropertyDescriptor
+ */
+ public static PropertyDescriptor getPropertyDescriptor(Object bean,
+ String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return PropertyUtilsBean.getInstance().getPropertyDescriptor(bean, name);
+
+ }
+
+
+ /**
+ * <p>Retrieve the property descriptors for the specified class,
+ * introspecting and caching them the first time a particular bean class
+ * is encountered.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getPropertyDescriptors(Class)
+ */
+ public static PropertyDescriptor[]
+ getPropertyDescriptors(Class beanClass) {
+
+ return PropertyUtilsBean.getInstance().getPropertyDescriptors(beanClass);
+
+ }
+
+
+ /**
+ * <p>Retrieve the property descriptors for the specified bean,
+ * introspecting and caching them the first time a particular bean class
+ * is encountered.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getPropertyDescriptors(Object)
+ */
+ public static PropertyDescriptor[] getPropertyDescriptors(Object bean) {
+
+ return PropertyUtilsBean.getInstance().getPropertyDescriptors(bean);
+
+ }
+
+
+ /**
+ * <p>Return the Java Class repesenting the property editor class that has
+ * been registered for this property (if any).</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getPropertyEditorClass(Object,String)
+ */
+ public static Class getPropertyEditorClass(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return PropertyUtilsBean.getInstance().getPropertyEditorClass(bean, name);
+
+ }
+
+
+ /**
+ * <p>Return the Java Class representing the property type of the specified
+ * property, or <code>null</code> if there is no such property for the
+ * specified bean.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getPropertyType
+ */
+ public static Class getPropertyType(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return PropertyUtilsBean.getInstance().getPropertyType(bean, name);
+ }
+
+
+ /**
+ * <p>Return an accessible property getter method for this property,
+ * if there is one; otherwise return <code>null</code>.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getReadMethod
+ */
+ public static Method getReadMethod(PropertyDescriptor descriptor) {
+
+ return (PropertyUtilsBean.getInstance().getReadMethod(descriptor));
+
+ }
+
+
+ /**
+ * <p>Return the value of the specified simple property of the specified
+ * bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getSimpleProperty
+ */
+ public static Object getSimpleProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return PropertyUtilsBean.getInstance().getSimpleProperty(bean, name);
+
+ }
+
+
+ /**
+ * <p>Return an accessible property setter method for this property,
+ * if there is one; otherwise return <code>null</code>.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#getWriteMethod
+ */
+ public static Method getWriteMethod(PropertyDescriptor descriptor) {
+
+ return PropertyUtilsBean.getInstance().getWriteMethod(descriptor);
+
+ }
+
+
+ /**
+ * <p>Return <code>true</code> if the specified property name identifies
+ * a readable property on the specified bean; otherwise, return
+ * <code>false</code>.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#isReadable
+ * @since BeanUtils 1.6
+ */
+ public static boolean isReadable(Object bean, String name) {
+
+ return PropertyUtilsBean.getInstance().isReadable(bean, name);
+ }
+
+
+ /**
+ * <p>Return <code>true</code> if the specified property name identifies
+ * a writeable property on the specified bean; otherwise, return
+ * <code>false</code>.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#isWriteable
+ * @since BeanUtils 1.6
+ */
+ public static boolean isWriteable(Object bean, String name) {
+
+ return PropertyUtilsBean.getInstance().isWriteable(bean, name);
+ }
+
+
+ /**
+ * <p>Sets the value of the specified indexed property of the specified
+ * bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#setIndexedProperty(Object, String, Object)
+ */
+ public static void setIndexedProperty(Object bean, String name,
+ Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ PropertyUtilsBean.getInstance().setIndexedProperty(bean, name, value);
+
+ }
+
+
+ /**
+ * <p>Sets the value of the specified indexed property of the specified
+ * bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#setIndexedProperty(Object, String, Object)
+ */
+ public static void setIndexedProperty(Object bean, String name,
+ int index, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ PropertyUtilsBean.getInstance().setIndexedProperty(bean, name, index, value);
+ }
+
+
+ /**
+ * <p>Sets the value of the specified mapped property of the
+ * specified bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#setMappedProperty(Object, String, Object)
+ */
+ public static void setMappedProperty(Object bean, String name,
+ Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ PropertyUtilsBean.getInstance().setMappedProperty(bean, name, value);
+ }
+
+
+ /**
+ * <p>Sets the value of the specified mapped property of the specified
+ * bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#setMappedProperty(Object, String, String, Object)
+ */
+ public static void setMappedProperty(Object bean, String name,
+ String key, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ PropertyUtilsBean.getInstance().setMappedProperty(bean, name, key, value);
+ }
+
+
+ /**
+ * <p>Sets the value of the (possibly nested) property of the specified
+ * name, for the specified bean, with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#setNestedProperty
+ */
+ public static void setNestedProperty(Object bean,
+ String name, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ PropertyUtilsBean.getInstance().setNestedProperty(bean, name, value);
+ }
+
+
+ /**
+ * <p>Set the value of the specified property of the specified bean,
+ * no matter which property reference format is used, with no
+ * type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#setProperty
+ */
+ public static void setProperty(Object bean, String name, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ PropertyUtilsBean.getInstance().setProperty(bean, name, value);
+
+ }
+
+
+ /**
+ * <p>Set the value of the specified simple property of the specified bean,
+ * with no type conversions.</p>
+ *
+ * <p>For more details see <code>PropertyUtilsBean</code>.</p>
+ *
+ * @see PropertyUtilsBean#setSimpleProperty
+ */
+ public static void setSimpleProperty(Object bean,
+ String name, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ PropertyUtilsBean.getInstance().setSimpleProperty(bean, name, value);
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/PropertyUtilsBean.java b/trunk/src/java/org/apache/commons/beanutils/PropertyUtilsBean.java
new file mode 100644
index 0000000..e3a5ae2
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/PropertyUtilsBean.java
@@ -0,0 +1,1784 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.beans.BeanInfo;
+import java.beans.IndexedPropertyDescriptor;
+import java.beans.IntrospectionException;
+import java.beans.Introspector;
+import java.beans.PropertyDescriptor;
+import java.lang.reflect.Array;
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Method;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Map;
+
+import org.apache.commons.collections.FastHashMap;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+
+/**
+ * Utility methods for using Java Reflection APIs to facilitate generic
+ * property getter and setter operations on Java objects. Much of this
+ * code was originally included in <code>BeanUtils</code>, but has been
+ * separated because of the volume of code involved.
+ * <p>
+ * In general, the objects that are examined and modified using these
+ * methods are expected to conform to the property getter and setter method
+ * naming conventions described in the JavaBeans Specification (Version 1.0.1).
+ * No data type conversions are performed, and there are no usage of any
+ * <code>PropertyEditor</code> classes that have been registered, although
+ * a convenient way to access the registered classes themselves is included.
+ * <p>
+ * For the purposes of this class, five formats for referencing a particular
+ * property value of a bean are defined, with the layout of an identifying
+ * String in parentheses:
+ * <ul>
+ * <li><strong>Simple (<code>name</code>)</strong> - The specified
+ * <code>name</code> identifies an individual property of a particular
+ * JavaBean. The name of the actual getter or setter method to be used
+ * is determined using standard JavaBeans instrospection, so that (unless
+ * overridden by a <code>BeanInfo</code> class, a property named "xyz"
+ * will have a getter method named <code>getXyz()</code> or (for boolean
+ * properties only) <code>isXyz()</code>, and a setter method named
+ * <code>setXyz()</code>.</li>
+ * <li><strong>Nested (<code>name1.name2.name3</code>)</strong> The first
+ * name element is used to select a property getter, as for simple
+ * references above. The object returned for this property is then
+ * consulted, using the same approach, for a property getter for a
+ * property named <code>name2</code>, and so on. The property value that
+ * is ultimately retrieved or modified is the one identified by the
+ * last name element.</li>
+ * <li><strong>Indexed (<code>name[index]</code>)</strong> - The underlying
+ * property value is assumed to be an array, or this JavaBean is assumed
+ * to have indexed property getter and setter methods. The appropriate
+ * (zero-relative) entry in the array is selected. <code>List</code>
+ * objects are now also supported for read/write. You simply need to define
+ * a getter that returns the <code>List</code></li>
+ * <li><strong>Mapped (<code>name(key)</code>)</strong> - The JavaBean
+ * is assumed to have an property getter and setter methods with an
+ * additional attribute of type <code>java.lang.String</code>.</li>
+ * <li><strong>Combined (<code>name1.name2[index].name3(key)</code>)</strong> -
+ * Combining mapped, nested, and indexed references is also
+ * supported.</li>
+ * </ul>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @author Rey Fran�ois
+ * @author Gregor Ra�man
+ * @author Jan Sorensen
+ * @author Scott Sanders
+ * @version $Revision: 1.14.2.1 $ $Date: 2004/07/27 21:31:00 $
+ * @see PropertyUtils
+ * @since 1.7
+ */
+
+public class PropertyUtilsBean {
+
+ // --------------------------------------------------------- Class Methods
+
+ protected static PropertyUtilsBean getInstance() {
+ return BeanUtilsBean.getInstance().getPropertyUtils();
+ }
+
+ // --------------------------------------------------------- Variables
+
+ /**
+ * The cache of PropertyDescriptor arrays for beans we have already
+ * introspected, keyed by the java.lang.Class of this object.
+ */
+ private FastHashMap descriptorsCache = null;
+ private FastHashMap mappedDescriptorsCache = null;
+
+ /** Log instance */
+ private Log log = LogFactory.getLog(PropertyUtils.class);
+
+ // ---------------------------------------------------------- Constructors
+
+ /** Base constructor */
+ public PropertyUtilsBean() {
+ descriptorsCache = new FastHashMap();
+ descriptorsCache.setFast(true);
+ mappedDescriptorsCache = new FastHashMap();
+ mappedDescriptorsCache.setFast(true);
+ }
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Clear any cached property descriptors information for all classes
+ * loaded by any class loaders. This is useful in cases where class
+ * loaders are thrown away to implement class reloading.
+ */
+ public void clearDescriptors() {
+
+ descriptorsCache.clear();
+ mappedDescriptorsCache.clear();
+ Introspector.flushCaches();
+
+ }
+
+
+ /**
+ * <p>Copy property values from the "origin" bean to the "destination" bean
+ * for all cases where the property names are the same (even though the
+ * actual getter and setter methods might have been customized via
+ * <code>BeanInfo</code> classes). No conversions are performed on the
+ * actual property values -- it is assumed that the values retrieved from
+ * the origin bean are assignment-compatible with the types expected by
+ * the destination bean.</p>
+ *
+ * <p>If the origin "bean" is actually a <code>Map</code>, it is assumed
+ * to contain String-valued <strong>simple</strong> property names as the keys, pointing
+ * at the corresponding property values that will be set in the destination
+ * bean.<strong>Note</strong> that this method is intended to perform
+ * a "shallow copy" of the properties and so complex properties
+ * (for example, nested ones) will not be copied.</p>
+ *
+ * @param dest Destination bean whose properties are modified
+ * @param orig Origin bean whose properties are retrieved
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if the <code>dest</code> or
+ * <code>orig</code> argument is null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public void copyProperties(Object dest, Object orig)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (dest == null) {
+ throw new IllegalArgumentException
+ ("No destination bean specified");
+ }
+ if (orig == null) {
+ throw new IllegalArgumentException("No origin bean specified");
+ }
+
+ if (orig instanceof DynaBean) {
+ DynaProperty origDescriptors[] =
+ ((DynaBean) orig).getDynaClass().getDynaProperties();
+ for (int i = 0; i < origDescriptors.length; i++) {
+ String name = origDescriptors[i].getName();
+ if (dest instanceof DynaBean) {
+ if (isWriteable(dest, name)) {
+ Object value = ((DynaBean) orig).get(name);
+ ((DynaBean) dest).set(name, value);
+ }
+ } else /* if (dest is a standard JavaBean) */ {
+ if (isWriteable(dest, name)) {
+ Object value = ((DynaBean) orig).get(name);
+ setSimpleProperty(dest, name, value);
+ }
+ }
+ }
+ } else if (orig instanceof Map) {
+ Iterator names = ((Map) orig).keySet().iterator();
+ while (names.hasNext()) {
+ String name = (String) names.next();
+ if (dest instanceof DynaBean) {
+ if (isWriteable(dest, name)) {
+ Object value = ((Map) orig).get(name);
+ ((DynaBean) dest).set(name, value);
+ }
+ } else /* if (dest is a standard JavaBean) */ {
+ if (isWriteable(dest, name)) {
+ Object value = ((Map) orig).get(name);
+ setSimpleProperty(dest, name, value);
+ }
+ }
+ }
+ } else /* if (orig is a standard JavaBean) */ {
+ PropertyDescriptor origDescriptors[] =
+ getPropertyDescriptors(orig);
+ for (int i = 0; i < origDescriptors.length; i++) {
+ String name = origDescriptors[i].getName();
+ if (isReadable(orig, name)) {
+ if (dest instanceof DynaBean) {
+ if (isWriteable(dest, name)) {
+ Object value = getSimpleProperty(orig, name);
+ ((DynaBean) dest).set(name, value);
+ }
+ } else /* if (dest is a standard JavaBean) */ {
+ if (isWriteable(dest, name)) {
+ Object value = getSimpleProperty(orig, name);
+ setSimpleProperty(dest, name, value);
+ }
+ }
+ }
+ }
+ }
+
+ }
+
+
+ /**
+ * <p>Return the entire set of properties for which the specified bean
+ * provides a read method. This map contains the unconverted property
+ * values for all properties for which a read method is provided
+ * (i.e. where the <code>getReadMethod()</code> returns non-null).</p>
+ *
+ * <p><strong>FIXME</strong> - Does not account for mapped properties.</p>
+ *
+ * @param bean Bean whose properties are to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> is null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Map describe(Object bean)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ Map description = new HashMap();
+ if (bean instanceof DynaBean) {
+ DynaProperty descriptors[] =
+ ((DynaBean) bean).getDynaClass().getDynaProperties();
+ for (int i = 0; i < descriptors.length; i++) {
+ String name = descriptors[i].getName();
+ description.put(name, getProperty(bean, name));
+ }
+ } else {
+ PropertyDescriptor descriptors[] =
+ getPropertyDescriptors(bean);
+ for (int i = 0; i < descriptors.length; i++) {
+ String name = descriptors[i].getName();
+ if (descriptors[i].getReadMethod() != null)
+ description.put(name, getProperty(bean, name));
+ }
+ }
+ return (description);
+
+ }
+
+
+ /**
+ * Return the value of the specified indexed property of the specified
+ * bean, with no type conversions. The zero-relative index of the
+ * required value must be included (in square brackets) as a suffix to
+ * the property name, or <code>IllegalArgumentException</code> will be
+ * thrown. In addition to supporting the JavaBeans specification, this
+ * method has been extended to support <code>List</code> objects as well.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name <code>propertyname[index]</code> of the property value
+ * to be extracted
+ *
+ * @exception ArrayIndexOutOfBoundsException if the specified index
+ * is outside the valid range for the underlying array
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Object getIndexedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Identify the index of the requested individual property
+ int delim = name.indexOf(PropertyUtils.INDEXED_DELIM);
+ int delim2 = name.indexOf(PropertyUtils.INDEXED_DELIM2);
+ if ((delim < 0) || (delim2 <= delim)) {
+ throw new IllegalArgumentException("Invalid indexed property '" +
+ name + "'");
+ }
+ int index = -1;
+ try {
+ String subscript = name.substring(delim + 1, delim2);
+ index = Integer.parseInt(subscript);
+ } catch (NumberFormatException e) {
+ throw new IllegalArgumentException("Invalid indexed property '" +
+ name + "'");
+ }
+ name = name.substring(0, delim);
+
+ // Request the specified indexed property value
+ return (getIndexedProperty(bean, name, index));
+
+ }
+
+
+ /**
+ * Return the value of the specified indexed property of the specified
+ * bean, with no type conversions. In addition to supporting the JavaBeans
+ * specification, this method has been extended to support
+ * <code>List</code> objects as well.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Simple property name of the property value to be extracted
+ * @param index Index of the property value to be extracted
+ *
+ * @exception ArrayIndexOutOfBoundsException if the specified index
+ * is outside the valid range for the underlying array
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Object getIndexedProperty(Object bean,
+ String name, int index)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Handle DynaBean instances specially
+ if (bean instanceof DynaBean) {
+ DynaProperty descriptor =
+ ((DynaBean) bean).getDynaClass().getDynaProperty(name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+ return (((DynaBean) bean).get(name, index));
+ }
+
+ // Retrieve the property descriptor for the specified property
+ PropertyDescriptor descriptor =
+ getPropertyDescriptor(bean, name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+
+ // Call the indexed getter method if there is one
+ if (descriptor instanceof IndexedPropertyDescriptor) {
+ Method readMethod = ((IndexedPropertyDescriptor) descriptor).
+ getIndexedReadMethod();
+ if (readMethod != null) {
+ Object subscript[] = new Object[1];
+ subscript[0] = new Integer(index);
+ try {
+ return (invokeMethod(readMethod,bean, subscript));
+ } catch (InvocationTargetException e) {
+ if (e.getTargetException() instanceof
+ ArrayIndexOutOfBoundsException) {
+ throw (ArrayIndexOutOfBoundsException)
+ e.getTargetException();
+ } else {
+ throw e;
+ }
+ }
+ }
+ }
+
+ // Otherwise, the underlying property must be an array
+ Method readMethod = getReadMethod(descriptor);
+ if (readMethod == null) {
+ throw new NoSuchMethodException("Property '" + name +
+ "' has no getter method");
+ }
+
+ // Call the property getter and return the value
+ Object value = invokeMethod(readMethod, bean, new Object[0]);
+ if (!value.getClass().isArray()) {
+ if (!(value instanceof java.util.List)) {
+ throw new IllegalArgumentException("Property '" + name
+ + "' is not indexed");
+ } else {
+ //get the List's value
+ return ((java.util.List) value).get(index);
+ }
+ } else {
+ //get the array's value
+ return (Array.get(value, index));
+ }
+
+ }
+
+
+ /**
+ * Return the value of the specified mapped property of the
+ * specified bean, with no type conversions. The key of the
+ * required value must be included (in brackets) as a suffix to
+ * the property name, or <code>IllegalArgumentException</code> will be
+ * thrown.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name <code>propertyname(key)</code> of the property value
+ * to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Object getMappedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Identify the index of the requested individual property
+ int delim = name.indexOf(PropertyUtils.MAPPED_DELIM);
+ int delim2 = name.indexOf(PropertyUtils.MAPPED_DELIM2);
+ if ((delim < 0) || (delim2 <= delim)) {
+ throw new IllegalArgumentException
+ ("Invalid mapped property '" + name + "'");
+ }
+
+ // Isolate the name and the key
+ String key = name.substring(delim + 1, delim2);
+ name = name.substring(0, delim);
+
+ // Request the specified indexed property value
+ return (getMappedProperty(bean, name, key));
+
+ }
+
+
+ /**
+ * Return the value of the specified mapped property of the specified
+ * bean, with no type conversions.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Mapped property name of the property value to be extracted
+ * @param key Key of the property value to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Object getMappedProperty(Object bean,
+ String name, String key)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+ if (key == null) {
+ throw new IllegalArgumentException("No key specified");
+ }
+
+ // Handle DynaBean instances specially
+ if (bean instanceof DynaBean) {
+ DynaProperty descriptor =
+ ((DynaBean) bean).getDynaClass().getDynaProperty(name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+ return (((DynaBean) bean).get(name, key));
+ }
+
+ Object result = null;
+
+ // Retrieve the property descriptor for the specified property
+ PropertyDescriptor descriptor = getPropertyDescriptor(bean, name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+
+ if (descriptor instanceof MappedPropertyDescriptor) {
+ // Call the keyed getter method if there is one
+ Method readMethod = ((MappedPropertyDescriptor) descriptor).
+ getMappedReadMethod();
+ if (readMethod != null) {
+ Object keyArray[] = new Object[1];
+ keyArray[0] = key;
+ result = invokeMethod(readMethod, bean, keyArray);
+ } else {
+ throw new NoSuchMethodException("Property '" + name +
+ "' has no mapped getter method");
+ }
+ } else {
+ /* means that the result has to be retrieved from a map */
+ Method readMethod = descriptor.getReadMethod();
+ if (readMethod != null) {
+ Object invokeResult = invokeMethod(readMethod, bean, new Object[0]);
+ /* test and fetch from the map */
+ if (invokeResult instanceof java.util.Map) {
+ result = ((java.util.Map)invokeResult).get(key);
+ }
+ } else {
+ throw new NoSuchMethodException("Property '" + name +
+ "' has no mapped getter method");
+ }
+ }
+ return result;
+
+ }
+
+
+ /**
+ * <p>Return the mapped property descriptors for this bean class.</p>
+ *
+ * <p><strong>FIXME</strong> - Does not work with DynaBeans.</p>
+ *
+ * @param beanClass Bean class to be introspected
+ * @deprecated This method should not be exposed
+ */
+ public FastHashMap getMappedPropertyDescriptors(Class beanClass) {
+
+ if (beanClass == null) {
+ return null;
+ }
+
+ // Look up any cached descriptors for this bean class
+ return (FastHashMap) mappedDescriptorsCache.get(beanClass);
+
+ }
+
+
+ /**
+ * <p>Return the mapped property descriptors for this bean.</p>
+ *
+ * <p><strong>FIXME</strong> - Does not work with DynaBeans.</p>
+ *
+ * @param bean Bean to be introspected
+ * @deprecated This method should not be exposed
+ */
+ public FastHashMap getMappedPropertyDescriptors(Object bean) {
+
+ if (bean == null) {
+ return null;
+ }
+ return (getMappedPropertyDescriptors(bean.getClass()));
+
+ }
+
+
+ /**
+ * Return the value of the (possibly nested) property of the specified
+ * name, for the specified bean, with no type conversions.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Possibly nested name of the property to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception NestedNullException if a nested reference to a
+ * property returns null
+ * @exception InvocationTargetException
+ * if the property accessor method throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Object getNestedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ int indexOfINDEXED_DELIM = -1;
+ int indexOfMAPPED_DELIM = -1;
+ int indexOfMAPPED_DELIM2 = -1;
+ int indexOfNESTED_DELIM = -1;
+ while (true) {
+ indexOfNESTED_DELIM = name.indexOf(PropertyUtils.NESTED_DELIM);
+ indexOfMAPPED_DELIM = name.indexOf(PropertyUtils.MAPPED_DELIM);
+ indexOfMAPPED_DELIM2 = name.indexOf(PropertyUtils.MAPPED_DELIM2);
+ if (indexOfMAPPED_DELIM2 >= 0 && indexOfMAPPED_DELIM >=0 &&
+ (indexOfNESTED_DELIM < 0 || indexOfNESTED_DELIM > indexOfMAPPED_DELIM)) {
+ indexOfNESTED_DELIM =
+ name.indexOf(PropertyUtils.NESTED_DELIM, indexOfMAPPED_DELIM2);
+ } else {
+ indexOfNESTED_DELIM = name.indexOf(PropertyUtils.NESTED_DELIM);
+ }
+ if (indexOfNESTED_DELIM < 0) {
+ break;
+ }
+ String next = name.substring(0, indexOfNESTED_DELIM);
+ indexOfINDEXED_DELIM = next.indexOf(PropertyUtils.INDEXED_DELIM);
+ indexOfMAPPED_DELIM = next.indexOf(PropertyUtils.MAPPED_DELIM);
+ if (bean instanceof Map) {
+ bean = ((Map) bean).get(next);
+ } else if (indexOfMAPPED_DELIM >= 0) {
+ bean = getMappedProperty(bean, next);
+ } else if (indexOfINDEXED_DELIM >= 0) {
+ bean = getIndexedProperty(bean, next);
+ } else {
+ bean = getSimpleProperty(bean, next);
+ }
+ if (bean == null) {
+ throw new NestedNullException
+ ("Null property value for '" +
+ name.substring(0, indexOfNESTED_DELIM) + "'");
+ }
+ name = name.substring(indexOfNESTED_DELIM + 1);
+ }
+
+ indexOfINDEXED_DELIM = name.indexOf(PropertyUtils.INDEXED_DELIM);
+ indexOfMAPPED_DELIM = name.indexOf(PropertyUtils.MAPPED_DELIM);
+
+ if (bean instanceof Map) {
+ bean = ((Map) bean).get(name);
+ } else if (indexOfMAPPED_DELIM >= 0) {
+ bean = getMappedProperty(bean, name);
+ } else if (indexOfINDEXED_DELIM >= 0) {
+ bean = getIndexedProperty(bean, name);
+ } else {
+ bean = getSimpleProperty(bean, name);
+ }
+ return bean;
+
+ }
+
+
+ /**
+ * Return the value of the specified property of the specified bean,
+ * no matter which property reference format is used, with no
+ * type conversions.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Possibly indexed and/or nested name of the property
+ * to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Object getProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return (getNestedProperty(bean, name));
+
+ }
+
+
+ /**
+ * <p>Retrieve the property descriptor for the specified property of the
+ * specified bean, or return <code>null</code> if there is no such
+ * descriptor. This method resolves indexed and nested property
+ * references in the same manner as other methods in this class, except
+ * that if the last (or only) name element is indexed, the descriptor
+ * for the last resolved property itself is returned.</p>
+ *
+ * <p><strong>FIXME</strong> - Does not work with DynaBeans.</p>
+ *
+ * @param bean Bean for which a property descriptor is requested
+ * @param name Possibly indexed and/or nested name of the property for
+ * which a property descriptor is requested
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception IllegalArgumentException if a nested reference to a
+ * property returns null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public PropertyDescriptor getPropertyDescriptor(Object bean,
+ String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Resolve nested references
+ while (true) {
+ int period = findNextNestedIndex(name);
+ if (period < 0) {
+ break;
+ }
+ String next = name.substring(0, period);
+ int indexOfINDEXED_DELIM = next.indexOf(PropertyUtils.INDEXED_DELIM);
+ int indexOfMAPPED_DELIM = next.indexOf(PropertyUtils.MAPPED_DELIM);
+ if (indexOfMAPPED_DELIM >= 0 &&
+ (indexOfINDEXED_DELIM < 0 ||
+ indexOfMAPPED_DELIM < indexOfINDEXED_DELIM)) {
+ bean = getMappedProperty(bean, next);
+ } else {
+ if (indexOfINDEXED_DELIM >= 0) {
+ bean = getIndexedProperty(bean, next);
+ } else {
+ bean = getSimpleProperty(bean, next);
+ }
+ }
+ if (bean == null) {
+ throw new IllegalArgumentException
+ ("Null property value for '" +
+ name.substring(0, period) + "'");
+ }
+ name = name.substring(period + 1);
+ }
+
+ // Remove any subscript from the final name value
+ int left = name.indexOf(PropertyUtils.INDEXED_DELIM);
+ if (left >= 0) {
+ name = name.substring(0, left);
+ }
+ left = name.indexOf(PropertyUtils.MAPPED_DELIM);
+ if (left >= 0) {
+ name = name.substring(0, left);
+ }
+
+ // Look up and return this property from our cache
+ // creating and adding it to the cache if not found.
+ if ((bean == null) || (name == null)) {
+ return (null);
+ }
+
+ PropertyDescriptor descriptors[] = getPropertyDescriptors(bean);
+ if (descriptors != null) {
+
+ for (int i = 0; i < descriptors.length; i++) {
+ if (name.equals(descriptors[i].getName()))
+ return (descriptors[i]);
+ }
+ }
+
+ PropertyDescriptor result = null;
+ FastHashMap mappedDescriptors =
+ getMappedPropertyDescriptors(bean);
+ if (mappedDescriptors == null) {
+ mappedDescriptors = new FastHashMap();
+ mappedDescriptors.setFast(true);
+ mappedDescriptorsCache.put(bean.getClass(), mappedDescriptors);
+ }
+ result = (PropertyDescriptor) mappedDescriptors.get(name);
+ if (result == null) {
+ // not found, try to create it
+ try {
+ result =
+ new MappedPropertyDescriptor(name, bean.getClass());
+ } catch (IntrospectionException ie) {
+ }
+ if (result != null) {
+ mappedDescriptors.put(name, result);
+ }
+ }
+
+ return result;
+
+ }
+
+ private int findNextNestedIndex(String expression)
+ {
+ // walk back from the end to the start
+ // and find the first index that
+ int bracketCount = 0;
+ for (int i=0, size=expression.length(); i<size ; i++) {
+ char at = expression.charAt(i);
+ switch (at) {
+ case PropertyUtils.NESTED_DELIM:
+ if (bracketCount < 1) {
+ return i;
+ }
+ break;
+
+ case PropertyUtils.MAPPED_DELIM:
+ case PropertyUtils.INDEXED_DELIM:
+ // not bothered which
+ ++bracketCount;
+ break;
+
+ case PropertyUtils.MAPPED_DELIM2:
+ case PropertyUtils.INDEXED_DELIM2:
+ // not bothered which
+ --bracketCount;
+ break;
+ }
+ }
+ // can't find any
+ return -1;
+ }
+
+
+ /**
+ * <p>Retrieve the property descriptors for the specified class,
+ * introspecting and caching them the first time a particular bean class
+ * is encountered.</p>
+ *
+ * <p><strong>FIXME</strong> - Does not work with DynaBeans.</p>
+ *
+ * @param beanClass Bean class for which property descriptors are requested
+ *
+ * @exception IllegalArgumentException if <code>beanClass</code> is null
+ */
+ public PropertyDescriptor[]
+ getPropertyDescriptors(Class beanClass) {
+
+ if (beanClass == null) {
+ throw new IllegalArgumentException("No bean class specified");
+ }
+
+ // Look up any cached descriptors for this bean class
+ PropertyDescriptor descriptors[] = null;
+ descriptors =
+ (PropertyDescriptor[]) descriptorsCache.get(beanClass);
+ if (descriptors != null) {
+ return (descriptors);
+ }
+
+ // Introspect the bean and cache the generated descriptors
+ BeanInfo beanInfo = null;
+ try {
+ beanInfo = Introspector.getBeanInfo(beanClass);
+ } catch (IntrospectionException e) {
+ return (new PropertyDescriptor[0]);
+ }
+ descriptors = beanInfo.getPropertyDescriptors();
+ if (descriptors == null) {
+ descriptors = new PropertyDescriptor[0];
+ }
+ descriptorsCache.put(beanClass, descriptors);
+ return (descriptors);
+
+ }
+
+
+ /**
+ * <p>Retrieve the property descriptors for the specified bean,
+ * introspecting and caching them the first time a particular bean class
+ * is encountered.</p>
+ *
+ * <p><strong>FIXME</strong> - Does not work with DynaBeans.</p>
+ *
+ * @param bean Bean for which property descriptors are requested
+ *
+ * @exception IllegalArgumentException if <code>bean</code> is null
+ */
+ public PropertyDescriptor[] getPropertyDescriptors(Object bean) {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ return (getPropertyDescriptors(bean.getClass()));
+
+ }
+
+
+ /**
+ * <p>Return the Java Class repesenting the property editor class that has
+ * been registered for this property (if any). This method follows the
+ * same name resolution rules used by <code>getPropertyDescriptor()</code>,
+ * so if the last element of a name reference is indexed, the property
+ * editor for the underlying property's class is returned.</p>
+ *
+ * <p>Note that <code>null</code> will be returned if there is no property,
+ * or if there is no registered property editor class. Because this
+ * return value is ambiguous, you should determine the existence of the
+ * property itself by other means.</p>
+ *
+ * <p><strong>FIXME</strong> - Does not work with DynaBeans.</p>
+ *
+ * @param bean Bean for which a property descriptor is requested
+ * @param name Possibly indexed and/or nested name of the property for
+ * which a property descriptor is requested
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception IllegalArgumentException if a nested reference to a
+ * property returns null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Class getPropertyEditorClass(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ PropertyDescriptor descriptor =
+ getPropertyDescriptor(bean, name);
+ if (descriptor != null) {
+ return (descriptor.getPropertyEditorClass());
+ } else {
+ return (null);
+ }
+
+ }
+
+
+ /**
+ * Return the Java Class representing the property type of the specified
+ * property, or <code>null</code> if there is no such property for the
+ * specified bean. This method follows the same name resolution rules
+ * used by <code>getPropertyDescriptor()</code>, so if the last element
+ * of a name reference is indexed, the type of the property itself will
+ * be returned. If the last (or only) element has no property with the
+ * specified name, <code>null</code> is returned.
+ *
+ * @param bean Bean for which a property descriptor is requested
+ * @param name Possibly indexed and/or nested name of the property for
+ * which a property descriptor is requested
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception IllegalArgumentException if a nested reference to a
+ * property returns null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Class getPropertyType(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Special handling for DynaBeans
+ if (bean instanceof DynaBean) {
+ DynaProperty descriptor =
+ ((DynaBean) bean).getDynaClass().getDynaProperty(name);
+ if (descriptor == null) {
+ return (null);
+ }
+ Class type = descriptor.getType();
+ if (type == null) {
+ return (null);
+ } else if (type.isArray()) {
+ return (type.getComponentType());
+ } else {
+ return (type);
+ }
+ }
+
+ PropertyDescriptor descriptor =
+ getPropertyDescriptor(bean, name);
+ if (descriptor == null) {
+ return (null);
+ } else if (descriptor instanceof IndexedPropertyDescriptor) {
+ return (((IndexedPropertyDescriptor) descriptor).
+ getIndexedPropertyType());
+ } else if (descriptor instanceof MappedPropertyDescriptor) {
+ return (((MappedPropertyDescriptor) descriptor).
+ getMappedPropertyType());
+ } else {
+ return (descriptor.getPropertyType());
+ }
+
+ }
+
+
+ /**
+ * <p>Return an accessible property getter method for this property,
+ * if there is one; otherwise return <code>null</code>.</p>
+ *
+ * <p><strong>FIXME</strong> - Does not work with DynaBeans.</p>
+ *
+ * @param descriptor Property descriptor to return a getter for
+ */
+ public Method getReadMethod(PropertyDescriptor descriptor) {
+
+ return (MethodUtils.getAccessibleMethod(descriptor.getReadMethod()));
+
+ }
+
+
+ /**
+ * Return the value of the specified simple property of the specified
+ * bean, with no type conversions.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Name of the property to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception IllegalArgumentException if the property name
+ * is nested or indexed
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public Object getSimpleProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Validate the syntax of the property name
+ if (name.indexOf(PropertyUtils.NESTED_DELIM) >= 0) {
+ throw new IllegalArgumentException
+ ("Nested property names are not allowed");
+ } else if (name.indexOf(PropertyUtils.INDEXED_DELIM) >= 0) {
+ throw new IllegalArgumentException
+ ("Indexed property names are not allowed");
+ } else if (name.indexOf(PropertyUtils.MAPPED_DELIM) >= 0) {
+ throw new IllegalArgumentException
+ ("Mapped property names are not allowed");
+ }
+
+ // Handle DynaBean instances specially
+ if (bean instanceof DynaBean) {
+ DynaProperty descriptor =
+ ((DynaBean) bean).getDynaClass().getDynaProperty(name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+ return (((DynaBean) bean).get(name));
+ }
+
+ // Retrieve the property getter method for the specified property
+ PropertyDescriptor descriptor =
+ getPropertyDescriptor(bean, name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+ Method readMethod = getReadMethod(descriptor);
+ if (readMethod == null) {
+ throw new NoSuchMethodException("Property '" + name +
+ "' has no getter method");
+ }
+
+ // Call the property getter and return the value
+ Object value = invokeMethod(readMethod, bean, new Object[0]);
+ return (value);
+
+ }
+
+
+ /**
+ * <p>Return an accessible property setter method for this property,
+ * if there is one; otherwise return <code>null</code>.</p>
+ *
+ * <p><strong>FIXME</strong> - Does not work with DynaBeans.</p>
+ *
+ * @param descriptor Property descriptor to return a setter for
+ */
+ public Method getWriteMethod(PropertyDescriptor descriptor) {
+
+ return (MethodUtils.getAccessibleMethod(descriptor.getWriteMethod()));
+
+ }
+
+
+ /**
+ * <p>Return <code>true</code> if the specified property name identifies
+ * a readable property on the specified bean; otherwise, return
+ * <code>false</code>.
+ *
+ * @param bean Bean to be examined (may be a {@link DynaBean}
+ * @param name Property name to be evaluated
+ *
+ * @exception IllegalArgumentException if <code>bean</code>
+ * or <code>name</code> is <code>null</code>
+ *
+ * @since BeanUtils 1.6
+ */
+ public boolean isReadable(Object bean, String name) {
+
+ // Validate method parameters
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Return the requested result
+ if (bean instanceof DynaBean) {
+ // All DynaBean properties are readable
+ return (((DynaBean) bean).getDynaClass().getDynaProperty(name) != null);
+ } else {
+ try {
+ PropertyDescriptor desc =
+ getPropertyDescriptor(bean, name);
+ if (desc != null) {
+ Method readMethod = desc.getReadMethod();
+ if ((readMethod == null) &&
+ (desc instanceof IndexedPropertyDescriptor)) {
+ readMethod = ((IndexedPropertyDescriptor) desc).getIndexedReadMethod();
+ }
+ return (readMethod != null);
+ } else {
+ return (false);
+ }
+ } catch (IllegalAccessException e) {
+ return (false);
+ } catch (InvocationTargetException e) {
+ return (false);
+ } catch (NoSuchMethodException e) {
+ return (false);
+ }
+ }
+
+ }
+
+
+ /**
+ * <p>Return <code>true</code> if the specified property name identifies
+ * a writeable property on the specified bean; otherwise, return
+ * <code>false</code>.
+ *
+ * @param bean Bean to be examined (may be a {@link DynaBean}
+ * @param name Property name to be evaluated
+ *
+ * @exception IllegalPointerException if <code>bean</code>
+ * or <code>name</code> is <code>null</code>
+ *
+ * @since BeanUtils 1.6
+ */
+ public boolean isWriteable(Object bean, String name) {
+
+ // Validate method parameters
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Return the requested result
+ if (bean instanceof DynaBean) {
+ // All DynaBean properties are writeable
+ return (((DynaBean) bean).getDynaClass().getDynaProperty(name) != null);
+ } else {
+ try {
+ PropertyDescriptor desc =
+ getPropertyDescriptor(bean, name);
+ if (desc != null) {
+ Method writeMethod = desc.getWriteMethod();
+ if ((writeMethod == null) &&
+ (desc instanceof IndexedPropertyDescriptor)) {
+ writeMethod = ((IndexedPropertyDescriptor) desc).getIndexedWriteMethod();
+ }
+ return (writeMethod != null);
+ } else {
+ return (false);
+ }
+ } catch (IllegalAccessException e) {
+ return (false);
+ } catch (InvocationTargetException e) {
+ return (false);
+ } catch (NoSuchMethodException e) {
+ return (false);
+ }
+ }
+
+ }
+
+
+ /**
+ * Set the value of the specified indexed property of the specified
+ * bean, with no type conversions. The zero-relative index of the
+ * required value must be included (in square brackets) as a suffix to
+ * the property name, or <code>IllegalArgumentException</code> will be
+ * thrown. In addition to supporting the JavaBeans specification, this
+ * method has been extended to support <code>List</code> objects as well.
+ *
+ * @param bean Bean whose property is to be modified
+ * @param name <code>propertyname[index]</code> of the property value
+ * to be modified
+ * @param value Value to which the specified property element
+ * should be set
+ *
+ * @exception ArrayIndexOutOfBoundsException if the specified index
+ * is outside the valid range for the underlying array
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public void setIndexedProperty(Object bean, String name,
+ Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Identify the index of the requested individual property
+ int delim = name.indexOf(PropertyUtils.INDEXED_DELIM);
+ int delim2 = name.indexOf(PropertyUtils.INDEXED_DELIM2);
+ if ((delim < 0) || (delim2 <= delim)) {
+ throw new IllegalArgumentException("Invalid indexed property '" +
+ name + "'");
+ }
+ int index = -1;
+ try {
+ String subscript = name.substring(delim + 1, delim2);
+ index = Integer.parseInt(subscript);
+ } catch (NumberFormatException e) {
+ throw new IllegalArgumentException("Invalid indexed property '" +
+ name + "'");
+ }
+ name = name.substring(0, delim);
+
+ // Set the specified indexed property value
+ setIndexedProperty(bean, name, index, value);
+
+ }
+
+
+ /**
+ * Set the value of the specified indexed property of the specified
+ * bean, with no type conversions. In addition to supporting the JavaBeans
+ * specification, this method has been extended to support
+ * <code>List</code> objects as well.
+ *
+ * @param bean Bean whose property is to be set
+ * @param name Simple property name of the property value to be set
+ * @param index Index of the property value to be set
+ * @param value Value to which the indexed property element is to be set
+ *
+ * @exception ArrayIndexOutOfBoundsException if the specified index
+ * is outside the valid range for the underlying array
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public void setIndexedProperty(Object bean, String name,
+ int index, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Handle DynaBean instances specially
+ if (bean instanceof DynaBean) {
+ DynaProperty descriptor =
+ ((DynaBean) bean).getDynaClass().getDynaProperty(name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+ ((DynaBean) bean).set(name, index, value);
+ return;
+ }
+
+ // Retrieve the property descriptor for the specified property
+ PropertyDescriptor descriptor =
+ getPropertyDescriptor(bean, name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+
+ // Call the indexed setter method if there is one
+ if (descriptor instanceof IndexedPropertyDescriptor) {
+ Method writeMethod = ((IndexedPropertyDescriptor) descriptor).
+ getIndexedWriteMethod();
+ if (writeMethod != null) {
+ Object subscript[] = new Object[2];
+ subscript[0] = new Integer(index);
+ subscript[1] = value;
+ try {
+ if (log.isTraceEnabled()) {
+ String valueClassName =
+ value == null ? "<null>"
+ : value.getClass().getName();
+ log.trace("setSimpleProperty: Invoking method "
+ + writeMethod +" with index=" + index
+ + ", value=" + value
+ + " (class " + valueClassName+ ")");
+ }
+ invokeMethod(writeMethod, bean, subscript);
+ } catch (InvocationTargetException e) {
+ if (e.getTargetException() instanceof
+ ArrayIndexOutOfBoundsException) {
+ throw (ArrayIndexOutOfBoundsException)
+ e.getTargetException();
+ } else {
+ throw e;
+ }
+ }
+ return;
+ }
+ }
+
+ // Otherwise, the underlying property must be an array or a list
+ Method readMethod = descriptor.getReadMethod();
+ if (readMethod == null) {
+ throw new NoSuchMethodException("Property '" + name +
+ "' has no getter method");
+ }
+
+ // Call the property getter to get the array or list
+ Object array = invokeMethod(readMethod, bean, new Object[0]);
+ if (!array.getClass().isArray()) {
+ if (array instanceof List) {
+ // Modify the specified value in the List
+ ((List) array).set(index, value);
+ } else {
+ throw new IllegalArgumentException("Property '" + name +
+ "' is not indexed");
+ }
+ } else {
+ // Modify the specified value in the array
+ Array.set(array, index, value);
+ }
+
+ }
+
+
+ /**
+ * Set the value of the specified mapped property of the
+ * specified bean, with no type conversions. The key of the
+ * value to set must be included (in brackets) as a suffix to
+ * the property name, or <code>IllegalArgumentException</code> will be
+ * thrown.
+ *
+ * @param bean Bean whose property is to be set
+ * @param name <code>propertyname(key)</code> of the property value
+ * to be set
+ * @param value The property value to be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public void setMappedProperty(Object bean, String name,
+ Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Identify the index of the requested individual property
+ int delim = name.indexOf(PropertyUtils.MAPPED_DELIM);
+ int delim2 = name.indexOf(PropertyUtils.MAPPED_DELIM2);
+ if ((delim < 0) || (delim2 <= delim)) {
+ throw new IllegalArgumentException
+ ("Invalid mapped property '" + name + "'");
+ }
+
+ // Isolate the name and the key
+ String key = name.substring(delim + 1, delim2);
+ name = name.substring(0, delim);
+
+ // Request the specified indexed property value
+ setMappedProperty(bean, name, key, value);
+
+ }
+
+
+ /**
+ * Set the value of the specified mapped property of the specified
+ * bean, with no type conversions.
+ *
+ * @param bean Bean whose property is to be set
+ * @param name Mapped property name of the property value to be set
+ * @param key Key of the property value to be set
+ * @param value The property value to be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public void setMappedProperty(Object bean, String name,
+ String key, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+ if (key == null) {
+ throw new IllegalArgumentException("No key specified");
+ }
+
+ // Handle DynaBean instances specially
+ if (bean instanceof DynaBean) {
+ DynaProperty descriptor =
+ ((DynaBean) bean).getDynaClass().getDynaProperty(name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+ ((DynaBean) bean).set(name, key, value);
+ return;
+ }
+
+ // Retrieve the property descriptor for the specified property
+ PropertyDescriptor descriptor =
+ getPropertyDescriptor(bean, name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+
+ if (descriptor instanceof MappedPropertyDescriptor) {
+ // Call the keyed setter method if there is one
+ Method mappedWriteMethod =
+ ((MappedPropertyDescriptor) descriptor).
+ getMappedWriteMethod();
+ if (mappedWriteMethod != null) {
+ Object params[] = new Object[2];
+ params[0] = key;
+ params[1] = value;
+ if (log.isTraceEnabled()) {
+ String valueClassName =
+ value == null ? "<null>" : value.getClass().getName();
+ log.trace("setSimpleProperty: Invoking method "
+ + mappedWriteMethod + " with key=" + key
+ + ", value=" + value
+ + " (class " + valueClassName +")");
+ }
+ invokeMethod(mappedWriteMethod, bean, params);
+ } else {
+ throw new NoSuchMethodException
+ ("Property '" + name +
+ "' has no mapped setter method");
+ }
+ } else {
+ /* means that the result has to be retrieved from a map */
+ Method readMethod = descriptor.getReadMethod();
+ if (readMethod != null) {
+ Object invokeResult = invokeMethod(readMethod, bean, new Object[0]);
+ /* test and fetch from the map */
+ if (invokeResult instanceof java.util.Map) {
+ ((java.util.Map)invokeResult).put(key, value);
+ }
+ } else {
+ throw new NoSuchMethodException("Property '" + name +
+ "' has no mapped getter method");
+ }
+ }
+
+ }
+
+
+ /**
+ * Set the value of the (possibly nested) property of the specified
+ * name, for the specified bean, with no type conversions.
+ *
+ * @param bean Bean whose property is to be modified
+ * @param name Possibly nested name of the property to be modified
+ * @param value Value to which the property is to be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception IllegalArgumentException if a nested reference to a
+ * property returns null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public void setNestedProperty(Object bean,
+ String name, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ int indexOfINDEXED_DELIM = -1;
+ int indexOfMAPPED_DELIM = -1;
+ while (true) {
+ int delim = name.indexOf(PropertyUtils.NESTED_DELIM);
+ if (delim < 0) {
+ break;
+ }
+ String next = name.substring(0, delim);
+ indexOfINDEXED_DELIM = next.indexOf(PropertyUtils.INDEXED_DELIM);
+ indexOfMAPPED_DELIM = next.indexOf(PropertyUtils.MAPPED_DELIM);
+ if (bean instanceof Map) {
+ bean = ((Map) bean).get(next);
+ } else if (indexOfMAPPED_DELIM >= 0) {
+ bean = getMappedProperty(bean, next);
+ } else if (indexOfINDEXED_DELIM >= 0) {
+ bean = getIndexedProperty(bean, next);
+ } else {
+ bean = getSimpleProperty(bean, next);
+ }
+ if (bean == null) {
+ throw new IllegalArgumentException
+ ("Null property value for '" +
+ name.substring(0, delim) + "'");
+ }
+ name = name.substring(delim + 1);
+ }
+
+ indexOfINDEXED_DELIM = name.indexOf(PropertyUtils.INDEXED_DELIM);
+ indexOfMAPPED_DELIM = name.indexOf(PropertyUtils.MAPPED_DELIM);
+
+ if (bean instanceof Map) {
+ // check to see if the class has a standard property
+ PropertyDescriptor descriptor =
+ getPropertyDescriptor(bean, name);
+ if (descriptor == null) {
+ // no - then put the value into the map
+ ((Map) bean).put(name, value);
+ } else {
+ // yes - use that instead
+ setSimpleProperty(bean, name, value);
+ }
+ } else if (indexOfMAPPED_DELIM >= 0) {
+ setMappedProperty(bean, name, value);
+ } else if (indexOfINDEXED_DELIM >= 0) {
+ setIndexedProperty(bean, name, value);
+ } else {
+ setSimpleProperty(bean, name, value);
+ }
+
+ }
+
+
+ /**
+ * Set the value of the specified property of the specified bean,
+ * no matter which property reference format is used, with no
+ * type conversions.
+ *
+ * @param bean Bean whose property is to be modified
+ * @param name Possibly indexed and/or nested name of the property
+ * to be modified
+ * @param value Value to which this property is to be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public void setProperty(Object bean, String name, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ setNestedProperty(bean, name, value);
+
+ }
+
+
+ /**
+ * Set the value of the specified simple property of the specified bean,
+ * with no type conversions.
+ *
+ * @param bean Bean whose property is to be modified
+ * @param name Name of the property to be modified
+ * @param value Value to which the property should be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if <code>bean</code> or
+ * <code>name</code> is null
+ * @exception IllegalArgumentException if the property name is
+ * nested or indexed
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public void setSimpleProperty(Object bean,
+ String name, Object value)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ if (bean == null) {
+ throw new IllegalArgumentException("No bean specified");
+ }
+ if (name == null) {
+ throw new IllegalArgumentException("No name specified");
+ }
+
+ // Validate the syntax of the property name
+ if (name.indexOf(PropertyUtils.NESTED_DELIM) >= 0) {
+ throw new IllegalArgumentException
+ ("Nested property names are not allowed");
+ } else if (name.indexOf(PropertyUtils.INDEXED_DELIM) >= 0) {
+ throw new IllegalArgumentException
+ ("Indexed property names are not allowed");
+ } else if (name.indexOf(PropertyUtils.MAPPED_DELIM) >= 0) {
+ throw new IllegalArgumentException
+ ("Mapped property names are not allowed");
+ }
+
+ // Handle DynaBean instances specially
+ if (bean instanceof DynaBean) {
+ DynaProperty descriptor =
+ ((DynaBean) bean).getDynaClass().getDynaProperty(name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+ ((DynaBean) bean).set(name, value);
+ return;
+ }
+
+ // Retrieve the property setter method for the specified property
+ PropertyDescriptor descriptor =
+ getPropertyDescriptor(bean, name);
+ if (descriptor == null) {
+ throw new NoSuchMethodException("Unknown property '" +
+ name + "'");
+ }
+ Method writeMethod = getWriteMethod(descriptor);
+ if (writeMethod == null) {
+ throw new NoSuchMethodException("Property '" + name +
+ "' has no setter method");
+ }
+
+ // Call the property setter method
+ Object values[] = new Object[1];
+ values[0] = value;
+ if (log.isTraceEnabled()) {
+ String valueClassName =
+ value == null ? "<null>" : value.getClass().getName();
+ log.trace("setSimpleProperty: Invoking method " + writeMethod
+ + " with value " + value + " (class " + valueClassName + ")");
+ }
+ invokeMethod(writeMethod, bean, values);
+
+ }
+
+ /** This just catches and wraps IllegalArgumentException. */
+ private Object invokeMethod(
+ Method method,
+ Object bean,
+ Object[] values)
+ throws
+ IllegalAccessException,
+ InvocationTargetException {
+ try {
+
+ return method.invoke(bean, values);
+
+ } catch (IllegalArgumentException e) {
+
+ log.error("Method invocation failed.", e);
+ throw new IllegalArgumentException(
+ "Cannot invoke " + method.getDeclaringClass().getName() + "."
+ + method.getName() + " - " + e.getMessage());
+
+ }
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/ResultSetDynaClass.java b/trunk/src/java/org/apache/commons/beanutils/ResultSetDynaClass.java
new file mode 100644
index 0000000..4f01797
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/ResultSetDynaClass.java
@@ -0,0 +1,200 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.sql.ResultSet;
+import java.sql.SQLException;
+import java.util.HashMap;
+import java.util.Iterator;
+
+
+/**
+ * <p>Implementation of <code>DynaClass</code> for DynaBeans that wrap the
+ * <code>java.sql.Row</code> objects of a <code>java.sql.ResultSet</code>.
+ * The normal usage pattern is something like:</p>
+ * <pre>
+ * ResultSet rs = ...;
+ * ResultSetDynaClass rsdc = new ResultSetDynaClass(rs);
+ * Iterator rows = rsdc.iterator();
+ * while (rows.hasNext()) {
+ * DynaBean row = (DynaBean) rows.next();
+ * ... process this row ...
+ * }
+ * rs.close();
+ * </pre>
+ *
+ * <p>Each column in the result set will be represented as a DynaBean
+ * property of the corresponding name (optionally forced to lower case
+ * for portability).</p>
+ *
+ * <p><strong>WARNING</strong> - Any {@link DynaBean} instance returned by
+ * this class, or from the <code>Iterator</code> returned by the
+ * <code>iterator()</code> method, is directly linked to the row that the
+ * underlying result set is currently positioned at. This has the following
+ * implications:</p>
+ * <ul>
+ * <li>Once you retrieve a different {@link DynaBean} instance, you should
+ * no longer use any previous instance.</li>
+ * <li>Changing the position of the underlying result set will change the
+ * data that the {@link DynaBean} references.</li>
+ * <li>Once the underlying result set is closed, the {@link DynaBean}
+ * instance may no longer be used.</li>
+ * </ul>
+ *
+ * <p>Any database data that you wish to utilize outside the context of the
+ * current row of an open result set must be copied. For example, you could
+ * use the following code to create standalone copies of the information in
+ * a result set:</p>
+ * <pre>
+ * ArrayList results = new ArrayList(); // To hold copied list
+ * ResultSetDynaClass rsdc = ...;
+ * DynaProperty properties[] = rsdc.getDynaProperties();
+ * BasicDynaClass bdc =
+ * new BasicDynaClass("foo", BasicDynaBean.class,
+ * rsdc.getDynaProperties());
+ * Iterator rows = rsdc.iterator();
+ * while (rows.hasNext()) {
+ * DynaBean oldRow = (DynaBean) rows.next();
+ * DynaBean newRow = bdc.newInstance();
+ * PropertyUtils.copyProperties(newRow, oldRow);
+ * results.add(newRow);
+ * }
+ * </pre>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.15 $ $Date: 2004/02/28 13:18:33 $
+ */
+
+public class ResultSetDynaClass extends JDBCDynaClass implements DynaClass {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * <p>Construct a new ResultSetDynaClass for the specified
+ * <code>ResultSet</code>. The property names corresponding
+ * to column names in the result set will be lower cased.</p>
+ *
+ * @param resultSet The result set to be wrapped
+ *
+ * @exception NullPointerException if <code>resultSet</code>
+ * is <code>null</code>
+ * @exception SQLException if the metadata for this result set
+ * cannot be introspected
+ */
+ public ResultSetDynaClass(ResultSet resultSet) throws SQLException {
+
+ this(resultSet, true);
+
+ }
+
+
+ /**
+ * <p>Construct a new ResultSetDynaClass for the specified
+ * <code>ResultSet</code>. The property names corresponding
+ * to the column names in the result set will be lower cased or not,
+ * depending on the specified <code>lowerCase</code> value.</p>
+ *
+ * <p><strong>WARNING</strong> - If you specify <code>false</code>
+ * for <code>lowerCase</code>, the returned property names will
+ * exactly match the column names returned by your JDBC driver.
+ * Because different drivers might return column names in different
+ * cases, the property names seen by your application will vary
+ * depending on which JDBC driver you are using.</p>
+ *
+ * @param resultSet The result set to be wrapped
+ * @param lowerCase Should property names be lower cased?
+ *
+ * @exception NullPointerException if <code>resultSet</code>
+ * is <code>null</code>
+ * @exception SQLException if the metadata for this result set
+ * cannot be introspected
+ */
+ public ResultSetDynaClass(ResultSet resultSet, boolean lowerCase)
+ throws SQLException {
+
+ if (resultSet == null) {
+ throw new NullPointerException();
+ }
+ this.resultSet = resultSet;
+ this.lowerCase = lowerCase;
+ introspect(resultSet);
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * <p>The <code>ResultSet</code> we are wrapping.</p>
+ */
+ protected ResultSet resultSet = null;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * <p>Return an <code>Iterator</code> of {@link DynaBean} instances for
+ * each row of the wrapped <code>ResultSet</code>, in "forward" order.
+ * Unless the underlying result set supports scrolling, this method
+ * should be called only once.</p>
+ */
+ public Iterator iterator() {
+
+ return (new ResultSetIterator(this));
+
+ }
+
+
+ // -------------------------------------------------------- Package Methods
+
+
+ /**
+ * <p>Return the result set we are wrapping.</p>
+ */
+ ResultSet getResultSet() {
+
+ return (this.resultSet);
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+ /**
+ * <p>Loads the class of the given name which by default uses the class loader used
+ * to load this library.
+ * Dervations of this class could implement alternative class loading policies such as
+ * using custom ClassLoader or using the Threads's context class loader etc.
+ * </p>
+ */
+ protected Class loadClass(String className) throws SQLException {
+
+ try {
+ return getClass().getClassLoader().loadClass(className);
+ }
+ catch (Exception e) {
+ throw new SQLException("Cannot load column class '" +
+ className + "': " + e);
+ }
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/ResultSetIterator.java b/trunk/src/java/org/apache/commons/beanutils/ResultSetIterator.java
new file mode 100644
index 0000000..e9d5163
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/ResultSetIterator.java
@@ -0,0 +1,344 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.sql.ResultSet;
+import java.sql.SQLException;
+import java.util.Iterator;
+import java.util.NoSuchElementException;
+
+
+/**
+ * <p>Implementation of <code>java.util.Iterator</code> returned by the
+ * <code>iterator()</code> method of {@link ResultSetDynaClass}. Each
+ * object returned by this iterator will be a {@link DynaBean} that
+ * represents a single row from the result set being wrapped.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.5 $ $Date: 2004/02/28 13:18:34 $
+ */
+
+public class ResultSetIterator implements DynaBean, Iterator {
+
+
+ // ------------------------------------------------------------ Constructor
+
+
+ /**
+ * <p>Construct an <code>Iterator</code> for the result set being wrapped
+ * by the specified {@link ResultSetDynaClass}.</p>
+ *
+ * @param dynaClass The {@link ResultSetDynaClass} wrapping the
+ * result set we will iterate over
+ */
+ ResultSetIterator(ResultSetDynaClass dynaClass) {
+
+ this.dynaClass = dynaClass;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+
+ /**
+ * <p>Flag indicating whether the result set is currently positioned at a
+ * row for which we have not yet returned an element in the iteration.</p>
+ */
+ protected boolean current = false;
+
+
+ /**
+ * <p>The {@link ResultSetDynaClass} we are associated with.</p>
+ */
+ protected ResultSetDynaClass dynaClass = null;
+
+
+ /**
+ * <p>Flag indicating whether the result set has indicated that there are
+ * no further rows.</p>
+ */
+ protected boolean eof = false;
+
+
+ // ------------------------------------------------------- DynaBean Methods
+
+
+ /**
+ * Does the specified mapped property contain a value for the specified
+ * key value?
+ *
+ * @param name Name of the property to check
+ * @param key Name of the key to check
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public boolean contains(String name, String key) {
+
+ throw new UnsupportedOperationException
+ ("FIXME - mapped properties not currently supported");
+
+ }
+
+
+ /**
+ * Return the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public Object get(String name) {
+
+ if (dynaClass.getDynaProperty(name) == null) {
+ throw new IllegalArgumentException(name);
+ }
+ try {
+ return (dynaClass.getResultSet().getObject(name));
+ } catch (SQLException e) {
+ throw new RuntimeException
+ ("get(" + name + "): SQLException: " + e);
+ }
+
+ }
+
+
+ /**
+ * Return the value of an indexed property with the specified name.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param index Index of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ * @exception NullPointerException if no array or List has been
+ * initialized for this property
+ */
+ public Object get(String name, int index) {
+
+ throw new UnsupportedOperationException
+ ("FIXME - indexed properties not currently supported");
+
+ }
+
+
+ /**
+ * Return the value of a mapped property with the specified name,
+ * or <code>null</code> if there is no value for the specified key.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param key Key of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public Object get(String name, String key) {
+
+ throw new UnsupportedOperationException
+ ("FIXME - mapped properties not currently supported");
+
+ }
+
+
+ /**
+ * Return the <code>DynaClass</code> instance that describes the set of
+ * properties available for this DynaBean.
+ */
+ public DynaClass getDynaClass() {
+
+ return (this.dynaClass);
+
+ }
+
+
+ /**
+ * Remove any existing value for the specified key on the
+ * specified mapped property.
+ *
+ * @param name Name of the property for which a value is to
+ * be removed
+ * @param key Key of the value to be removed
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public void remove(String name, String key) {
+
+ throw new UnsupportedOperationException
+ ("FIXME - mapped operations not currently supported");
+
+ }
+
+
+ /**
+ * Set the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception NullPointerException if an attempt is made to set a
+ * primitive property to null
+ */
+ public void set(String name, Object value) {
+
+ if (dynaClass.getDynaProperty(name) == null) {
+ throw new IllegalArgumentException(name);
+ }
+ try {
+ dynaClass.getResultSet().updateObject(name, value);
+ } catch (SQLException e) {
+ throw new RuntimeException
+ ("set(" + name + "): SQLException: " + e);
+ }
+
+ }
+
+
+ /**
+ * Set the value of an indexed property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param index Index of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ */
+ public void set(String name, int index, Object value) {
+
+ throw new UnsupportedOperationException
+ ("FIXME - indexed properties not currently supported");
+
+ }
+
+
+ /**
+ * Set the value of a mapped property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param key Key of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public void set(String name, String key, Object value) {
+
+ throw new UnsupportedOperationException
+ ("FIXME - mapped properties not currently supported");
+
+ }
+
+
+ // ------------------------------------------------------- Iterator Methods
+
+
+ /**
+ * <p>Return <code>true</code> if the iteration has more elements.</p>
+ */
+ public boolean hasNext() {
+
+ try {
+ advance();
+ return (!eof);
+ } catch (SQLException e) {
+ throw new RuntimeException("hasNext(): SQLException: " + e);
+ }
+
+ }
+
+
+ /**
+ * <p>Return the next element in the iteration.</p>
+ */
+ public Object next() {
+
+ try {
+ advance();
+ if (eof) {
+ throw new NoSuchElementException();
+ }
+ current = false;
+ return (this);
+ } catch (SQLException e) {
+ throw new RuntimeException("next(): SQLException: " + e);
+ }
+
+ }
+
+
+ /**
+ * <p>Remove the current element from the iteration. This method is
+ * not supported.</p>
+ */
+ public void remove() {
+
+ throw new UnsupportedOperationException("remove()");
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * <p>Advance the result set to the next row, if there is not a current
+ * row (and if we are not already at eof).</p>
+ *
+ * @exception SQLException if the result set throws an exception
+ */
+ protected void advance() throws SQLException {
+
+ if (!current && !eof) {
+ if (dynaClass.getResultSet().next()) {
+ current = true;
+ eof = false;
+ } else {
+ current = false;
+ eof = true;
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/RowSetDynaClass.java b/trunk/src/java/org/apache/commons/beanutils/RowSetDynaClass.java
new file mode 100644
index 0000000..23f45da
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/RowSetDynaClass.java
@@ -0,0 +1,247 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.io.Serializable;
+import java.sql.ResultSet;
+import java.sql.SQLException;
+import java.util.ArrayList;
+import java.util.List;
+
+
+/**
+ * <p>Implementation of {@link DynaClass} that creates an in-memory collection
+ * of {@link DynaBean}s representing the results of an SQL query. Once the
+ * {@link DynaClass} instance has been created, the JDBC <code>ResultSet</code>
+ * and <code>Statement</code> on which it is based can be closed, and the
+ * underlying <code>Connection</code> can be returned to its connection pool
+ * (if you are using one).</p>
+ *
+ * <p>The normal usage pattern is something like:</p>
+ * <pre>
+ * Connection conn = ...; // Acquire connection from pool
+ * Statement stmt = conn.createStatement();
+ * ResultSet rs = stmt.executeQuery("SELECT ...");
+ * RowSetDynaClass rsdc = new RowSetDynaClass(rs);
+ * rs.close();
+ * stmt.close();
+ * ...; // Return connection to pool
+ * List rows = rsdc.getRows();
+ * ...; // Process the rows as desired
+ * </pre>
+ *
+ * <p>Each column in the result set will be represented as a {@link DynaBean}
+ * property of the corresponding name (optionally forced to lower case
+ * for portability). There will be one {@link DynaBean} in the
+ * <code>List</code> returned by <code>getRows()</code> for each
+ * row in the original <code>ResultSet</code>.</p>
+ *
+ * <p>In general, instances of {@link RowSetDynaClass} can be serialized
+ * and deserialized, which will automatically include the list of
+ * {@link DynaBean}s representing the data content. The only exception
+ * to this rule would be when the underlying property values that were
+ * copied from the <code>ResultSet</code> originally cannot themselves
+ * be serialized. Therefore, a {@link RowSetDynaClass} makes a very
+ * convenient mechanism for transporting data sets to remote Java-based
+ * application components.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.9 $ $Date: 2004/02/28 13:18:34 $
+ */
+
+public class RowSetDynaClass extends JDBCDynaClass implements DynaClass, Serializable {
+
+
+ // ----------------------------------------------------- Instance variables
+
+ /**
+ * <p>Limits the size of the returned list. The call to
+ * <code>getRows()</code> will return at most limit number of rows.
+ * If less than or equal to 0, does not limit the size of the result.
+ */
+ protected int limit = -1;
+
+ /**
+ * <p>The list of {@link DynaBean}s representing the contents of
+ * the original <code>ResultSet</code> on which this
+ * {@link RowSetDynaClass} was based.</p>
+ */
+ protected List rows = new ArrayList();
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * <p>Construct a new {@link RowSetDynaClass} for the specified
+ * <code>ResultSet</code>. The property names corresponding
+ * to column names in the result set will be lower cased.</p>
+ *
+ * @param resultSet The result set to be wrapped
+ *
+ * @exception NullPointerException if <code>resultSet</code>
+ * is <code>null</code>
+ * @exception SQLException if the metadata for this result set
+ * cannot be introspected
+ */
+ public RowSetDynaClass(ResultSet resultSet) throws SQLException {
+
+ this(resultSet, true, -1);
+
+ }
+
+ /**
+ * <p>Construct a new {@link RowSetDynaClass} for the specified
+ * <code>ResultSet</code>. The property names corresponding
+ * to column names in the result set will be lower cased.</p>
+ *
+ * If <code>limit</code> is not less than 0, max <code>limit</code>
+ * number of rows will be copied into the list.
+ *
+ * @param resultSet The result set to be wrapped
+ * @param limit The maximum for the size of the result.
+ *
+ * @exception NullPointerException if <code>resultSet</code>
+ * is <code>null</code>
+ * @exception SQLException if the metadata for this result set
+ * cannot be introspected
+ */
+ public RowSetDynaClass(ResultSet resultSet, int limit) throws SQLException {
+
+ this(resultSet, true, limit);
+
+ }
+
+
+ /**
+ * <p>Construct a new {@link RowSetDynaClass} for the specified
+ * <code>ResultSet</code>. The property names corresponding
+ * to the column names in the result set will be lower cased or not,
+ * depending on the specified <code>lowerCase</code> value.</p>
+ *
+ * If <code>limit</code> is not less than 0, max <code>limit</code>
+ * number of rows will be copied into the resultset.
+ *
+ *
+ * @param resultSet The result set to be wrapped
+ * @param lowerCase Should property names be lower cased?
+ *
+ * @exception NullPointerException if <code>resultSet</code>
+ * is <code>null</code>
+ * @exception SQLException if the metadata for this result set
+ * cannot be introspected
+ */
+ public RowSetDynaClass(ResultSet resultSet, boolean lowerCase)
+ throws SQLException {
+ this(resultSet, lowerCase, -1);
+
+ }
+
+ /**
+ * <p>Construct a new {@link RowSetDynaClass} for the specified
+ * <code>ResultSet</code>. The property names corresponding
+ * to the column names in the result set will be lower cased or not,
+ * depending on the specified <code>lowerCase</code> value.</p>
+ *
+ * <p><strong>WARNING</strong> - If you specify <code>false</code>
+ * for <code>lowerCase</code>, the returned property names will
+ * exactly match the column names returned by your JDBC driver.
+ * Because different drivers might return column names in different
+ * cases, the property names seen by your application will vary
+ * depending on which JDBC driver you are using.</p>
+ *
+ * @param resultSet The result set to be wrapped
+ * @param lowerCase Should property names be lower cased?
+ *
+ * @exception NullPointerException if <code>resultSet</code>
+ * is <code>null</code>
+ * @exception SQLException if the metadata for this result set
+ * cannot be introspected
+ */
+ public RowSetDynaClass(ResultSet resultSet, boolean lowerCase, int limit)
+ throws SQLException {
+
+ if (resultSet == null) {
+ throw new NullPointerException();
+ }
+ this.lowerCase = lowerCase;
+ this.limit = limit;
+ introspect(resultSet);
+ copy(resultSet);
+
+ }
+
+ /**
+ * <p>Return a <code>List</code> containing the {@link DynaBean}s that
+ * represent the contents of each <code>Row</code> from the
+ * <code>ResultSet</code> that was the basis of this
+ * {@link RowSetDynaClass} instance. These {@link DynaBean}s are
+ * disconnected from the database itself, so there is no problem with
+ * modifying the contents of the list, or the values of the properties
+ * of these {@link DynaBean}s. However, it is the application's
+ * responsibility to persist any such changes back to the database,
+ * if it so desires.</p>
+ */
+ public List getRows() {
+
+ return (this.rows);
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * <p>Copy the column values for each row in the specified
+ * <code>ResultSet</code> into a newly created {@link DynaBean}, and add
+ * this bean to the list of {@link DynaBean}s that will later by
+ * returned by a call to <code>getRows()</code>.</p>
+ *
+ * @param resultSet The <code>ResultSet</code> whose data is to be
+ * copied
+ *
+ * @exception SQLException if an error is encountered copying the data
+ */
+ protected void copy(ResultSet resultSet) throws SQLException {
+
+ int cnt = 0;
+ while (resultSet.next() && (limit < 0 || cnt++ < limit) ) {
+ DynaBean bean = createDynaBean();
+ for (int i = 0; i < properties.length; i++) {
+ String name = properties[i].getName();
+ bean.set(name, resultSet.getObject(name));
+ }
+ rows.add(bean);
+ }
+
+ }
+
+
+ /**
+ * <p>Create and return a new {@link DynaBean} instance to be used for
+ * representing a row in the underlying result set.</p>
+ */
+ protected DynaBean createDynaBean() {
+
+ return (new BasicDynaBean(this));
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/WrapDynaBean.java b/trunk/src/java/org/apache/commons/beanutils/WrapDynaBean.java
new file mode 100644
index 0000000..bec7dc2
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/WrapDynaBean.java
@@ -0,0 +1,333 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+
+
+
+/**
+ * <p>Implementation of <code>DynaBean</code> that wraps a standard JavaBean
+ * instance, so that DynaBean APIs can be used to access its properties.</p>
+ *
+ * <p>
+ * The most common use cases for this class involve wrapping an existing java bean.
+ * (This makes it different from the typical use cases for other <code>DynaBean</code>'s.)
+ * For example:
+ * </p>
+ * <code><pre>
+ * Object aJavaBean = ...;
+ * ...
+ * DynaBean db = new WrapDynaBean(aJavaBean);
+ * ...
+ * </pre></code>
+ *
+ * <p><strong>IMPLEMENTATION NOTE</strong> - This implementation does not
+ * support the <code>contains()</code> and <code>remove()</code> methods.</p>
+ *
+ * @author Craig McClanahan
+ * @version $Revision: 1.9 $ $Date: 2004/02/28 13:18:34 $
+ */
+
+public class WrapDynaBean implements DynaBean {
+
+
+ // ---------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new <code>DynaBean</code> associated with the specified
+ * JavaBean instance.
+ *
+ * @param instance JavaBean instance to be wrapped
+ */
+ public WrapDynaBean(Object instance) {
+
+ super();
+ this.instance = instance;
+ this.dynaClass = WrapDynaClass.createDynaClass(instance.getClass());
+
+ }
+
+
+ // ---------------------------------------------------- Instance Variables
+
+
+ /**
+ * The <code>DynaClass</code> "base class" that this DynaBean
+ * is associated with.
+ */
+ protected WrapDynaClass dynaClass = null;
+
+
+ /**
+ * The JavaBean instance wrapped by this WrapDynaBean.
+ */
+ protected Object instance = null;
+
+
+ // ------------------------------------------------------ DynaBean Methods
+
+
+ /**
+ * Does the specified mapped property contain a value for the specified
+ * key value?
+ *
+ * @param name Name of the property to check
+ * @param key Name of the key to check
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public boolean contains(String name, String key) {
+
+ throw new UnsupportedOperationException
+ ("WrapDynaBean does not support contains()");
+
+ }
+
+
+ /**
+ * Return the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public Object get(String name) {
+
+ Object value = null;
+ try {
+ value = PropertyUtils.getSimpleProperty(instance, name);
+ } catch (Throwable t) {
+ throw new IllegalArgumentException
+ ("Property '" + name + "' has no read method");
+ }
+ return (value);
+
+ }
+
+
+ /**
+ * Return the value of an indexed property with the specified name.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param index Index of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ * @exception NullPointerException if no array or List has been
+ * initialized for this property
+ */
+ public Object get(String name, int index) {
+
+ Object value = null;
+ try {
+ value = PropertyUtils.getIndexedProperty(instance, name, index);
+ } catch (IndexOutOfBoundsException e) {
+ throw e;
+ } catch (Throwable t) {
+ throw new IllegalArgumentException
+ ("Property '" + name + "' has no indexed read method");
+ }
+ return (value);
+
+ }
+
+
+ /**
+ * Return the value of a mapped property with the specified name,
+ * or <code>null</code> if there is no value for the specified key.
+ *
+ * @param name Name of the property whose value is to be retrieved
+ * @param key Key of the value to be retrieved
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public Object get(String name, String key) {
+
+ Object value = null;
+ try {
+ value = PropertyUtils.getMappedProperty(instance, name, key);
+ } catch (Throwable t) {
+ throw new IllegalArgumentException
+ ("Property '" + name + "' has no mapped read method");
+ }
+ return (value);
+
+ }
+
+
+ /**
+ * Return the <code>DynaClass</code> instance that describes the set of
+ * properties available for this DynaBean.
+ */
+ public DynaClass getDynaClass() {
+
+ return (this.dynaClass);
+
+ }
+
+
+ /**
+ * Remove any existing value for the specified key on the
+ * specified mapped property.
+ *
+ * @param name Name of the property for which a value is to
+ * be removed
+ * @param key Key of the value to be removed
+ *
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ */
+ public void remove(String name, String key) {
+
+
+ throw new UnsupportedOperationException
+ ("WrapDynaBean does not support remove()");
+
+ }
+
+
+ /**
+ * Set the value of a simple property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception NullPointerException if an attempt is made to set a
+ * primitive property to null
+ */
+ public void set(String name, Object value) {
+
+ try {
+ PropertyUtils.setSimpleProperty(instance, name, value);
+ } catch (Throwable t) {
+ throw new IllegalArgumentException
+ ("Property '" + name + "' has no write method");
+ }
+
+ }
+
+
+ /**
+ * Set the value of an indexed property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param index Index of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not indexed
+ * @exception IndexOutOfBoundsException if the specified index
+ * is outside the range of the underlying property
+ */
+ public void set(String name, int index, Object value) {
+
+ try {
+ PropertyUtils.setIndexedProperty(instance, name, index, value);
+ } catch (IndexOutOfBoundsException e) {
+ throw e;
+ } catch (Throwable t) {
+ throw new IllegalArgumentException
+ ("Property '" + name + "' has no indexed write method");
+ }
+
+ }
+
+
+ /**
+ * Set the value of a mapped property with the specified name.
+ *
+ * @param name Name of the property whose value is to be set
+ * @param key Key of the property to be set
+ * @param value Value to which this property is to be set
+ *
+ * @exception ConversionException if the specified value cannot be
+ * converted to the type required for this property
+ * @exception IllegalArgumentException if there is no property
+ * of the specified name
+ * @exception IllegalArgumentException if the specified property
+ * exists, but is not mapped
+ */
+ public void set(String name, String key, Object value) {
+
+ try {
+ PropertyUtils.setMappedProperty(instance, name, key, value);
+ } catch (Throwable t) {
+ throw new IllegalArgumentException
+ ("Property '" + name + "' has no mapped write method");
+ }
+
+ }
+
+ /**
+ * Gets the bean instance wrapped by this DynaBean.
+ * For most common use cases,
+ * this object should already be known
+ * and this method safely be ignored.
+ * But some creators of frameworks using <code>DynaBean</code>'s may
+ * find this useful.
+ *
+ * @return the java bean Object wrapped by this <code>DynaBean</code>
+ */
+ public Object getInstance() {
+ return instance;
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * Return the property descriptor for the specified property name.
+ *
+ * @param name Name of the property for which to retrieve the descriptor
+ *
+ * @exception IllegalArgumentException if this is not a valid property
+ * name for our DynaClass
+ */
+ protected DynaProperty getDynaProperty(String name) {
+
+ DynaProperty descriptor = getDynaClass().getDynaProperty(name);
+ if (descriptor == null) {
+ throw new IllegalArgumentException
+ ("Invalid property name '" + name + "'");
+ }
+ return (descriptor);
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/WrapDynaClass.java b/trunk/src/java/org/apache/commons/beanutils/WrapDynaClass.java
new file mode 100644
index 0000000..6540cf2
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/WrapDynaClass.java
@@ -0,0 +1,298 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.beans.PropertyDescriptor;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.Map;
+
+
+/**
+ * <p>Implementation of <code>DynaClass</code> for DynaBeans that wrap
+ * standard JavaBean instances.</p>
+ *
+ * <p>
+ * It is suggested that this class should not usually need to be used directly
+ * to create new <code>WrapDynaBean</code> instances.
+ * It's usually better to call the <code>WrapDynaBean</code> constructor directly.
+ * For example:</p>
+ * <code><pre>
+ * Object javaBean = ...;
+ * DynaBean wrapper = new WrapDynaBean(javaBean);
+ * </pre></code>
+ * <p>
+ *
+ * @author Craig McClanahan
+ * @version $Revision: 1.8 $ $Date: 2004/02/28 13:18:34 $
+ */
+
+public class WrapDynaClass implements DynaClass {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new WrapDynaClass for the specified JavaBean class. This
+ * constructor is private; WrapDynaClass instances will be created as
+ * needed via calls to the <code>createDynaClass(Class)</code> method.
+ *
+ * @param beanClass JavaBean class to be introspected around
+ */
+ private WrapDynaClass(Class beanClass) {
+
+ this.beanClass = beanClass;
+ introspect();
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The JavaBean <code>Class</code> which is represented by this
+ * <code>WrapDynaClass</code>.
+ */
+ protected Class beanClass = null;
+
+
+ /**
+ * The set of PropertyDescriptors for this bean class.
+ */
+ protected PropertyDescriptor descriptors[] = null;
+
+
+ /**
+ * The set of PropertyDescriptors for this bean class, keyed by the
+ * property name. Individual descriptor instances will be the same
+ * instances as those in the <code>descriptors</code> list.
+ */
+ protected HashMap descriptorsMap = new HashMap();
+
+
+ /**
+ * The set of dynamic properties that are part of this DynaClass.
+ */
+ protected DynaProperty properties[] = null;
+
+
+ /**
+ * The set of dynamic properties that are part of this DynaClass,
+ * keyed by the property name. Individual descriptor instances will
+ * be the same instances as those in the <code>properties</code> list.
+ */
+ protected HashMap propertiesMap = new HashMap();
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * The set of <code>WrapDynaClass</code> instances that have ever been
+ * created, keyed by the underlying bean Class.
+ */
+ protected static HashMap dynaClasses = new HashMap();
+
+
+ // ------------------------------------------------------ DynaClass Methods
+
+
+ /**
+ * Return the name of this DynaClass (analogous to the
+ * <code>getName()</code> method of <code>java.lang.Class</code), which
+ * allows the same <code>DynaClass</code> implementation class to support
+ * different dynamic classes, with different sets of properties.
+ */
+ public String getName() {
+
+ return (this.beanClass.getName());
+
+ }
+
+
+ /**
+ * Return a property descriptor for the specified property, if it exists;
+ * otherwise, return <code>null</code>.
+ *
+ * @param name Name of the dynamic property for which a descriptor
+ * is requested
+ *
+ * @exception IllegalArgumentException if no property name is specified
+ */
+ public DynaProperty getDynaProperty(String name) {
+
+ if (name == null) {
+ throw new IllegalArgumentException
+ ("No property name specified");
+ }
+ return ((DynaProperty) propertiesMap.get(name));
+
+ }
+
+
+ /**
+ * <p>Return an array of <code>ProperyDescriptors</code> for the properties
+ * currently defined in this DynaClass. If no properties are defined, a
+ * zero-length array will be returned.</p>
+ *
+ * <p><strong>FIXME</strong> - Should we really be implementing
+ * <code>getBeanInfo()</code> instead, which returns property descriptors
+ * and a bunch of other stuff?</p>
+ */
+ public DynaProperty[] getDynaProperties() {
+
+ return (properties);
+
+ }
+
+
+ /**
+ * <p>Instantiates a new standard JavaBean instance associated with
+ * this DynaClass and return it wrapped in a new WrapDynaBean
+ * instance. <strong>NOTE</strong> the JavaBean should have a
+ * no argument constructor.</p>
+ *
+ * <strong>NOTE</strong> - Most common use cases should not need to use
+ * this method. It is usually better to create new
+ * <code>WrapDynaBean</code> instances by calling its constructor.
+ * For example:</p>
+ * <code><pre>
+ * Object javaBean = ...;
+ * DynaBean wrapper = new WrapDynaBean(javaBean);
+ * </pre></code>
+ * <p>
+ * (This method is needed for some kinds of <code>DynaBean</code> framework.)
+ * </p>
+ *
+ * @exception IllegalAccessException if the Class or the appropriate
+ * constructor is not accessible
+ * @exception InstantiationException if this Class represents an abstract
+ * class, an array class, a primitive type, or void; or if instantiation
+ * fails for some other reason
+ */
+ public DynaBean newInstance()
+ throws IllegalAccessException, InstantiationException {
+
+ return new WrapDynaBean(beanClass.newInstance());
+
+ }
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Return the PropertyDescriptor for the specified property name, if any;
+ * otherwise return <code>null</code>.
+ *
+ * @param name Name of the property to be retrieved
+ */
+ public PropertyDescriptor getPropertyDescriptor(String name) {
+
+ return ((PropertyDescriptor) descriptorsMap.get(name));
+
+ }
+
+
+ // --------------------------------------------------------- Static Methods
+
+
+ /**
+ * Clear our cache of WrapDynaClass instances.
+ */
+ public static void clear() {
+
+ synchronized (dynaClasses) {
+ dynaClasses.clear();
+ }
+
+ }
+
+
+ /**
+ * Create (if necessary) and return a new <code>WrapDynaClass</code>
+ * instance for the specified bean class.
+ *
+ * @param beanClass Bean class for which a WrapDynaClass is requested
+ */
+ public static WrapDynaClass createDynaClass(Class beanClass) {
+
+ synchronized (dynaClasses) {
+ WrapDynaClass dynaClass =
+ (WrapDynaClass) dynaClasses.get(beanClass);
+ if (dynaClass == null) {
+ dynaClass = new WrapDynaClass(beanClass);
+ dynaClasses.put(beanClass, dynaClass);
+ }
+ return (dynaClass);
+ }
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * Introspect our bean class to identify the supported properties.
+ */
+ protected void introspect() {
+
+ // Look up the property descriptors for this bean class
+ PropertyDescriptor regulars[] =
+ PropertyUtils.getPropertyDescriptors(beanClass);
+ if (regulars == null) {
+ regulars = new PropertyDescriptor[0];
+ }
+ HashMap mappeds =
+ PropertyUtils.getMappedPropertyDescriptors(beanClass);
+ if (mappeds == null) {
+ mappeds = new HashMap();
+ }
+
+ // Construct corresponding DynaProperty information
+ properties = new DynaProperty[regulars.length + mappeds.size()];
+ for (int i = 0; i < regulars.length; i++) {
+ descriptorsMap.put(regulars[i].getName(),
+ regulars[i]);
+ properties[i] =
+ new DynaProperty(regulars[i].getName(),
+ regulars[i].getPropertyType());
+ propertiesMap.put(properties[i].getName(),
+ properties[i]);
+ }
+ int j = regulars.length;
+ Iterator names = mappeds.keySet().iterator();
+ while (names.hasNext()) {
+ String name = (String) names.next();
+ PropertyDescriptor descriptor =
+ (PropertyDescriptor) mappeds.get(name);
+ properties[j] =
+ new DynaProperty(descriptor.getName(),
+ Map.class);
+ propertiesMap.put(properties[j].getName(),
+ properties[j]);
+ j++;
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/AbstractArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/AbstractArrayConverter.java
new file mode 100644
index 0000000..0ae3312
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/AbstractArrayConverter.java
@@ -0,0 +1,168 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.io.IOException;
+import java.io.StreamTokenizer;
+import java.io.StringReader;
+import java.util.ArrayList;
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+
+/**
+ * <p>Convenience base class for converters that translate the String
+ * representation of an array into a corresponding array of primitives
+ * object. This class encapsulates the functionality required to parse
+ * the String into a list of String elements that can later be
+ * individually converted to the appropriate primitive type.</p>
+ *
+ * <p>The input syntax accepted by the <code>parseElements()</code> method
+ * is designed to be compatible with the syntax used to initialize arrays
+ * in a Java source program, except that only String literal values are
+ * supported. For maximum flexibility, the surrounding '{' and '}'
+ * characters are optional, and individual elements may be separated by
+ * any combination of whitespace and comma characters.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.6 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public abstract class AbstractArrayConverter implements Converter {
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ protected Object defaultValue = null;
+
+
+ /**
+ * <p>Model object for string arrays.</p>
+ */
+ protected static String strings[] = new String[0];
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ protected boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type. This method must be implemented by a concrete
+ * subclass.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public abstract Object convert(Class type, Object value);
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * <p>Parse an incoming String of the form similar to an array initializer
+ * in the Java language into a <code>List</code> individual Strings
+ * for each element, according to the following rules.</p>
+ * <ul>
+ * <li>The string must have matching '{' and '}' delimiters around
+ * a comma-delimited list of values.</li>
+ * <li>Whitespace before and after each element is stripped.
+ * <li>If an element is itself delimited by matching single or double
+ * quotes, the usual rules for interpreting a quoted String apply.</li>
+ * </ul>
+ *
+ * @param svalue String value to be parsed
+ *
+ * @exception ConversionException if the syntax of <code>svalue</code>
+ * is not syntactically valid
+ * @exception NullPointerException if <code>svalue</code>
+ * is <code>null</code>
+ */
+ protected List parseElements(String svalue) {
+
+ // Validate the passed argument
+ if (svalue == null) {
+ throw new NullPointerException();
+ }
+
+ // Trim any matching '{' and '}' delimiters
+ svalue = svalue.trim();
+ if (svalue.startsWith("{") && svalue.endsWith("}")) {
+ svalue = svalue.substring(1, svalue.length() - 1);
+ }
+
+ try {
+
+ // Set up a StreamTokenizer on the characters in this String
+ StreamTokenizer st =
+ new StreamTokenizer(new StringReader(svalue));
+ st.whitespaceChars(',',','); // Commas are delimiters
+ st.ordinaryChars('0', '9'); // Needed to turn off numeric flag
+ st.ordinaryChars('.', '.');
+ st.ordinaryChars('-', '-');
+ st.wordChars('0', '9'); // Needed to make part of tokens
+ st.wordChars('.', '.');
+ st.wordChars('-', '-');
+
+ // Split comma-delimited tokens into a List
+ ArrayList list = new ArrayList();
+ while (true) {
+ int ttype = st.nextToken();
+ if ((ttype == StreamTokenizer.TT_WORD) ||
+ (ttype > 0)) {
+ list.add(st.sval);
+ } else if (ttype == StreamTokenizer.TT_EOF) {
+ break;
+ } else {
+ throw new ConversionException
+ ("Encountered token of type " + ttype);
+ }
+ }
+
+ // Return the completed list
+ return (list);
+
+ } catch (IOException e) {
+
+ throw new ConversionException(e);
+
+ }
+
+
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/BigDecimalConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/BigDecimalConverter.java
new file mode 100644
index 0000000..eb8acac
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/BigDecimalConverter.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.math.BigDecimal;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.math.BigDecimal</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class BigDecimalConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public BigDecimalConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public BigDecimalConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof BigDecimal) {
+ return (value);
+ }
+
+ try {
+ return (new BigDecimal(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/BigIntegerConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/BigIntegerConverter.java
new file mode 100644
index 0000000..7584f7b
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/BigIntegerConverter.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.math.BigInteger;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.math.BigInteger</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class BigIntegerConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public BigIntegerConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public BigIntegerConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof BigInteger) {
+ return (value);
+ }
+
+ try {
+ return (new BigInteger(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/BooleanArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/BooleanArrayConverter.java
new file mode 100644
index 0000000..4943b6c
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/BooleanArrayConverter.java
@@ -0,0 +1,183 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a primitive array of boolean. On a conversion failure, returns
+ * a specified default value or throws a {@link ConversionException} depending
+ * on how this instance is constructed.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class BooleanArrayConverter extends AbstractArrayConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public BooleanArrayConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public BooleanArrayConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * <p>Model object for type comparisons.</p>
+ */
+ private static boolean model[] = new boolean[0];
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ // Deal with a null value
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // Deal with the no-conversion-needed case
+ if (model.getClass() == value.getClass()) {
+ return (value);
+ }
+
+ // Deal with input value as a String array
+ if (strings.getClass() == value.getClass()) {
+ try {
+ String values[] = (String[]) value;
+ boolean results[] = new boolean[values.length];
+ for (int i = 0; i < values.length; i++) {
+ String stringValue = values[i];
+ if (stringValue.equalsIgnoreCase("yes") ||
+ stringValue.equalsIgnoreCase("y") ||
+ stringValue.equalsIgnoreCase("true") ||
+ stringValue.equalsIgnoreCase("on") ||
+ stringValue.equalsIgnoreCase("1")) {
+ results[i] = true;
+ } else if (stringValue.equalsIgnoreCase("no") ||
+ stringValue.equalsIgnoreCase("n") ||
+ stringValue.equalsIgnoreCase("false") ||
+ stringValue.equalsIgnoreCase("off") ||
+ stringValue.equalsIgnoreCase("0")) {
+ results[i] = false;
+ } else {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString());
+ }
+ }
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+ }
+
+ // Parse the input value as a String into elements
+ // and convert to the appropriate type
+ try {
+ List list = parseElements(value.toString());
+ boolean results[] = new boolean[list.size()];
+ for (int i = 0; i < results.length; i++) {
+ String stringValue = (String) list.get(i);
+ if (stringValue.equalsIgnoreCase("yes") ||
+ stringValue.equalsIgnoreCase("y") ||
+ stringValue.equalsIgnoreCase("true") ||
+ stringValue.equalsIgnoreCase("on") ||
+ stringValue.equalsIgnoreCase("1")) {
+ results[i] = true;
+ } else if (stringValue.equalsIgnoreCase("no") ||
+ stringValue.equalsIgnoreCase("n") ||
+ stringValue.equalsIgnoreCase("false") ||
+ stringValue.equalsIgnoreCase("off") ||
+ stringValue.equalsIgnoreCase("0")) {
+ results[i] = false;
+ } else {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString());
+ }
+ }
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/BooleanConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/BooleanConverter.java
new file mode 100644
index 0000000..6ff1572
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/BooleanConverter.java
@@ -0,0 +1,140 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.Boolean</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.9 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class BooleanConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public BooleanConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public BooleanConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Boolean) {
+ return (value);
+ }
+
+ try {
+ String stringValue = value.toString();
+ if (stringValue.equalsIgnoreCase("yes") ||
+ stringValue.equalsIgnoreCase("y") ||
+ stringValue.equalsIgnoreCase("true") ||
+ stringValue.equalsIgnoreCase("on") ||
+ stringValue.equalsIgnoreCase("1")) {
+ return (Boolean.TRUE);
+ } else if (stringValue.equalsIgnoreCase("no") ||
+ stringValue.equalsIgnoreCase("n") ||
+ stringValue.equalsIgnoreCase("false") ||
+ stringValue.equalsIgnoreCase("off") ||
+ stringValue.equalsIgnoreCase("0")) {
+ return (Boolean.FALSE);
+ } else if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(stringValue);
+ }
+ } catch (ClassCastException e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/ByteArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/ByteArrayConverter.java
new file mode 100644
index 0000000..c428408
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/ByteArrayConverter.java
@@ -0,0 +1,145 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a primitive array of byte. On a conversion failure, returns
+ * a specified default value or throws a {@link ConversionException} depending
+ * on how this instance is constructed.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class ByteArrayConverter extends AbstractArrayConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public ByteArrayConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public ByteArrayConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * <p>Model object for type comparisons.</p>
+ */
+ private static byte model[] = new byte[0];
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ // Deal with a null value
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // Deal with the no-conversion-needed case
+ if (model.getClass() == value.getClass()) {
+ return (value);
+ }
+
+ // Deal with input value as a String array
+ if (strings.getClass() == value.getClass()) {
+ try {
+ String values[] = (String[]) value;
+ byte results[] = new byte[values.length];
+ for (int i = 0; i < values.length; i++) {
+ results[i] = Byte.parseByte(values[i]);
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+ }
+
+ // Parse the input value as a String into elements
+ // and convert to the appropriate type
+ try {
+ List list = parseElements(value.toString());
+ byte results[] = new byte[list.size()];
+ for (int i = 0; i < results.length; i++) {
+ results[i] = Byte.parseByte((String) list.get(i));
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/ByteConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/ByteConverter.java
new file mode 100644
index 0000000..140b3e8
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/ByteConverter.java
@@ -0,0 +1,127 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.Byte</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.9 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class ByteConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public ByteConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public ByteConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // System.err.println("VALUE=" + value + ", TYPE=" + value.getClass().getName());
+
+ if (value instanceof Byte) {
+ return (value);
+ } else if (value instanceof Number) {
+ return new Byte(((Number)value).byteValue());
+ }
+
+ try {
+ return (new Byte(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/CharacterArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/CharacterArrayConverter.java
new file mode 100644
index 0000000..8353ce4
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/CharacterArrayConverter.java
@@ -0,0 +1,145 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a primitive array of char. On a conversion failure, returns
+ * a specified default value or throws a {@link ConversionException} depending
+ * on how this instance is constructed.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class CharacterArrayConverter extends AbstractArrayConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public CharacterArrayConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public CharacterArrayConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * <p>Model object for type comparisons.</p>
+ */
+ private static char model[] = new char[0];
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ // Deal with a null value
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // Deal with the no-conversion-needed case
+ if (model.getClass() == value.getClass()) {
+ return (value);
+ }
+
+ // Deal with input value as a String array
+ if (strings.getClass() == value.getClass()) {
+ try {
+ String values[] = (String[]) value;
+ char results[] = new char[values.length];
+ for (int i = 0; i < values.length; i++) {
+ results[i] = values[i].charAt(0);
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+ }
+
+ // Parse the input value as a String into elements
+ // and convert to the appropriate type
+ try {
+ List list = parseElements(value.toString());
+ char results[] = new char[list.size()];
+ for (int i = 0; i < results.length; i++) {
+ results[i] = ((String) list.get(i)).charAt(0);
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/CharacterConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/CharacterConverter.java
new file mode 100644
index 0000000..dfae43c
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/CharacterConverter.java
@@ -0,0 +1,123 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.Character</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class CharacterConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public CharacterConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public CharacterConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Character) {
+ return (value);
+ }
+
+ try {
+ return (new Character(value.toString().charAt(0)));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/ClassConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/ClassConverter.java
new file mode 100644
index 0000000..7e671f0
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/ClassConverter.java
@@ -0,0 +1,130 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.Class</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs. The class will be loaded from the thread context class
+ * loader (if it exists); otherwise the class loader that loaded this class
+ * will be used.</p>
+ *
+ * @author Tomas Viberg
+ * @version $Revision: 1.6 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class ClassConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public ClassConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public ClassConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Class) {
+ return (value);
+ }
+
+ try {
+ ClassLoader classLoader =
+ Thread.currentThread().getContextClassLoader();
+ if (classLoader == null) {
+ classLoader = ClassConverter.class.getClassLoader();
+ }
+ return (classLoader.loadClass(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/DoubleArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/DoubleArrayConverter.java
new file mode 100644
index 0000000..a7c9799
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/DoubleArrayConverter.java
@@ -0,0 +1,145 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a primitive array of double. On a conversion failure, returns
+ * a specified default value or throws a {@link ConversionException} depending
+ * on how this instance is constructed.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class DoubleArrayConverter extends AbstractArrayConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public DoubleArrayConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public DoubleArrayConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * <p>Model object for type comparisons.</p>
+ */
+ private static double model[] = new double[0];
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ // Deal with a null value
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // Deal with the no-conversion-needed case
+ if (model.getClass() == value.getClass()) {
+ return (value);
+ }
+
+ // Deal with input value as a String array
+ if (strings.getClass() == value.getClass()) {
+ try {
+ String values[] = (String[]) value;
+ double results[] = new double[values.length];
+ for (int i = 0; i < values.length; i++) {
+ results[i] = Double.parseDouble(values[i]);
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+ }
+
+ // Parse the input value as a String into elements
+ // and convert to the appropriate type
+ try {
+ List list = parseElements(value.toString());
+ double results[] = new double[list.size()];
+ for (int i = 0; i < results.length; i++) {
+ results[i] = Double.parseDouble((String) list.get(i));
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/DoubleConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/DoubleConverter.java
new file mode 100644
index 0000000..8930e8c
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/DoubleConverter.java
@@ -0,0 +1,126 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.Double</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.8 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class DoubleConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public DoubleConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public DoubleConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Double) {
+ return (value);
+ } else if(value instanceof Number) {
+ return new Double(((Number)value).doubleValue());
+ }
+
+
+ try {
+ return (new Double(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/FileConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/FileConverter.java
new file mode 100644
index 0000000..a77db84
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/FileConverter.java
@@ -0,0 +1,110 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import java.io.File;
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.io.FileL</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author James Strachan
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.6
+ */
+public final class FileConverter implements Converter {
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public FileConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public FileConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof File) {
+ return (value);
+ }
+
+ return new File(value.toString());
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/FloatArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/FloatArrayConverter.java
new file mode 100644
index 0000000..9ce5225
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/FloatArrayConverter.java
@@ -0,0 +1,145 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a primitive array of float. On a conversion failure, returns
+ * a specified default value or throws a {@link ConversionException} depending
+ * on how this instance is constructed.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class FloatArrayConverter extends AbstractArrayConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public FloatArrayConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public FloatArrayConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * <p>Model object for type comparisons.</p>
+ */
+ private static float model[] = new float[0];
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ // Deal with a null value
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // Deal with the no-conversion-needed case
+ if (model.getClass() == value.getClass()) {
+ return (value);
+ }
+
+ // Deal with input value as a String array
+ if (strings.getClass() == value.getClass()) {
+ try {
+ String values[] = (String[]) value;
+ float results[] = new float[values.length];
+ for (int i = 0; i < values.length; i++) {
+ results[i] = Float.parseFloat(values[i]);
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+ }
+
+ // Parse the input value as a String into elements
+ // and convert to the appropriate type
+ try {
+ List list = parseElements(value.toString());
+ float results[] = new float[list.size()];
+ for (int i = 0; i < results.length; i++) {
+ results[i] = Float.parseFloat((String) list.get(i));
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/FloatConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/FloatConverter.java
new file mode 100644
index 0000000..6235ad2
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/FloatConverter.java
@@ -0,0 +1,125 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.Float</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.8 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class FloatConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public FloatConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public FloatConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Float) {
+ return (value);
+ } else if(value instanceof Number) {
+ return new Float(((Number)value).floatValue());
+ }
+
+ try {
+ return (new Float(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/IntegerArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/IntegerArrayConverter.java
new file mode 100644
index 0000000..6220d96
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/IntegerArrayConverter.java
@@ -0,0 +1,145 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a primitive array of int. On a conversion failure, returns
+ * a specified default value or throws a {@link ConversionException} depending
+ * on how this instance is constructed.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class IntegerArrayConverter extends AbstractArrayConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public IntegerArrayConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public IntegerArrayConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * <p>Model object for type comparisons.</p>
+ */
+ private static int model[] = new int[0];
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ // Deal with a null value
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // Deal with the no-conversion-needed case
+ if (model.getClass() == value.getClass()) {
+ return (value);
+ }
+
+ // Deal with input value as a String array
+ if (strings.getClass() == value.getClass()) {
+ try {
+ String values[] = (String[]) value;
+ int results[] = new int[values.length];
+ for (int i = 0; i < values.length; i++) {
+ results[i] = Integer.parseInt(values[i]);
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+ }
+
+ // Parse the input value as a String into elements
+ // and convert to the appropriate type
+ try {
+ List list = parseElements(value.toString());
+ int results[] = new int[list.size()];
+ for (int i = 0; i < results.length; i++) {
+ results[i] = Integer.parseInt((String) list.get(i));
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/IntegerConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/IntegerConverter.java
new file mode 100644
index 0000000..403afd9
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/IntegerConverter.java
@@ -0,0 +1,125 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.Integer</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.8 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class IntegerConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public IntegerConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public IntegerConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Integer) {
+ return (value);
+ } else if(value instanceof Number) {
+ return new Integer(((Number)value).intValue());
+ }
+
+ try {
+ return (new Integer(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/LongArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/LongArrayConverter.java
new file mode 100644
index 0000000..2f52e1f
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/LongArrayConverter.java
@@ -0,0 +1,145 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a primitive array of long. On a conversion failure, returns
+ * a specified default value or throws a {@link ConversionException} depending
+ * on how this instance is constructed.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class LongArrayConverter extends AbstractArrayConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public LongArrayConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public LongArrayConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * <p>Model object for type comparisons.</p>
+ */
+ private static long model[] = new long[0];
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ // Deal with a null value
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // Deal with the no-conversion-needed case
+ if (model.getClass() == value.getClass()) {
+ return (value);
+ }
+
+ // Deal with input value as a String array
+ if (strings.getClass() == value.getClass()) {
+ try {
+ String values[] = (String[]) value;
+ long results[] = new long[values.length];
+ for (int i = 0; i < values.length; i++) {
+ results[i] = Long.parseLong(values[i]);
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+ }
+
+ // Parse the input value as a String into elements
+ // and convert to the appropriate type
+ try {
+ List list = parseElements(value.toString());
+ long results[] = new long[list.size()];
+ for (int i = 0; i < results.length; i++) {
+ results[i] = Long.parseLong((String) list.get(i));
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/LongConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/LongConverter.java
new file mode 100644
index 0000000..cd6bcf7
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/LongConverter.java
@@ -0,0 +1,125 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.Long</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.8 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class LongConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public LongConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public LongConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Long) {
+ return (value);
+ } else if(value instanceof Number) {
+ return new Long(((Number)value).longValue());
+ }
+
+ try {
+ return (new Long(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/ShortArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/ShortArrayConverter.java
new file mode 100644
index 0000000..7c98454
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/ShortArrayConverter.java
@@ -0,0 +1,145 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a primitive array of short. On a conversion failure, returns
+ * a specified default value or throws a {@link ConversionException} depending
+ * on how this instance is constructed.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class ShortArrayConverter extends AbstractArrayConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public ShortArrayConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public ShortArrayConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * <p>Model object for type comparisons.</p>
+ */
+ private static short model[] = new short[0];
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ // Deal with a null value
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // Deal with the no-conversion-needed case
+ if (model.getClass() == value.getClass()) {
+ return (value);
+ }
+
+ // Deal with input value as a String array
+ if (strings.getClass() == value.getClass()) {
+ try {
+ String values[] = (String[]) value;
+ short results[] = new short[values.length];
+ for (int i = 0; i < values.length; i++) {
+ results[i] = Short.parseShort(values[i]);
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+ }
+
+ // Parse the input value as a String into elements
+ // and convert to the appropriate type
+ try {
+ List list = parseElements(value.toString());
+ short results[] = new short[list.size()];
+ for (int i = 0; i < results.length; i++) {
+ results[i] = Short.parseShort((String) list.get(i));
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/ShortConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/ShortConverter.java
new file mode 100644
index 0000000..786f1a3
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/ShortConverter.java
@@ -0,0 +1,125 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.Short</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.8 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class ShortConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public ShortConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public ShortConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Short) {
+ return (value);
+ } else if (value instanceof Number) {
+ return new Short(((Number)value).shortValue());
+ }
+
+ try {
+ return (new Short(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/SqlDateConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/SqlDateConverter.java
new file mode 100644
index 0000000..e5d7867
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/SqlDateConverter.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.sql.Date;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.sql.Date</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class SqlDateConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public SqlDateConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public SqlDateConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Date) {
+ return (value);
+ }
+
+ try {
+ return (Date.valueOf(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/SqlTimeConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/SqlTimeConverter.java
new file mode 100644
index 0000000..01c2a38
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/SqlTimeConverter.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.sql.Time;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.sql.Time</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class SqlTimeConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public SqlTimeConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public SqlTimeConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Time) {
+ return (value);
+ }
+
+ try {
+ return (Time.valueOf(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/SqlTimestampConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/SqlTimestampConverter.java
new file mode 100644
index 0000000..64c83ab
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/SqlTimestampConverter.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.sql.Timestamp;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.sql.Timestamp</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class SqlTimestampConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public SqlTimestampConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public SqlTimestampConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof Timestamp) {
+ return (value);
+ }
+
+ try {
+ return (Timestamp.valueOf(value.toString()));
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/StringArrayConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/StringArrayConverter.java
new file mode 100644
index 0000000..5ba0bab
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/StringArrayConverter.java
@@ -0,0 +1,146 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import java.util.List;
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into an array of String. On a conversion failure, returns
+ * a specified default value or throws a {@link ConversionException} depending
+ * on how this instance is constructed.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.4
+ */
+
+public final class StringArrayConverter extends AbstractArrayConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public StringArrayConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public StringArrayConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * <p>Model object for type comparisons.</p>
+ */
+ private static String model[] = new String[0];
+
+ /**
+ * <p> Model object for int arrays.</p>
+ */
+ private static int ints[] = new int[0];
+
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ // Deal with a null value
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ // Deal with the no-conversion-needed case
+ if (model.getClass() == value.getClass()) {
+ return (value);
+ }
+
+ // Deal with the input value as an int array
+ if (ints.getClass() == value.getClass())
+ {
+ int[] values = (int[]) value;
+ String[] results = new String[values.length];
+ for (int i = 0; i < values.length; i++)
+ {
+ results[i] = Integer.toString(values[i]);
+ }
+
+ return (results);
+ }
+
+ // Parse the input value as a String into elements
+ // and convert to the appropriate type
+ try {
+ List list = parseElements(value.toString());
+ String results[] = new String[list.size()];
+ for (int i = 0; i < results.length; i++) {
+ results[i] = (String) list.get(i);
+ }
+ return (results);
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(value.toString(), e);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/StringConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/StringConverter.java
new file mode 100644
index 0000000..c01bec8
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/StringConverter.java
@@ -0,0 +1,63 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.lang.String</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.5 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class StringConverter implements Converter {
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ return ((String) null);
+ } else {
+ return (value.toString());
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/URLConverter.java b/trunk/src/java/org/apache/commons/beanutils/converters/URLConverter.java
new file mode 100644
index 0000000..8e2e24f
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/URLConverter.java
@@ -0,0 +1,127 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.converters;
+
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+import java.net.URL;
+import java.net.MalformedURLException;
+
+
+
+/**
+ * <p>Standard {@link Converter} implementation that converts an incoming
+ * String into a <code>java.net.URL</code> object, optionally using a
+ * default value or throwing a {@link ConversionException} if a conversion
+ * error occurs.</p>
+ *
+ * @author Henri Yandell
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:34 $
+ * @since 1.3
+ */
+
+public final class URLConverter implements Converter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a {@link Converter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ */
+ public URLConverter() {
+
+ this.defaultValue = null;
+ this.useDefault = false;
+
+ }
+
+
+ /**
+ * Create a {@link Converter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public URLConverter(Object defaultValue) {
+
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+
+ }
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The default value specified to our Constructor, if any.
+ */
+ private Object defaultValue = null;
+
+
+ /**
+ * Should we return the default value on conversion errors?
+ */
+ private boolean useDefault = true;
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Convert the specified input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException("No value specified");
+ }
+ }
+
+ if (value instanceof URL) {
+ return (value);
+ }
+
+ try {
+ return new URL(value.toString());
+ } catch(MalformedURLException murle) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(murle);
+ }
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/converters/package.html b/trunk/src/java/org/apache/commons/beanutils/converters/package.html
new file mode 100644
index 0000000..3b52be3
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/converters/package.html
@@ -0,0 +1,12 @@
+<html>
+<head>
+<title>Package Documentation for org.apache.commons.beanutils.converters</title>
+</head>
+<body bgcolor="white">
+<p>Standard implementations of the
+<code>Converter</code>
+interface that are pre-registered with
+<code>ConvertUtils</code>
+at startup time.</p>
+</body>
+</html>
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/BaseLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/BaseLocaleConverter.java
new file mode 100755
index 0000000..3e5ec59
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/BaseLocaleConverter.java
@@ -0,0 +1,235 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale;
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.locale.converters.DateLocaleConverter;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.text.ParseException;
+import java.util.Locale;
+
+
+/**
+ * <p>The base class for all standart type locale-sensitive converters.
+ * It has {@link LocaleConverter} and {@link org.apache.commons.beanutils.Converter} implementations,
+ * that convert an incoming locale-sensitive Object into an object of correspond type,
+ * optionally using a default value or throwing a {@link ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public abstract class BaseLocaleConverter implements LocaleConverter {
+
+ // ----------------------------------------------------- Instance Variables
+
+ /** All logging goes through this logger */
+ private static Log log = LogFactory.getLog(BaseLocaleConverter.class);
+
+ /** The default value specified to our Constructor, if any. */
+ private Object defaultValue = null;
+
+ /** Should we return the default value on conversion errors? */
+ protected boolean useDefault = false;
+
+ /** The locale specified to our Constructor, by default - system locale. */
+ protected Locale locale = Locale.getDefault();
+
+ /** The default pattern specified to our Constructor, if any. */
+ protected String pattern = null;
+
+ /** The flag indicating whether the given pattern string is localized or not. */
+ protected boolean locPattern = false;
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link LocaleConverter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ * An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ protected BaseLocaleConverter(Locale locale, String pattern) {
+
+ this(null, locale, pattern, false, false);
+ }
+
+ /**
+ * Create a {@link LocaleConverter} that will throw a {@link ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ protected BaseLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ this(null, locale, pattern, false, locPattern);
+ }
+
+ /**
+ * Create a {@link LocaleConverter} that will return the specified default value
+ * if a conversion error occurs.
+ * An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ protected BaseLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link LocaleConverter} that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ protected BaseLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ this(defaultValue, locale, pattern, true, locPattern);
+ }
+
+ /**
+ * Create a {@link LocaleConverter} that will return the specified default value
+ * or throw a {@link ConversionException} if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param useDefault Indicate whether the default value is used or not
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ private BaseLocaleConverter(Object defaultValue, Locale locale,
+ String pattern, boolean useDefault, boolean locPattern) {
+
+ if (useDefault) {
+ this.defaultValue = defaultValue;
+ this.useDefault = true;
+ }
+
+ if (locale != null) {
+ this.locale = locale;
+ }
+
+ this.pattern = pattern;
+ this.locPattern = locPattern;
+ }
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+
+ abstract protected Object parse(Object value, String pattern) throws ParseException;
+
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object.
+ * The default pattern is used for the convertion.
+ *
+ * @param value The input object to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Object value) {
+ return convert(value, null);
+ }
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Object value, String pattern) {
+ return convert(null, value, pattern);
+ }
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type. The default pattern is used for the convertion.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input object to be converted
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value) {
+ return convert(type, value, null);
+ }
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value, String pattern) {
+ if (value == null) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ // symmetric beanutils function allows null
+ // so do not: throw new ConversionException("No value specified");
+ log.debug("Null value specified for conversion, returing null");
+ return null;
+ }
+ }
+
+ try {
+ if (pattern != null) {
+ return parse(value, pattern);
+ } else {
+ return parse(value, this.pattern);
+ }
+ } catch (Exception e) {
+ if (useDefault) {
+ return (defaultValue);
+ } else {
+ throw new ConversionException(e);
+ }
+ }
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/LocaleBeanUtils.java b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleBeanUtils.java
new file mode 100755
index 0000000..34c73ac
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleBeanUtils.java
@@ -0,0 +1,533 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale;
+
+
+import org.apache.commons.beanutils.*;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.beans.IndexedPropertyDescriptor;
+import java.beans.PropertyDescriptor;
+import java.lang.reflect.InvocationTargetException;
+import java.util.Locale;
+
+
+/**
+ * <p>Utility methods for populating JavaBeans properties
+ * via reflection in a locale-dependent manner.</p>
+ *
+ * <p>The implementations for these methods are provided by <code>LocaleBeanUtilsBean</code>.
+ * For more details see {@link LocaleBeanUtilsBean}.</p>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @author Rey Fran�ois
+ * @author Gregor Ra�man
+ * @author Yauheny Mikulski
+ */
+
+public class LocaleBeanUtils extends BeanUtils {
+
+
+ // ----------------------------------------------------- Instance Variables
+
+ /** All logging goes through this logger */
+ private static Log log = LogFactory.getLog(LocaleBeanUtils.class);
+
+ /**
+ * <p>Gets the locale used when no locale is passed.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getDefaultLocale()
+ */
+ public static Locale getDefaultLocale() {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getDefaultLocale();
+ }
+
+
+ /**
+ * <p>Sets the locale used when no locale is passed.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#setDefaultLocale(Locale)
+ */
+ public static void setDefaultLocale(Locale locale) {
+
+ LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().setDefaultLocale(locale);
+ }
+
+ /**
+ * <p>Gets whether the pattern is localized or not.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getApplyLocalized()
+ */
+ public static boolean getApplyLocalized() {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getApplyLocalized();
+ }
+
+ /**
+ * <p>Sets whether the pattern is localized or not.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#setApplyLocalized(boolean)
+ */
+ public static void setApplyLocalized(boolean newApplyLocalized) {
+
+ LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().setApplyLocalized(newApplyLocalized);
+ }
+
+
+ // --------------------------------------------------------- Public Methods
+
+ /**
+ * <p>Return the value of the specified locale-sensitive indexed property
+ * of the specified bean, as a String.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getIndexedProperty(Object, String, String)
+ */
+ public static String getIndexedProperty(Object bean, String name, String pattern)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getIndexedProperty(bean, name, pattern);
+ }
+
+ /**
+ * Return the value of the specified locale-sensitive indexed property
+ * of the specified bean, as a String using the default convertion pattern of
+ * the corresponding {@link LocaleConverter}.
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getIndexedProperty(Object, String)
+ */
+ public static String getIndexedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getIndexedProperty(bean, name);
+ }
+
+ /**
+ * <p>Return the value of the specified locale-sensetive indexed property
+ * of the specified bean, as a String using the specified convertion pattern.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getIndexedProperty(Object, String, int, String)
+ */
+ public static String getIndexedProperty(Object bean,
+ String name, int index, String pattern)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getIndexedProperty(bean, name, index, pattern);
+ }
+
+ /**
+ * <p>Return the value of the specified locale-sensetive indexed property
+ * of the specified bean, as a String using the default convertion pattern of
+ * the corresponding {@link LocaleConverter}.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getIndexedProperty(Object, String, int)
+ */
+ public static String getIndexedProperty(Object bean,
+ String name, int index)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getIndexedProperty(bean, name, index);
+ }
+
+ /**
+ * <p>Return the value of the specified simple locale-sensitive property
+ * of the specified bean, converted to a String using the specified
+ * convertion pattern.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getSimpleProperty(Object, String, String)
+ */
+ public static String getSimpleProperty(Object bean, String name, String pattern)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getSimpleProperty(bean, name, pattern);
+ }
+
+ /**
+ * <p>Return the value of the specified simple locale-sensitive property
+ * of the specified bean, converted to a String using the default
+ * convertion pattern of the corresponding {@link LocaleConverter}.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getSimpleProperty(Object, String)
+ */
+ public static String getSimpleProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getSimpleProperty(bean, name);
+ }
+
+ /**
+ * <p>Return the value of the specified mapped locale-sensitive property
+ * of the specified bean, as a String using the specified convertion pattern.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getMappedProperty(Object, String, String, String)
+ */
+ public static String getMappedProperty(Object bean,
+ String name, String key, String pattern)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getMappedProperty(bean, name, key, pattern);
+ }
+
+ /**
+ * <p>Return the value of the specified mapped locale-sensitive property
+ * of the specified bean, as a String
+ * The key is specified as a method parameter and must *not* be included
+ * in the property name expression.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getMappedProperty(Object, String, String)
+ */
+ public static String getMappedProperty(Object bean,
+ String name, String key)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getMappedProperty(bean, name, key);
+ }
+
+
+ /**
+ * <p>Return the value of the specified locale-sensitive mapped property
+ * of the specified bean, as a String using the specified pattern.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getMappedPropertyLocale(Object, String, String)
+ */
+ public static String getMappedPropertyLocale(Object bean, String name, String pattern)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getMappedPropertyLocale(bean, name, pattern);
+ }
+
+
+ /**
+ * <p>Return the value of the specified locale-sensitive mapped property
+ * of the specified bean, as a String using the default
+ * convertion pattern of the corresponding {@link LocaleConverter}.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getMappedProperty(Object, String)
+ */
+ public static String getMappedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getMappedProperty(bean, name);
+ }
+
+ /**
+ * <p>Return the value of the (possibly nested) locale-sensitive property
+ * of the specified name, for the specified bean,
+ * as a String using the specified pattern.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getNestedProperty(Object, String, String)
+ */
+ public static String getNestedProperty(Object bean, String name, String pattern)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getNestedProperty(bean, name, pattern);
+ }
+
+ /**
+ * <p>Return the value of the (possibly nested) locale-sensitive property
+ * of the specified name.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getNestedProperty(Object, String)
+ */
+ public static String getNestedProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getNestedProperty(bean, name);
+ }
+
+ /**
+ * <p>Return the value of the specified locale-sensitive property
+ * of the specified bean.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getProperty(Object, String, String)
+ */
+ public static String getProperty(Object bean, String name, String pattern)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getProperty(bean, name, pattern);
+ }
+
+ /**
+ * <p>Return the value of the specified locale-sensitive property
+ * of the specified bean.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#getProperty(Object, String)
+ */
+ public static String getProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getProperty(bean, name);
+ }
+
+ /**
+ * <p>Set the specified locale-sensitive property value, performing type
+ * conversions as required to conform to the type of the destination property
+ * using the default convertion pattern of the corresponding {@link LocaleConverter}.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#setProperty(Object, String, Object)
+ */
+ public static void setProperty(Object bean, String name, Object value)
+ throws IllegalAccessException, InvocationTargetException {
+
+ LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().setProperty(bean, name, value);
+ }
+
+ /**
+ * <p>Set the specified locale-sensitive property value, performing type
+ * conversions as required to conform to the type of the destination
+ * property using the specified convertion pattern.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#setProperty(Object, String, Object, String)
+ */
+ public static void setProperty(Object bean, String name, Object value, String pattern)
+ throws IllegalAccessException, InvocationTargetException {
+
+ LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().setProperty(bean, name, value, pattern);
+ }
+
+ /**
+ * <p>Calculate the property type.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#definePropertyType(Object, String, String)
+ */
+ protected static Class definePropertyType(Object target, String name, String propName)
+ throws IllegalAccessException, InvocationTargetException {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().definePropertyType(target, name, propName);
+ }
+
+ /**
+ * <p>Convert the specified value to the required type using the
+ * specified convertion pattern.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#convert(Class, int, Object, String)
+ */
+ protected static Object convert(Class type, int index, Object value, String pattern) {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().convert(type, index, value, pattern);
+ }
+
+ /**
+ * <p>Convert the specified value to the required type.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#convert(Class, int, Object)
+ */
+ protected static Object convert(Class type, int index, Object value) {
+
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().convert(type, index, value);
+ }
+
+ /**
+ * <p>Invoke the setter method.</p>
+ *
+ * <p>For more details see <code>LocaleBeanUtilsBean</code></p>
+ *
+ * @see LocaleBeanUtilsBean#invokeSetter(Object, String, String, int, Object)
+ */
+ protected static void invokeSetter(Object target, String propName, String key, int index, Object newValue)
+ throws IllegalAccessException, InvocationTargetException {
+
+ LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().invokeSetter(target, propName, key, index, newValue);
+ }
+
+ /**
+ * Resolve any nested expression to get the actual target bean.
+ *
+ * @deprecated moved into <code>LocaleBeanUtilsBean</code>
+ * @param bean The bean
+ * @param name The property name
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ protected static Descriptor calculate(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException {
+
+ String propName = null; // Simple name of target property
+ int index = -1; // Indexed subscript value (if any)
+ String key = null; // Mapped key value (if any)
+
+ Object target = bean;
+ int delim = name.lastIndexOf(PropertyUtils.NESTED_DELIM);
+ if (delim >= 0) {
+ try {
+ target =
+ PropertyUtils.getProperty(bean, name.substring(0, delim));
+ }
+ catch (NoSuchMethodException e) {
+ return null; // Skip this property setter
+ }
+ name = name.substring(delim + 1);
+ if (log.isTraceEnabled()) {
+ log.trace(" Target bean = " + target);
+ log.trace(" Target name = " + name);
+ }
+ }
+
+ // Calculate the property name, index, and key values
+ propName = name;
+ int i = propName.indexOf(PropertyUtils.INDEXED_DELIM);
+ if (i >= 0) {
+ int k = propName.indexOf(PropertyUtils.INDEXED_DELIM2);
+ try {
+ index =
+ Integer.parseInt(propName.substring(i + 1, k));
+ }
+ catch (NumberFormatException e) {
+ ;
+ }
+ propName = propName.substring(0, i);
+ }
+ int j = propName.indexOf(PropertyUtils.MAPPED_DELIM);
+ if (j >= 0) {
+ int k = propName.indexOf(PropertyUtils.MAPPED_DELIM2);
+ try {
+ key = propName.substring(j + 1, k);
+ }
+ catch (IndexOutOfBoundsException e) {
+ ;
+ }
+ propName = propName.substring(0, j);
+ }
+ return new Descriptor(target, name, propName, key, index);
+ }
+
+ /** @deprecated moved into <code>LocaleBeanUtils</code> */
+ protected static class Descriptor {
+
+ private int index = -1; // Indexed subscript value (if any)
+ private String name;
+ private String propName; // Simple name of target property
+ private String key; // Mapped key value (if any)
+ private Object target;
+
+ public Descriptor(Object target, String name, String propName, String key, int index) {
+
+ setTarget(target);
+ setName(name);
+ setPropName(propName);
+ setKey(key);
+ setIndex(index);
+ }
+
+ public Object getTarget() {
+ return target;
+ }
+
+ public void setTarget(Object target) {
+ this.target = target;
+ }
+
+ public String getKey() {
+ return key;
+ }
+
+ public void setKey(String key) {
+ this.key = key;
+ }
+
+ public int getIndex() {
+ return index;
+ }
+
+ public void setIndex(int index) {
+ this.index = index;
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public void setName(String name) {
+ this.name = name;
+ }
+
+ public String getPropName() {
+ return propName;
+ }
+
+ public void setPropName(String propName) {
+ this.propName = propName;
+ }
+ }
+}
+
+
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/LocaleBeanUtilsBean.java b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleBeanUtilsBean.java
new file mode 100644
index 0000000..3081350
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleBeanUtilsBean.java
@@ -0,0 +1,957 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale;
+
+
+import org.apache.commons.beanutils.*;
+import org.apache.commons.beanutils.ContextClassLoaderLocal;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.beans.IndexedPropertyDescriptor;
+import java.beans.PropertyDescriptor;
+import java.lang.reflect.InvocationTargetException;
+import java.util.Locale;
+
+
+/**
+ * <p>Utility methods for populating JavaBeans properties
+ * via reflection in a locale-dependent manner.</p>
+ *
+ * @author Craig R. McClanahan
+ * @author Ralph Schaer
+ * @author Chris Audley
+ * @author Rey Fran�ois
+ * @author Gregor Ra�man
+ * @author Yauheny Mikulski
+ * @since 1.7
+ */
+
+public class LocaleBeanUtilsBean extends BeanUtilsBean {
+
+ /**
+ * Contains <code>LocaleBeanUtilsBean</code> instances indexed by context classloader.
+ */
+ private static final ContextClassLoaderLocal
+ localeBeansByClassLoader = new ContextClassLoaderLocal() {
+ // Creates the default instance used when the context classloader is unavailable
+ protected Object initialValue() {
+ return new LocaleBeanUtilsBean();
+ }
+ };
+
+ /** Gets singleton instance */
+ public synchronized static LocaleBeanUtilsBean getLocaleBeanUtilsInstance() {
+ return (LocaleBeanUtilsBean)localeBeansByClassLoader.get();
+ }
+
+ /**
+ * Sets the instance which provides the functionality for {@link LocaleBeanUtils}.
+ * This is a pseudo-singleton - an single instance is provided per (thread) context classloader.
+ * This mechanism provides isolation for web apps deployed in the same container.
+ */
+ public synchronized static void setInstance(LocaleBeanUtilsBean newInstance) {
+ localeBeansByClassLoader.set(newInstance);
+ }
+
+ /** All logging goes through this logger */
+ private static Log log = LogFactory.getLog(LocaleBeanUtilsBean.class);
+
+ // ----------------------------------------------------- Instance Variables
+
+ /** Convertor used by this class */
+ private LocaleConvertUtilsBean localeConvertUtils;
+
+ // --------------------------------------------------------- Constructors
+
+ /** Construct instance with standard conversion bean */
+ public LocaleBeanUtilsBean() {
+ this.localeConvertUtils = new LocaleConvertUtilsBean();
+ }
+
+ /**
+ * Construct instance that uses given locale conversion
+ *
+ * @param localeConvertUtils use this <code>localeConvertUtils</code> to perform
+ * conversions
+ * @param convertUtilsBean use this for standard conversions
+ * @param propertyUtilsBean use this for property conversions
+ */
+ public LocaleBeanUtilsBean(
+ LocaleConvertUtilsBean localeConvertUtils,
+ ConvertUtilsBean convertUtilsBean,
+ PropertyUtilsBean propertyUtilsBean) {
+ super(convertUtilsBean, propertyUtilsBean);
+ this.localeConvertUtils = localeConvertUtils;
+ }
+
+ /**
+ * Construct instance that uses given locale conversion
+ *
+ * @param localeConvertUtils use this <code>localeConvertUtils</code> to perform
+ * conversions
+ */
+ public LocaleBeanUtilsBean(LocaleConvertUtilsBean localeConvertUtils) {
+ this.localeConvertUtils = localeConvertUtils;
+ }
+
+ // --------------------------------------------------------- Public Methods
+
+ /** Gets the bean instance used for conversions */
+ public LocaleConvertUtilsBean getLocaleConvertUtils() {
+ return localeConvertUtils;
+ }
+
+ /**
+ * Gets the default Locale
+ */
+ public Locale getDefaultLocale() {
+
+ return getLocaleConvertUtils().getDefaultLocale();
+ }
+
+
+ /**
+ * Sets the default Locale
+ */
+ public void setDefaultLocale(Locale locale) {
+
+ getLocaleConvertUtils().setDefaultLocale(locale);
+ }
+
+ /**
+ * Is the pattern to be applied localized
+ * (Indicate whether the pattern is localized or not)
+ */
+ public boolean getApplyLocalized() {
+
+ return getLocaleConvertUtils().getApplyLocalized();
+ }
+
+ /**
+ * Sets whether the pattern is applied localized
+ * (Indicate whether the pattern is localized or not)
+ */
+ public void setApplyLocalized(boolean newApplyLocalized) {
+
+ getLocaleConvertUtils().setApplyLocalized(newApplyLocalized);
+ }
+
+
+ // --------------------------------------------------------- Public Methods
+
+ /**
+ * Return the value of the specified locale-sensitive indexed property
+ * of the specified bean, as a String. The zero-relative index of the
+ * required value must be included (in square brackets) as a suffix to
+ * the property name, or <code>IllegalArgumentException</code> will be
+ * thrown.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name <code>propertyname[index]</code> of the property value
+ * to be extracted
+ * @param pattern The convertion pattern
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getIndexedProperty(
+ Object bean,
+ String name,
+ String pattern)
+ throws
+ IllegalAccessException,
+ InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getIndexedProperty(bean, name);
+ return getLocaleConvertUtils().convert(value, pattern);
+ }
+
+ /**
+ * Return the value of the specified locale-sensitive indexed property
+ * of the specified bean, as a String using the default convertion pattern of
+ * the corresponding {@link LocaleConverter}. The zero-relative index
+ * of the required value must be included (in square brackets) as a suffix
+ * to the property name, or <code>IllegalArgumentException</code> will be thrown.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name <code>propertyname[index]</code> of the property value
+ * to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getIndexedProperty(
+ Object bean,
+ String name)
+ throws
+ IllegalAccessException,
+ InvocationTargetException,
+ NoSuchMethodException {
+
+ return getIndexedProperty(bean, name, null);
+ }
+
+ /**
+ * Return the value of the specified locale-sensetive indexed property
+ * of the specified bean, as a String using the specified convertion pattern.
+ * The index is specified as a method parameter and
+ * must *not* be included in the property name expression
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Simple property name of the property value to be extracted
+ * @param index Index of the property value to be extracted
+ * @param pattern The convertion pattern
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getIndexedProperty(Object bean,
+ String name, int index, String pattern)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getIndexedProperty(bean, name, index);
+ return getLocaleConvertUtils().convert(value, pattern);
+ }
+
+ /**
+ * Return the value of the specified locale-sensetive indexed property
+ * of the specified bean, as a String using the default convertion pattern of
+ * the corresponding {@link LocaleConverter}.
+ * The index is specified as a method parameter and
+ * must *not* be included in the property name expression
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Simple property name of the property value to be extracted
+ * @param index Index of the property value to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getIndexedProperty(Object bean,
+ String name, int index)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+ return getIndexedProperty(bean, name, index, null);
+ }
+
+ /**
+ * Return the value of the specified simple locale-sensitive property
+ * of the specified bean, converted to a String using the specified
+ * convertion pattern.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Name of the property to be extracted
+ * @param pattern The convertion pattern
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getSimpleProperty(Object bean, String name, String pattern)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getSimpleProperty(bean, name);
+ return getLocaleConvertUtils().convert(value, pattern);
+ }
+
+ /**
+ * Return the value of the specified simple locale-sensitive property
+ * of the specified bean, converted to a String using the default
+ * convertion pattern of the corresponding {@link LocaleConverter}.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Name of the property to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getSimpleProperty(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return getSimpleProperty(bean, name, null);
+ }
+
+ /**
+ * Return the value of the specified mapped locale-sensitive property
+ * of the specified bean, as a String using the specified convertion pattern.
+ * The key is specified as a method parameter and must *not* be included in
+ * the property name expression.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Simple property name of the property value to be extracted
+ * @param key Lookup key of the property value to be extracted
+ * @param pattern The convertion pattern
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getMappedProperty(
+ Object bean,
+ String name,
+ String key,
+ String pattern)
+ throws
+ IllegalAccessException,
+ InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getMappedProperty(bean, name, key);
+ return getLocaleConvertUtils().convert(value, pattern);
+ }
+
+ /**
+ * Return the value of the specified mapped locale-sensitive property
+ * of the specified bean, as a String
+ * The key is specified as a method parameter and must *not* be included
+ * in the property name expression
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Simple property name of the property value to be extracted
+ * @param key Lookup key of the property value to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getMappedProperty(Object bean,
+ String name, String key)
+ throws IllegalAccessException, InvocationTargetException,
+ NoSuchMethodException {
+
+ return getMappedProperty(bean, name, key, null);
+ }
+
+
+ /**
+ * Return the value of the specified locale-sensitive mapped property
+ * of the specified bean, as a String using the specified pattern.
+ * The String-valued key of the required value
+ * must be included (in parentheses) as a suffix to
+ * the property name, or <code>IllegalArgumentException</code> will be
+ * thrown.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name <code>propertyname(index)</code> of the property value
+ * to be extracted
+ * @param pattern The convertion pattern
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getMappedPropertyLocale(
+ Object bean,
+ String name,
+ String pattern)
+ throws
+ IllegalAccessException,
+ InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getMappedProperty(bean, name);
+ return getLocaleConvertUtils().convert(value, pattern);
+ }
+
+
+ /**
+ * Return the value of the specified locale-sensitive mapped property
+ * of the specified bean, as a String using the default
+ * convertion pattern of the corresponding {@link LocaleConverter}.
+ * The String-valued key of the required value
+ * must be included (in parentheses) as a suffix to
+ * the property name, or <code>IllegalArgumentException</code> will be
+ * thrown.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name <code>propertyname(index)</code> of the property value
+ * to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getMappedProperty(Object bean, String name)
+ throws
+ IllegalAccessException,
+ InvocationTargetException,
+ NoSuchMethodException {
+
+ return getMappedPropertyLocale(bean, name, null);
+ }
+
+ /**
+ * Return the value of the (possibly nested) locale-sensitive property
+ * of the specified name, for the specified bean,
+ * as a String using the specified pattern.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Possibly nested name of the property to be extracted
+ * @param pattern The convertion pattern
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if a nested reference to a
+ * property returns null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getNestedProperty(
+ Object bean,
+ String name,
+ String pattern)
+ throws
+ IllegalAccessException,
+ InvocationTargetException,
+ NoSuchMethodException {
+
+ Object value = getPropertyUtils().getNestedProperty(bean, name);
+ return getLocaleConvertUtils().convert(value, pattern);
+ }
+
+ /**
+ * Return the value of the (possibly nested) locale-sensitive property
+ * of the specified name, for the specified bean, as a String using the default
+ * convertion pattern of the corresponding {@link LocaleConverter}.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Possibly nested name of the property to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception IllegalArgumentException if a nested reference to a
+ * property returns null
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getNestedProperty(Object bean, String name)
+ throws
+ IllegalAccessException,
+ InvocationTargetException,
+ NoSuchMethodException {
+
+ return getNestedProperty(bean, name, null);
+ }
+
+ /**
+ * Return the value of the specified locale-sensitive property
+ * of the specified bean, no matter which property reference
+ * format is used, as a String using the specified convertion pattern.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Possibly indexed and/or nested name of the property
+ * to be extracted
+ * @param pattern The convertion pattern
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getProperty(Object bean, String name, String pattern)
+ throws
+ IllegalAccessException,
+ InvocationTargetException,
+ NoSuchMethodException {
+
+ return getNestedProperty(bean, name, pattern);
+ }
+
+ /**
+ * Return the value of the specified locale-sensitive property
+ * of the specified bean, no matter which property reference
+ * format is used, as a String using the default
+ * convertion pattern of the corresponding {@link LocaleConverter}.
+ *
+ * @param bean Bean whose property is to be extracted
+ * @param name Possibly indexed and/or nested name of the property
+ * to be extracted
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ * @exception NoSuchMethodException if an accessor method for this
+ * propety cannot be found
+ */
+ public String getProperty(Object bean, String name)
+ throws
+ IllegalAccessException,
+ InvocationTargetException,
+ NoSuchMethodException {
+
+ return getNestedProperty(bean, name);
+ }
+
+ /**
+ * Set the specified locale-sensitive property value, performing type
+ * conversions as required to conform to the type of the destination property
+ * using the default convertion pattern of the corresponding {@link LocaleConverter}.
+ *
+ * @param bean Bean on which setting is to be performed
+ * @param name Property name (can be nested/indexed/mapped/combo)
+ * @param value Value to be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ public void setProperty(Object bean, String name, Object value)
+ throws
+ IllegalAccessException,
+ InvocationTargetException {
+
+ setProperty(bean, name, value, null);
+ }
+
+ /**
+ * Set the specified locale-sensitive property value, performing type
+ * conversions as required to conform to the type of the destination
+ * property using the specified convertion pattern.
+ *
+ * @param bean Bean on which setting is to be performed
+ * @param name Property name (can be nested/indexed/mapped/combo)
+ * @param value Value to be set
+ * @param pattern The convertion pattern
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ public void setProperty(
+ Object bean,
+ String name,
+ Object value,
+ String pattern)
+ throws
+ IllegalAccessException,
+ InvocationTargetException {
+
+ // Trace logging (if enabled)
+ if (log.isTraceEnabled()) {
+ StringBuffer sb = new StringBuffer(" setProperty(");
+ sb.append(bean);
+ sb.append(", ");
+ sb.append(name);
+ sb.append(", ");
+ if (value == null) {
+ sb.append("<NULL>");
+ }
+ else if (value instanceof String) {
+ sb.append((String) value);
+ }
+ else if (value instanceof String[]) {
+ String values[] = (String[]) value;
+ sb.append('[');
+ for (int i = 0; i < values.length; i++) {
+ if (i > 0) {
+ sb.append(',');
+ }
+ sb.append(values[i]);
+ }
+ sb.append(']');
+ }
+ else {
+ sb.append(value.toString());
+ }
+ sb.append(')');
+ log.trace(sb.toString());
+ }
+
+ Descriptor propInfo = calculate(bean, name);
+
+ if (propInfo != null) {
+ Class type = definePropertyType(propInfo.getTarget(), name, propInfo.getPropName());
+
+ if (type != null) {
+
+ Object newValue = convert(type, propInfo.getIndex(), value, pattern);
+ invokeSetter(propInfo.getTarget(), propInfo.getPropName(),
+ propInfo.getKey(), propInfo.getIndex(), newValue);
+ }
+ }
+ }
+
+ /**
+ * Calculate the property type.
+ *
+ * @param target The bean
+ * @param name The property name
+ * @param propName The Simple name of target property
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ protected Class definePropertyType(Object target, String name, String propName)
+ throws IllegalAccessException, InvocationTargetException {
+
+ Class type = null; // Java type of target property
+
+ if (target instanceof DynaBean) {
+ DynaClass dynaClass = ((DynaBean) target).getDynaClass();
+ DynaProperty dynaProperty = dynaClass.getDynaProperty(propName);
+ if (dynaProperty == null) {
+ return null; // Skip this property setter
+ }
+ type = dynaProperty.getType();
+ }
+ else {
+ PropertyDescriptor descriptor = null;
+ try {
+ descriptor =
+ getPropertyUtils().getPropertyDescriptor(target, name);
+ if (descriptor == null) {
+ return null; // Skip this property setter
+ }
+ }
+ catch (NoSuchMethodException e) {
+ return null; // Skip this property setter
+ }
+ if (descriptor instanceof MappedPropertyDescriptor) {
+ type = ((MappedPropertyDescriptor) descriptor).
+ getMappedPropertyType();
+ }
+ else if (descriptor instanceof IndexedPropertyDescriptor) {
+ type = ((IndexedPropertyDescriptor) descriptor).
+ getIndexedPropertyType();
+ }
+ else {
+ type = descriptor.getPropertyType();
+ }
+ }
+ return type;
+ }
+
+ /**
+ * Convert the specified value to the required type using the
+ * specified convertion pattern.
+ *
+ * @param type The Java type of target property
+ * @param index The indexed subscript value (if any)
+ * @param value The value to be converted
+ * @param pattern The convertion pattern
+ */
+ protected Object convert(Class type, int index, Object value, String pattern) {
+
+ if (log.isTraceEnabled()) {
+ log.trace("Converting value '" + value + "' to type:" + type);
+ }
+
+ Object newValue = null;
+
+ if (type.isArray() && (index < 0)) { // Scalar value into array
+ if (value instanceof String) {
+ String values[] = new String[1];
+ values[0] = (String) value;
+ newValue = getLocaleConvertUtils().convert((String[]) values, type, pattern);
+ }
+ else if (value instanceof String[]) {
+ newValue = getLocaleConvertUtils().convert((String[]) value, type, pattern);
+ }
+ else {
+ newValue = value;
+ }
+ }
+ else if (type.isArray()) { // Indexed value into array
+ if (value instanceof String) {
+ newValue = getLocaleConvertUtils().convert((String) value,
+ type.getComponentType(), pattern);
+ }
+ else if (value instanceof String[]) {
+ newValue = getLocaleConvertUtils().convert(((String[]) value)[0],
+ type.getComponentType(), pattern);
+ }
+ else {
+ newValue = value;
+ }
+ }
+ else { // Value into scalar
+ if (value instanceof String) {
+ newValue = getLocaleConvertUtils().convert((String) value, type, pattern);
+ }
+ else if (value instanceof String[]) {
+ newValue = getLocaleConvertUtils().convert(((String[]) value)[0],
+ type, pattern);
+ }
+ else {
+ newValue = value;
+ }
+ }
+ return newValue;
+ }
+
+ /**
+ * Convert the specified value to the required type.
+ *
+ * @param type The Java type of target property
+ * @param index The indexed subscript value (if any)
+ * @param value The value to be converted
+ */
+ protected Object convert(Class type, int index, Object value) {
+
+ Object newValue = null;
+
+ if (type.isArray() && (index < 0)) { // Scalar value into array
+ if (value instanceof String) {
+ String values[] = new String[1];
+ values[0] = (String) value;
+ newValue = ConvertUtils.convert((String[]) values, type);
+ }
+ else if (value instanceof String[]) {
+ newValue = ConvertUtils.convert((String[]) value, type);
+ }
+ else {
+ newValue = value;
+ }
+ }
+ else if (type.isArray()) { // Indexed value into array
+ if (value instanceof String) {
+ newValue = ConvertUtils.convert((String) value,
+ type.getComponentType());
+ }
+ else if (value instanceof String[]) {
+ newValue = ConvertUtils.convert(((String[]) value)[0],
+ type.getComponentType());
+ }
+ else {
+ newValue = value;
+ }
+ }
+ else { // Value into scalar
+ if (value instanceof String) {
+ newValue = ConvertUtils.convert((String) value, type);
+ }
+ else if (value instanceof String[]) {
+ newValue = ConvertUtils.convert(((String[]) value)[0],
+ type);
+ }
+ else {
+ newValue = value;
+ }
+ }
+ return newValue;
+ }
+
+ /**
+ * Invoke the setter method.
+ *
+ * @param target The bean
+ * @param propName The Simple name of target property
+ * @param key The Mapped key value (if any)
+ * @param index The indexed subscript value (if any)
+ * @param newValue The value to be set
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ protected void invokeSetter(Object target, String propName, String key, int index, Object newValue)
+ throws IllegalAccessException, InvocationTargetException {
+
+ try {
+ if (index >= 0) {
+ getPropertyUtils().setIndexedProperty(target, propName,
+ index, newValue);
+ }
+ else if (key != null) {
+ getPropertyUtils().setMappedProperty(target, propName,
+ key, newValue);
+ }
+ else {
+ getPropertyUtils().setProperty(target, propName, newValue);
+ }
+ }
+ catch (NoSuchMethodException e) {
+ throw new InvocationTargetException
+ (e, "Cannot set " + propName);
+ }
+ }
+
+ /**
+ * Resolve any nested expression to get the actual target bean.
+ *
+ * @param bean The bean
+ * @param name The property name
+ *
+ * @exception IllegalAccessException if the caller does not have
+ * access to the property accessor method
+ * @exception InvocationTargetException if the property accessor method
+ * throws an exception
+ */
+ protected Descriptor calculate(Object bean, String name)
+ throws IllegalAccessException, InvocationTargetException {
+
+ String propName = null; // Simple name of target property
+ int index = -1; // Indexed subscript value (if any)
+ String key = null; // Mapped key value (if any)
+
+ Object target = bean;
+ int delim = name.lastIndexOf(PropertyUtils.NESTED_DELIM);
+ if (delim >= 0) {
+ try {
+ target =
+ getPropertyUtils().getProperty(bean, name.substring(0, delim));
+ }
+ catch (NoSuchMethodException e) {
+ return null; // Skip this property setter
+ }
+ name = name.substring(delim + 1);
+ if (log.isTraceEnabled()) {
+ log.trace(" Target bean = " + target);
+ log.trace(" Target name = " + name);
+ }
+ }
+
+ // Calculate the property name, index, and key values
+ propName = name;
+ int i = propName.indexOf(PropertyUtils.INDEXED_DELIM);
+ if (i >= 0) {
+ int k = propName.indexOf(PropertyUtils.INDEXED_DELIM2);
+ try {
+ index =
+ Integer.parseInt(propName.substring(i + 1, k));
+ }
+ catch (NumberFormatException e) {
+ ;
+ }
+ propName = propName.substring(0, i);
+ }
+ int j = propName.indexOf(PropertyUtils.MAPPED_DELIM);
+ if (j >= 0) {
+ int k = propName.indexOf(PropertyUtils.MAPPED_DELIM2);
+ try {
+ key = propName.substring(j + 1, k);
+ }
+ catch (IndexOutOfBoundsException e) {
+ ;
+ }
+ propName = propName.substring(0, j);
+ }
+ return new Descriptor(target, name, propName, key, index);
+ }
+
+ protected class Descriptor {
+
+ private int index = -1; // Indexed subscript value (if any)
+ private String name;
+ private String propName; // Simple name of target property
+ private String key; // Mapped key value (if any)
+ private Object target;
+
+ public Descriptor(Object target, String name, String propName, String key, int index) {
+
+ setTarget(target);
+ setName(name);
+ setPropName(propName);
+ setKey(key);
+ setIndex(index);
+ }
+
+ public Object getTarget() {
+ return target;
+ }
+
+ public void setTarget(Object target) {
+ this.target = target;
+ }
+
+ public String getKey() {
+ return key;
+ }
+
+ public void setKey(String key) {
+ this.key = key;
+ }
+
+ public int getIndex() {
+ return index;
+ }
+
+ public void setIndex(int index) {
+ this.index = index;
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public void setName(String name) {
+ this.name = name;
+ }
+
+ public String getPropName() {
+ return propName;
+ }
+
+ public void setPropName(String propName) {
+ this.propName = propName;
+ }
+ }
+}
+
+
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/LocaleConvertUtils.java b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleConvertUtils.java
new file mode 100755
index 0000000..ee0ad86
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleConvertUtils.java
@@ -0,0 +1,291 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale;
+
+import org.apache.commons.collections.FastHashMap;
+
+import java.util.Locale;
+
+/**
+ * <p>Utility methods for converting locale-sensitive String scalar values to objects of the
+ * specified Class, String arrays to arrays of the specified Class and
+ * object to locale-sensitive String scalar value.</p>
+ *
+ * <p>The implementations for these method are provided by {@link LocaleConvertUtilsBean}.
+ * These static utility method use the default instance. More sophisticated can be provided
+ * by using a <code>LocaleConvertUtilsBean</code> instance.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+public class LocaleConvertUtils {
+
+ // ----------------------------------------------------- Instance Variables
+
+ /**
+ * <p>Gets the <code>Locale</code> which will be used when
+ * no <code>Locale</code> is passed to a method.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#getDefaultLocale()
+ */
+ public static Locale getDefaultLocale() {
+
+ return LocaleConvertUtilsBean.getInstance().getDefaultLocale();
+ }
+
+ /**
+ * <p>Sets the <code>Locale</code> which will be used when
+ * no <code>Locale</code> is passed to a method.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#setDefaultLocale(Locale)
+ */
+ public static void setDefaultLocale(Locale locale) {
+
+ LocaleConvertUtilsBean.getInstance().setDefaultLocale(locale);
+ }
+
+ /**
+ * <p>Gets applyLocalized.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#getApplyLocalized()
+ */
+ public static boolean getApplyLocalized() {
+ return LocaleConvertUtilsBean.getInstance().getApplyLocalized();
+ }
+
+ /**
+ * <p>Sets applyLocalized.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#setApplyLocalized(boolean)
+ */
+ public static void setApplyLocalized(boolean newApplyLocalized) {
+ LocaleConvertUtilsBean.getInstance().setApplyLocalized(newApplyLocalized);
+ }
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * <p>Convert the specified locale-sensitive value into a String.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#convert(Object)
+ */
+ public static String convert(Object value) {
+ return LocaleConvertUtilsBean.getInstance().convert(value);
+ }
+
+ /**
+ * <p>Convert the specified locale-sensitive value into a String
+ * using the convertion pattern.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#convert(Object, String)
+ */
+ public static String convert(Object value, String pattern) {
+ return LocaleConvertUtilsBean.getInstance().convert(value, pattern);
+ }
+
+ /**
+ * <p>Convert the specified locale-sensitive value into a String
+ * using the paticular convertion pattern.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#convert(Object, Locale, String)
+ */
+ public static String convert(Object value, Locale locale, String pattern) {
+
+ return LocaleConvertUtilsBean.getInstance().convert(value, locale, pattern);
+ }
+
+ /**
+ * <p>Convert the specified value to an object of the specified class (if
+ * possible). Otherwise, return a String representation of the value.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#convert(String, Class)
+ */
+ public static Object convert(String value, Class clazz) {
+
+ return LocaleConvertUtilsBean.getInstance().convert(value, clazz);
+ }
+
+ /**
+ * <p>Convert the specified value to an object of the specified class (if
+ * possible) using the convertion pattern. Otherwise, return a String
+ * representation of the value.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#convert(String, Class, String)
+ */
+ public static Object convert(String value, Class clazz, String pattern) {
+
+ return LocaleConvertUtilsBean.getInstance().convert(value, clazz, pattern);
+ }
+
+ /**
+ * <p>Convert the specified value to an object of the specified class (if
+ * possible) using the convertion pattern. Otherwise, return a String
+ * representation of the value.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#convert(String, Class, Locale, String)
+ */
+ public static Object convert(String value, Class clazz, Locale locale, String pattern) {
+
+ return LocaleConvertUtilsBean.getInstance().convert(value, clazz, locale, pattern);
+ }
+
+ /**
+ * <p>Convert an array of specified values to an array of objects of the
+ * specified class (if possible) using the convertion pattern.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#convert(String[], Class, String)
+ */
+ public static Object convert(String values[], Class clazz, String pattern) {
+
+ return LocaleConvertUtilsBean.getInstance().convert(values, clazz, pattern);
+ }
+
+ /**
+ * <p>Convert an array of specified values to an array of objects of the
+ * specified class (if possible).</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#convert(String[], Class)
+ */
+ public static Object convert(String values[], Class clazz) {
+
+ return LocaleConvertUtilsBean.getInstance().convert(values, clazz);
+ }
+
+ /**
+ * <p>Convert an array of specified values to an array of objects of the
+ * specified class (if possible) using the convertion pattern.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#convert(String[], Class, Locale, String)
+ */
+ public static Object convert(String values[], Class clazz, Locale locale, String pattern) {
+
+ return LocaleConvertUtilsBean.getInstance().convert(values, clazz, locale, pattern);
+ }
+
+ /**
+ * <p>Register a custom {@link LocaleConverter} for the specified destination
+ * <code>Class</code>, replacing any previously registered converter.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#register(LocaleConverter, Class, Locale)
+ */
+ public static void register(LocaleConverter converter, Class clazz, Locale locale) {
+
+ LocaleConvertUtilsBean.getInstance().register(converter, clazz, locale);
+ }
+
+ /**
+ * <p>Remove any registered {@link LocaleConverter}.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#deregister()
+ */
+ public static void deregister() {
+
+ LocaleConvertUtilsBean.getInstance().deregister();
+ }
+
+
+ /**
+ * <p>Remove any registered {@link LocaleConverter} for the specified locale.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#deregister(Locale)
+ */
+ public static void deregister(Locale locale) {
+
+ LocaleConvertUtilsBean.getInstance().deregister(locale);
+ }
+
+
+ /**
+ * <p>Remove any registered {@link LocaleConverter} for the specified locale and Class.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#deregister(Class, Locale)
+ */
+ public static void deregister(Class clazz, Locale locale) {
+
+ LocaleConvertUtilsBean.getInstance().deregister(clazz, locale);
+ }
+
+ /**
+ * <p>Look up and return any registered {@link LocaleConverter} for the specified
+ * destination class and locale; if there is no registered Converter, return
+ * <code>null</code>.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#lookup(Class, Locale)
+ */
+ public static LocaleConverter lookup(Class clazz, Locale locale) {
+
+ return LocaleConvertUtilsBean.getInstance().lookup(clazz, locale);
+ }
+
+ /**
+ * <p>Look up and return any registered FastHashMap instance for the specified locale.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#lookup(Locale)
+ */
+ protected static FastHashMap lookup(Locale locale) {
+ return LocaleConvertUtilsBean.getInstance().lookup(locale);
+ }
+
+ /**
+ * <p>Create all {@link LocaleConverter} types for specified locale.</p>
+ *
+ * <p>For more details see <code>LocaleConvertUtilsBean</code></p>
+ *
+ * @see LocaleConvertUtilsBean#create(Locale)
+ */
+ protected static FastHashMap create(Locale locale) {
+
+ return LocaleConvertUtilsBean.getInstance().create(locale);
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/LocaleConvertUtilsBean.java b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleConvertUtilsBean.java
new file mode 100644
index 0000000..796c81f
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleConvertUtilsBean.java
@@ -0,0 +1,456 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale;
+
+import org.apache.commons.beanutils.locale.converters.*;
+import org.apache.commons.beanutils.Converter;
+import org.apache.commons.collections.FastHashMap;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.lang.reflect.Array;
+import java.math.BigDecimal;
+import java.math.BigInteger;
+import java.sql.Date;
+import java.sql.Time;
+import java.sql.Timestamp;
+import java.util.Locale;
+
+/**
+ * <p>Utility methods for converting locale-sensitive String scalar values to objects of the
+ * specified Class, String arrays to arrays of the specified Class and
+ * object to locale-sensitive String scalar value.</p>
+ *
+ * <p>This class provides the implementations used by the static utility methods in
+ * {@link LocaleConvertUtils}.</p>
+ *
+ * <p>The actual {@link LocaleConverter} instance to be used
+ * can be registered for each possible destination Class. Unless you override them, standard
+ * {@link LocaleConverter} instances are provided for all of the following
+ * destination Classes:</p>
+ * <ul>
+ * <li>java.lang.BigDecimal</li>
+ * <li>java.lang.BigInteger</li>
+ * <li>byte and java.lang.Byte</li>
+ * <li>double and java.lang.Double</li>
+ * <li>float and java.lang.Float</li>
+ * <li>int and java.lang.Integer</li>
+ * <li>long and java.lang.Long</li>
+ * <li>short and java.lang.Short</li>
+ * <li>java.lang.String</li>
+ * <li>java.sql.Date</li>
+ * <li>java.sql.Time</li>
+ * <li>java.sql.Timestamp</li>
+ * </ul>
+ *
+ * <p>For backwards compatibility, the standard locale converters
+ * for primitive types (and the corresponding wrapper classes).
+ *
+ * If you prefer to have another {@link LocaleConverter}
+ * thrown instead, replace the standard {@link LocaleConverter} instances
+ * with ones created with the one of the appropriate constructors.
+ *
+ * It's important that {@link LocaleConverter} should be registered for
+ * the specified locale and Class (or primitive type).
+ *
+ * @author Yauheny Mikulski
+ * @since 1.7
+ */
+public class LocaleConvertUtilsBean {
+
+ /**
+ * Gets singleton instance.
+ * This is the same as the instance used by the default {@link LocaleBeanUtilsBean} singleton.
+ */
+ public static LocaleConvertUtilsBean getInstance() {
+ return LocaleBeanUtilsBean.getLocaleBeanUtilsInstance().getLocaleConvertUtils();
+ }
+
+ // ----------------------------------------------------- Instance Variables
+
+ /** The locale - default for convertion. */
+ private Locale defaultLocale = Locale.getDefault();
+
+ /** Indicate whether the pattern is localized or not */
+ private boolean applyLocalized = false;
+
+ /** The <code>Log</code> instance for this class. */
+ private Log log = LogFactory.getLog(LocaleConvertUtils.class);
+
+ /** Every entry of the mapConverters is:
+ * key = locale
+ * value = FastHashMap of converters for the certain locale.
+ */
+ private FastHashMap mapConverters = new FastHashMap();
+
+ // --------------------------------------------------------- Constructors
+
+ /**
+ * Makes the state by default (deregisters all converters for all locales)
+ * and then registers default locale converters.
+ */
+ public LocaleConvertUtilsBean() {
+ deregister();
+ }
+
+ // --------------------------------------------------------- Properties
+
+ /**
+ * getter for defaultLocale
+ */
+ public Locale getDefaultLocale() {
+
+ return defaultLocale;
+ }
+
+ /**
+ * setter for defaultLocale
+ */
+ public void setDefaultLocale(Locale locale) {
+
+ if (locale == null) {
+ defaultLocale = Locale.getDefault();
+ }
+ else {
+ defaultLocale = locale;
+ }
+ }
+
+ /**
+ * getter for applyLocalized
+ */
+ public boolean getApplyLocalized() {
+ return applyLocalized;
+ }
+
+ /**
+ * setter for applyLocalized
+ */
+ public void setApplyLocalized(boolean newApplyLocalized) {
+ applyLocalized = newApplyLocalized;
+ }
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * Convert the specified locale-sensitive value into a String.
+ *
+ * @param value The Value to be converted
+ *
+ * @exception org.apache.commons.beanutils.ConversionException if thrown by an underlying Converter
+ */
+ public String convert(Object value) {
+ return convert(value, defaultLocale, null);
+ }
+
+ /**
+ * Convert the specified locale-sensitive value into a String
+ * using the convertion pattern.
+ *
+ * @param value The Value to be converted
+ * @param pattern The convertion pattern
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public String convert(Object value, String pattern) {
+ return convert(value, defaultLocale, pattern);
+ }
+
+ /**
+ * Convert the specified locale-sensitive value into a String
+ * using the paticular convertion pattern.
+ *
+ * @param value The Value to be converted
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public String convert(Object value, Locale locale, String pattern) {
+
+ LocaleConverter converter = lookup(String.class, locale);
+
+ return (String) converter.convert(String.class, value, pattern);
+ }
+
+ /**
+ * Convert the specified value to an object of the specified class (if
+ * possible). Otherwise, return a String representation of the value.
+ *
+ * @param value The String scalar value to be converted
+ * @param clazz The Data type to which this value should be converted.
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public Object convert(String value, Class clazz) {
+
+ return convert(value, clazz, defaultLocale, null);
+ }
+
+ /**
+ * Convert the specified value to an object of the specified class (if
+ * possible) using the convertion pattern. Otherwise, return a String
+ * representation of the value.
+ *
+ * @param value The String scalar value to be converted
+ * @param clazz The Data type to which this value should be converted.
+ * @param pattern The convertion pattern
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public Object convert(String value, Class clazz, String pattern) {
+
+ return convert(value, clazz, defaultLocale, pattern);
+ }
+
+ /**
+ * Convert the specified value to an object of the specified class (if
+ * possible) using the convertion pattern. Otherwise, return a String
+ * representation of the value.
+ *
+ * @param value The String scalar value to be converted
+ * @param clazz The Data type to which this value should be converted.
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public Object convert(String value, Class clazz, Locale locale, String pattern) {
+
+ if (log.isDebugEnabled()) {
+ log.debug("Convert string " + value + " to class " +
+ clazz.getName() + " using " + locale.toString() +
+ " locale and " + pattern + " pattern");
+ }
+
+ LocaleConverter converter = lookup(clazz, locale);
+
+ if (converter == null) {
+ converter = (LocaleConverter) lookup(String.class, locale);
+ }
+ if (log.isTraceEnabled()) {
+ log.trace(" Using converter " + converter);
+ }
+
+ return (converter.convert(clazz, value, pattern));
+ }
+
+ /**
+ * Convert an array of specified values to an array of objects of the
+ * specified class (if possible) using the convertion pattern.
+ *
+ * @param values Value to be converted (may be null)
+ * @param clazz Java array or element class to be converted to
+ * @param pattern The convertion pattern
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public Object convert(String values[], Class clazz, String pattern) {
+
+ return convert(values, clazz, getDefaultLocale(), pattern);
+ }
+
+ /**
+ * Convert an array of specified values to an array of objects of the
+ * specified class (if possible) .
+ *
+ * @param values Value to be converted (may be null)
+ * @param clazz Java array or element class to be converted to
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public Object convert(String values[], Class clazz) {
+
+ return convert(values, clazz, getDefaultLocale(), null);
+ }
+
+ /**
+ * Convert an array of specified values to an array of objects of the
+ * specified class (if possible) using the convertion pattern.
+ *
+ * @param values Value to be converted (may be null)
+ * @param clazz Java array or element class to be converted to
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ *
+ * @exception ConversionException if thrown by an underlying Converter
+ */
+ public Object convert(String values[], Class clazz, Locale locale, String pattern) {
+
+ Class type = clazz;
+ if (clazz.isArray()) {
+ type = clazz.getComponentType();
+ }
+ if (log.isDebugEnabled()) {
+ log.debug("Convert String[" + values.length + "] to class " +
+ type.getName() + "[] using " + locale.toString() +
+ " locale and " + pattern + " pattern");
+ }
+
+ Object array = Array.newInstance(type, values.length);
+ for (int i = 0; i < values.length; i++) {
+ Array.set(array, i, convert(values[i], type, locale, pattern));
+ }
+
+ return (array);
+ }
+
+ /**
+ * Register a custom {@link LocaleConverter} for the specified destination
+ * <code>Class</code>, replacing any previously registered converter.
+ *
+ * @param converter The LocaleConverter to be registered
+ * @param clazz The Destination class for conversions performed by this
+ * Converter
+ * @param locale The locale
+ */
+ public void register(LocaleConverter converter, Class clazz, Locale locale) {
+
+ lookup(locale).put(clazz, converter);
+ }
+
+ /**
+ * Remove any registered {@link LocaleConverter}.
+ */
+ public void deregister() {
+
+ FastHashMap defaultConverter = lookup(defaultLocale);
+
+ mapConverters.setFast(false);
+
+ mapConverters.clear();
+ mapConverters.put(defaultLocale, defaultConverter);
+
+ mapConverters.setFast(true);
+ }
+
+
+ /**
+ * Remove any registered {@link LocaleConverter} for the specified locale
+ *
+ * @param locale The locale
+ */
+ public void deregister(Locale locale) {
+
+ mapConverters.remove(locale);
+ }
+
+
+ /**
+ * Remove any registered {@link LocaleConverter} for the specified locale and Class.
+ *
+ * @param clazz Class for which to remove a registered Converter
+ * @param locale The locale
+ */
+ public void deregister(Class clazz, Locale locale) {
+
+ lookup(locale).remove(clazz);
+ }
+
+ /**
+ * Look up and return any registered {@link LocaleConverter} for the specified
+ * destination class and locale; if there is no registered Converter, return
+ * <code>null</code>.
+ *
+ * @param clazz Class for which to return a registered Converter
+ * @param locale The Locale
+ */
+ public LocaleConverter lookup(Class clazz, Locale locale) {
+
+ LocaleConverter converter = (LocaleConverter) lookup(locale).get(clazz);
+
+ if (log.isTraceEnabled()) {
+ log.trace("LocaleConverter:" + converter);
+ }
+
+ return converter;
+ }
+
+ /**
+ * Look up and return any registered FastHashMap instance for the specified locale;
+ * if there is no registered one, return <code>null</code>.
+ *
+ * @param locale The Locale
+ * @return The FastHashMap instance contains the all {@link LocaleConverter} types for
+ * the specified locale.
+ */
+ protected FastHashMap lookup(Locale locale) {
+ FastHashMap localeConverters;
+
+ if (locale == null) {
+ localeConverters = (FastHashMap) mapConverters.get(defaultLocale);
+ }
+ else {
+ localeConverters = (FastHashMap) mapConverters.get(locale);
+
+ if (localeConverters == null) {
+ localeConverters = create(locale);
+ mapConverters.put(locale, localeConverters);
+ }
+ }
+
+ return localeConverters;
+ }
+
+ /**
+ * Create all {@link LocaleConverter} types for specified locale.
+ *
+ * @param locale The Locale
+ * @return The FastHashMap instance contains the all {@link LocaleConverter} types
+ * for the specified locale.
+ */
+ protected FastHashMap create(Locale locale) {
+
+ FastHashMap converter = new FastHashMap();
+ converter.setFast(false);
+
+ converter.put(BigDecimal.class, new BigDecimalLocaleConverter(locale, applyLocalized));
+ converter.put(BigInteger.class, new BigIntegerLocaleConverter(locale, applyLocalized));
+
+ converter.put(Byte.class, new ByteLocaleConverter(locale, applyLocalized));
+ converter.put(Byte.TYPE, new ByteLocaleConverter(locale, applyLocalized));
+
+ converter.put(Double.class, new DoubleLocaleConverter(locale, applyLocalized));
+ converter.put(Double.TYPE, new DoubleLocaleConverter(locale, applyLocalized));
+
+ converter.put(Float.class, new FloatLocaleConverter(locale, applyLocalized));
+ converter.put(Float.TYPE, new FloatLocaleConverter(locale, applyLocalized));
+
+ converter.put(Integer.class, new IntegerLocaleConverter(locale, applyLocalized));
+ converter.put(Integer.TYPE, new IntegerLocaleConverter(locale, applyLocalized));
+
+ converter.put(Long.class, new LongLocaleConverter(locale, applyLocalized));
+ converter.put(Long.TYPE, new LongLocaleConverter(locale, applyLocalized));
+
+ converter.put(Short.class, new ShortLocaleConverter(locale, applyLocalized));
+ converter.put(Short.TYPE, new ShortLocaleConverter(locale, applyLocalized));
+
+ converter.put(String.class, new StringLocaleConverter(locale, applyLocalized));
+
+ // conversion format patterns of java.sql.* types should correspond to default
+ // behaviour of toString and valueOf methods of these classes
+ converter.put(java.sql.Date.class, new SqlDateLocaleConverter(locale, "yyyy-MM-dd"));
+ converter.put(java.sql.Time.class, new SqlTimeLocaleConverter(locale, "HH:mm:ss"));
+ converter.put( java.sql.Timestamp.class,
+ new SqlTimestampLocaleConverter(locale, "yyyy-MM-dd HH:mm:ss.S")
+ );
+
+ converter.setFast(true);
+
+ return converter;
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/LocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleConverter.java
new file mode 100755
index 0000000..ba757c2
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/LocaleConverter.java
@@ -0,0 +1,45 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale;
+
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * <p>General purpose locale-sensitive data type converter that can be registered and used
+ * within the BeanUtils package to manage the conversion of objects from
+ * one type to another.
+ *
+ * @author Yauheny Mikulski
+ */
+
+public interface LocaleConverter extends Converter {
+
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type.
+ *
+ * @param type Data type to which this value should be converted
+ * @param value The input value to be converted
+ * @param pattern The user-defined pattern is used for the input object formatting.
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ public Object convert(Class type, Object value, String pattern);
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/BigDecimalLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/BigDecimalLocaleConverter.java
new file mode 100755
index 0000000..2f4629b
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/BigDecimalLocaleConverter.java
@@ -0,0 +1,198 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import java.util.Locale;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.math.BigDecimal</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class BigDecimalLocaleConverter extends DecimalLocaleConverter {
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public BigDecimalLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigDecimalLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public BigDecimalLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigDecimalLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public BigDecimalLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigDecimalLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public BigDecimalLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigDecimalLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public BigDecimalLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigDecimalLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public BigDecimalLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigDecimalLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern);
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/BigIntegerLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/BigIntegerLocaleConverter.java
new file mode 100755
index 0000000..9806888
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/BigIntegerLocaleConverter.java
@@ -0,0 +1,198 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import java.util.Locale;
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.math.BigInteger</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class BigIntegerLocaleConverter extends DecimalLocaleConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public BigIntegerLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigIntegerLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public BigIntegerLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigIntegerLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public BigIntegerLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigIntegerLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public BigIntegerLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigIntegerLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public BigIntegerLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigIntegerLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public BigIntegerLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public BigIntegerLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern);
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/ByteLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/ByteLocaleConverter.java
new file mode 100755
index 0000000..187b236
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/ByteLocaleConverter.java
@@ -0,0 +1,221 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import org.apache.commons.beanutils.ConversionException;
+
+import java.util.Locale;
+import java.text.ParseException;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.lang.Byte</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class ByteLocaleConverter extends DecimalLocaleConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public ByteLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ByteLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public ByteLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ByteLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public ByteLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ByteLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public ByteLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ByteLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public ByteLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ByteLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public ByteLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ByteLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern);
+ }
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type. This method will return values of type Byte.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception org.apache.commons.beanutils.ConversionException if conversion cannot be performed
+ * successfully
+ */
+ protected Object parse(Object value, String pattern) throws ParseException {
+ final Number parsed = (Number) super.parse(value, pattern);
+ if (parsed.longValue() != parsed.byteValue()) {
+ throw new ConversionException("Supplied number is not of type Byte: " + parsed.longValue());
+ }
+ // now returns property Byte
+ return new Byte(parsed.byteValue());
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/DateLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/DateLocaleConverter.java
new file mode 100755
index 0000000..7bf77d6
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/DateLocaleConverter.java
@@ -0,0 +1,279 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import org.apache.commons.beanutils.locale.BaseLocaleConverter;
+import org.apache.commons.logging.LogFactory;
+import org.apache.commons.logging.Log;
+
+import java.text.ParseException;
+import java.text.ParsePosition;
+import java.text.SimpleDateFormat;
+import java.util.Locale;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.util.Date</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ * @author Michael Szlapa
+ */
+
+public class DateLocaleConverter extends BaseLocaleConverter {
+
+ // ----------------------------------------------------- Instance Variables
+
+ /** All logging goes through this logger */
+ private static Log log = LogFactory.getLog(DateLocaleConverter.class);
+
+ /** Should the date conversion be lenient? */
+ boolean isLenient = false;
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public DateLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DateLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public DateLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DateLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public DateLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DateLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public DateLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DateLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public DateLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DateLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public DateLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DateLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern, locPattern);
+ }
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * Returns whether date formatting is lenient.
+ *
+ * @return true if the <code>DateFormat</code> used for formatting is lenient
+ * @see java.text.DateFormat#isLenient
+ */
+ public boolean isLenient() {
+ return isLenient;
+ }
+
+ /**
+ * Specify whether or not date-time parsing should be lenient.
+ *
+ * @param lenient true if the <code>DateFormat</code> used for formatting should be lenient
+ * @see java.text.DateFormat#setLenient
+ */
+ public void setLenient(boolean lenient) {
+ isLenient = lenient;
+ }
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception org.apache.commons.beanutils.ConversionException if conversion cannot be performed
+ * successfully
+ */
+ protected Object parse(Object value, String pattern) throws ParseException {
+ SimpleDateFormat formatter = getFormatter(pattern, locale);
+ if (locPattern) {
+ formatter.applyLocalizedPattern(pattern);
+ }
+ else {
+ formatter.applyPattern(pattern);
+ }
+ return formatter.parse((String) value);
+ }
+
+ /**
+ * Gets an appropriate <code>SimpleDateFormat</code> for given locale,
+ * default Date format pattern is not provided.
+ */
+ private SimpleDateFormat getFormatter(String pattern, Locale locale) {
+ // This method is a fix for null pattern, which would cause
+ // Null pointer exception when applied
+ // Note: that many constructors default the pattern to null,
+ // so it only makes sense to handle nulls gracefully
+ if(pattern == null) {
+ pattern = locPattern ?
+ new SimpleDateFormat().toLocalizedPattern() : new SimpleDateFormat().toPattern();
+ log.warn("Null pattern was provided, defaulting to: " + pattern);
+ }
+ SimpleDateFormat format = new SimpleDateFormat(pattern, locale);
+ format.setLenient(isLenient);
+ return format;
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/DecimalLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/DecimalLocaleConverter.java
new file mode 100755
index 0000000..3395d54
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/DecimalLocaleConverter.java
@@ -0,0 +1,242 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import org.apache.commons.beanutils.locale.BaseLocaleConverter;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.text.DecimalFormat;
+import java.text.ParseException;
+import java.util.Locale;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.lang.Decimal</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ * @author Yoav Shapira
+ * @since 1.7
+ */
+
+public class DecimalLocaleConverter extends BaseLocaleConverter {
+
+
+ // ----------------------------------------------------- Instance Variables
+
+ /** All logging goes through this logger */
+ private static Log log = LogFactory.getLog(DecimalLocaleConverter.class);
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public DecimalLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DecimalLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public DecimalLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DecimalLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public DecimalLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DecimalLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ this(null, locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public DecimalLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DecimalLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public DecimalLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DecimalLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public DecimalLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DecimalLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern, locPattern);
+
+ }
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ protected Object parse(Object value, String pattern) throws ParseException {
+ // DecimalFormat is not thread safe so best to construct one each time
+ DecimalFormat formatter = (DecimalFormat) DecimalFormat.getInstance(locale);
+ // if some constructors default pattern to null, it makes only sense to handle null pattern gracefully
+ if (pattern != null) {
+ if (locPattern) {
+ formatter.applyLocalizedPattern(pattern);
+ } else {
+ formatter.applyPattern(pattern);
+ }
+ } else {
+ log.warn("No pattern provided, using default.");
+ }
+
+ return formatter.parse((String) value);
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/DoubleLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/DoubleLocaleConverter.java
new file mode 100755
index 0000000..a3dcdfe
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/DoubleLocaleConverter.java
@@ -0,0 +1,214 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+
+import java.text.ParseException;
+import java.util.Locale;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.lang.Double</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class DoubleLocaleConverter extends DecimalLocaleConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public DoubleLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DoubleLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public DoubleLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DoubleLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public DoubleLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DoubleLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public DoubleLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DoubleLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public DoubleLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DoubleLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public DoubleLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public DoubleLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern);
+ }
+
+ protected Object parse(Object value, String pattern) throws ParseException {
+ Number result = (Number) super.parse(value, pattern);
+ if (result instanceof Long) {
+ return new Double(result.doubleValue());
+ } else {
+ return (result);
+ }
+ }
+
+
+}
+
+
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/FloatLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/FloatLocaleConverter.java
new file mode 100755
index 0000000..de6c10b
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/FloatLocaleConverter.java
@@ -0,0 +1,221 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import org.apache.commons.beanutils.ConversionException;
+
+import java.util.Locale;
+import java.text.ParseException;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.math.BigDecimal</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class FloatLocaleConverter extends DecimalLocaleConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public FloatLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public FloatLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public FloatLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public FloatLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public FloatLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public FloatLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public FloatLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public FloatLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public FloatLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public FloatLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public FloatLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public FloatLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern);
+ }
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type. This method will return Float value or throw exception if value
+ * can not be stored in the Float.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ protected Object parse(Object value, String pattern) throws ParseException {
+ final Number parsed = (Number) super.parse(value, pattern);
+ if( Math.abs(parsed.doubleValue() - parsed.floatValue()) > parsed.floatValue() * 0.00001 ) {
+ throw new ConversionException("Suplied number is not of type Float: "+parsed.longValue());
+ }
+ return new Float(parsed.floatValue()); // unlike superclass it returns Float type
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/IntegerLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/IntegerLocaleConverter.java
new file mode 100755
index 0000000..96820de
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/IntegerLocaleConverter.java
@@ -0,0 +1,221 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import org.apache.commons.beanutils.ConversionException;
+
+import java.util.Locale;
+import java.text.ParseException;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.lang.Integer</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class IntegerLocaleConverter extends DecimalLocaleConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+
+ public IntegerLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public IntegerLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public IntegerLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public IntegerLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public IntegerLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public IntegerLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public IntegerLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public IntegerLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public IntegerLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public IntegerLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public IntegerLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public IntegerLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern, locPattern);
+ }
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type. This method will return Integer type.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ protected Object parse(Object value, String pattern) throws ParseException {
+ final Number parsed = (Number) super.parse(value, pattern);
+ if (parsed.longValue() != parsed.intValue()) {
+ throw new ConversionException("Suplied number is not of type Integer: " + parsed.longValue());
+ }
+ return new Integer(parsed.intValue()); // unlike superclass it will return proper Integer
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/LongLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/LongLocaleConverter.java
new file mode 100755
index 0000000..edc4c86
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/LongLocaleConverter.java
@@ -0,0 +1,199 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import java.util.Locale;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.lang.Long</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class LongLocaleConverter extends DecimalLocaleConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public LongLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public LongLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public LongLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public LongLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public LongLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public LongLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public LongLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public LongLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public LongLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public LongLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public LongLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public LongLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern, locPattern);
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/ShortLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/ShortLocaleConverter.java
new file mode 100755
index 0000000..ff78204
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/ShortLocaleConverter.java
@@ -0,0 +1,199 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import java.util.Locale;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.lang.Short</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class ShortLocaleConverter extends DecimalLocaleConverter {
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public ShortLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ShortLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public ShortLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ShortLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public ShortLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ShortLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public ShortLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ShortLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public ShortLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ShortLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public ShortLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public ShortLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern);
+ }
+}
+
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/SqlDateLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/SqlDateLocaleConverter.java
new file mode 100755
index 0000000..2f9fdee
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/SqlDateLocaleConverter.java
@@ -0,0 +1,218 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import java.sql.Date;
+import java.text.ParseException;
+import java.util.Locale;
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.sql.Date</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class SqlDateLocaleConverter extends DateLocaleConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public SqlDateLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlDateLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public SqlDateLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlDateLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public SqlDateLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlDateLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public SqlDateLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlDateLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public SqlDateLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlDateLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public SqlDateLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlDateLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern, locPattern);
+ }
+
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ protected Object parse(Object value, String pattern) throws ParseException {
+
+ return new Date(((java.util.Date) super.parse((String) value, pattern)).getTime());
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/SqlTimeLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/SqlTimeLocaleConverter.java
new file mode 100755
index 0000000..2cacd26
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/SqlTimeLocaleConverter.java
@@ -0,0 +1,217 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import java.sql.Time;
+import java.text.ParseException;
+import java.util.Locale;
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.sql.Time</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class SqlTimeLocaleConverter extends DateLocaleConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public SqlTimeLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimeLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public SqlTimeLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimeLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public SqlTimeLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimeLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public SqlTimeLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimeLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public SqlTimeLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimeLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public SqlTimeLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimeLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern, locPattern);
+ }
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ protected Object parse(Object value, String pattern) throws ParseException {
+
+ return new Time(((java.util.Date) super.parse((String) value, pattern)).getTime());
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/SqlTimestampLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/SqlTimestampLocaleConverter.java
new file mode 100755
index 0000000..2e0593a
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/SqlTimestampLocaleConverter.java
@@ -0,0 +1,216 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import java.sql.Timestamp;
+import java.text.ParseException;
+import java.util.Locale;
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive String into a <code>java.sql.Timestamp</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class SqlTimestampLocaleConverter extends DateLocaleConverter {
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public SqlTimestampLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimestampLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public SqlTimestampLocaleConverter(Locale locale) {
+
+ this(locale, (String) null);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimestampLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public SqlTimestampLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimestampLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public SqlTimestampLocaleConverter(Object defaultValue) {
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimestampLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public SqlTimestampLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimestampLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public SqlTimestampLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public SqlTimestampLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern, locPattern);
+ }
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ protected Object parse(Object value, String pattern) throws ParseException {
+
+ return new Timestamp(((java.util.Date) super.parse(value, pattern)).getTime());
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/StringLocaleConverter.java b/trunk/src/java/org/apache/commons/beanutils/locale/converters/StringLocaleConverter.java
new file mode 100755
index 0000000..a7e6870
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/StringLocaleConverter.java
@@ -0,0 +1,286 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import org.apache.commons.beanutils.locale.BaseLocaleConverter;
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import java.math.BigDecimal;
+import java.math.BigInteger;
+import java.text.DecimalFormat;
+import java.text.NumberFormat;
+import java.text.ParseException;
+import java.text.SimpleDateFormat;
+import java.util.Date;
+import java.util.Locale;
+
+
+/**
+ * <p>Standard {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * implementation that converts an incoming
+ * locale-sensitive object into a <code>java.lang.String</code> object,
+ * optionally using a default value or throwing a
+ * {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.</p>
+ *
+ * @author Yauheny Mikulski
+ */
+
+public class StringLocaleConverter extends BaseLocaleConverter {
+
+ // ----------------------------------------------------- Instance Variables
+
+ /** All logging goes through this logger */
+ private static Log log = LogFactory.getLog(StringLocaleConverter.class); //msz fix
+
+
+ // ----------------------------------------------------------- Constructors
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ */
+ public StringLocaleConverter() {
+
+ this(false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public StringLocaleConverter(boolean locPattern) {
+
+ this(Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ */
+ public StringLocaleConverter(Locale locale) {
+
+ this(locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public StringLocaleConverter(Locale locale, boolean locPattern) {
+
+ this(locale, (String) null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public StringLocaleConverter(Locale locale, String pattern) {
+
+ this(locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will throw a {@link org.apache.commons.beanutils.ConversionException}
+ * if a conversion error occurs.
+ *
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public StringLocaleConverter(Locale locale, String pattern, boolean locPattern) {
+
+ super(locale, pattern, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine and an unlocalized pattern is used
+ * for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ */
+ public StringLocaleConverter(Object defaultValue) {
+
+ this(defaultValue, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. The locale is the default locale for
+ * this instance of the Java Virtual Machine.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public StringLocaleConverter(Object defaultValue, boolean locPattern) {
+
+ this(defaultValue, Locale.getDefault(), locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ */
+ public StringLocaleConverter(Object defaultValue, Locale locale) {
+
+ this(defaultValue, locale, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public StringLocaleConverter(Object defaultValue, Locale locale, boolean locPattern) {
+
+ this(defaultValue, locale, null, locPattern);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs. An unlocalized pattern is used for the convertion.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ */
+ public StringLocaleConverter(Object defaultValue, Locale locale, String pattern) {
+
+ this(defaultValue, locale, pattern, false);
+ }
+
+ /**
+ * Create a {@link org.apache.commons.beanutils.locale.LocaleConverter}
+ * that will return the specified default value
+ * if a conversion error occurs.
+ *
+ * @param defaultValue The default value to be returned
+ * @param locale The locale
+ * @param pattern The convertion pattern
+ * @param locPattern Indicate whether the pattern is localized or not
+ */
+ public StringLocaleConverter(Object defaultValue, Locale locale, String pattern, boolean locPattern) {
+
+ super(defaultValue, locale, pattern, locPattern);
+ }
+
+ // --------------------------------------------------------- Methods
+
+ /**
+ * Convert the specified locale-sensitive input object into an output object of the
+ * specified type.
+ *
+ * @param value The input object to be converted
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ protected Object parse(Object value, String pattern) throws ParseException {
+
+ String result = null;
+
+ if ((value instanceof Integer) ||
+ (value instanceof Long) ||
+ (value instanceof BigInteger) ||
+ (value instanceof Byte) ||
+ (value instanceof Short)) {
+
+ result = getDecimalFormat(locale, pattern).format(((Number) value).longValue());
+ }
+ else if ((value instanceof Double) ||
+ (value instanceof BigDecimal) ||
+ (value instanceof Float)) {
+
+ result = getDecimalFormat(locale, pattern).format(((Number) value).doubleValue());
+ }
+ else if (value instanceof Date) { // java.util.Date, java.sql.Date, java.sql.Time, java.sql.Timestamp
+
+ SimpleDateFormat dateFormat =
+ new SimpleDateFormat(pattern, locale);
+
+ result = dateFormat.format(value);
+ }
+ else {
+ result = value.toString();
+ }
+
+ return result;
+ }
+
+ /**
+ * Make an instance of DecimalFormat.
+ *
+ * @param locale The locale
+ * @param pattern The pattern is used for the convertion
+ *
+ * @exception ConversionException if conversion cannot be performed
+ * successfully
+ */
+ private DecimalFormat getDecimalFormat(Locale locale, String pattern) {
+
+ DecimalFormat numberFormat = (DecimalFormat) NumberFormat.getInstance(locale);
+
+ // if some constructors default pattern to null, it makes only sense to handle null pattern gracefully
+ if (pattern != null) {
+ if (locPattern) {
+ numberFormat.applyLocalizedPattern(pattern);
+ } else {
+ numberFormat.applyPattern(pattern);
+ }
+ } else {
+ log.warn("No pattern provided, using default.");
+ }
+
+ return numberFormat;
+ }
+}
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/converters/package.html b/trunk/src/java/org/apache/commons/beanutils/locale/converters/package.html
new file mode 100644
index 0000000..4100820
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/converters/package.html
@@ -0,0 +1,12 @@
+<html>
+<head>
+<title>Package Documentation for org.apache.commons.beanutils.locale.converters</title>
+</head>
+<body bgcolor="white">
+<p>Standard implementations of the locale-aware
+<code>LocaleConverter</code>
+interface that are pre-registered with locale-aware
+<code>LocaleConvertUtils</code>
+at startup time.</p>
+</body>
+</html>
diff --git a/trunk/src/java/org/apache/commons/beanutils/locale/package.html b/trunk/src/java/org/apache/commons/beanutils/locale/package.html
new file mode 100644
index 0000000..ca32f73
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/locale/package.html
@@ -0,0 +1,10 @@
+<html>
+<head>
+<title>Package Documentation for org.apache.commons.beanutils.locale Package</title>
+</head>
+<body bgcolor="white">
+<p>Locale-aware extensions of the standard beanutils classes.
+This package allows locale-dependent population of JavaBeans.</p>
+
+</body>
+</html>
diff --git a/trunk/src/java/org/apache/commons/beanutils/package.html b/trunk/src/java/org/apache/commons/beanutils/package.html
new file mode 100644
index 0000000..d5e8205
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/beanutils/package.html
@@ -0,0 +1,990 @@
+<html>
+<head>
+<title>Package Documentation for org.apache.commons.beanutils Package</title>
+</head>
+<body bgcolor="white">
+<p>The <em>Bean Introspection Utilities</em> component of the Jakarta Commons
+subproject offers low-level utility classes that assist in getting and setting
+property values on Java classes that follow the naming design patterns outlined
+in the JavaBeans Specification, as well as mechanisms for dynamically defining
+and accessing bean properties.</p>
+
+<h1>Table of Contents</h1>
+
+<ol>
+<li><a href="#overview">Overview</a>
+ <ul>
+ <li><a href="#overview.background">Background</a></li>
+ <li><a href="#overview.dependencies">External Dependencies</a></li>
+ </ul></li>
+<li><a href="#standard">Standard JavaBeans</a>
+ <ul>
+ <li><a href="#standard.background">Background</a></li>
+ <li><a href="#standard.basic">Basic Property Access</a></li>
+ <li><a href="#standard.nested">Nested Property Access</a></li>
+ </ul></li>
+<li><a href="#dynamic">Dynamic Beans (DynaBeans)</a>
+ <ul>
+ <li><a href="#dynamic.background">Background</a></li>
+ <li><a href="#dynamic.basic">BasicDynaBean and BasicDynaClass</a></li>
+ <li><a href="#dynamic.resultSet">ResultSetDynaClass (Wraps ResultSet in DynaBeans)</a></li>
+ <li><a href="#dynamic.rowSet">RowSetDynaClass (Disconnected ResultSet as DynaBeans)</a></li>
+ <li><a href="#dynamic.wrap">WrapDynaBean and WrapDynaClass</a></li>
+ <li><a href="#dynamic.lazy"><i>Lazy</i> DynaBeans</a></li>
+ </ul></li>
+<li><a href="#conversion">Data Type Conversions</a>
+ <ul>
+ <li><a href="#conversion.background">Background</a></li>
+ <li><a href="#conversion.beanutils">BeanUtils and ConvertUtils
+ Conversions</a></li>
+ <li><a href="#conversion.defining">Defining Your Own Converters</a></li>
+ <li><a href="#conversion.i18n">Locale Aware Conversions</a></li>
+ </ul></li>
+<li><a href="#instances">Utility Objects And Static Utility Classes</a>
+ <ul>
+ <li><a href="#instances.background">Background</a></li>
+ </ul></li>
+<li><a href="#collections">Collections</a>
+ <ul>
+ <li><a href="#bean-comparator">Comparing Beans</a></li>
+ <li><a href="#bean-property-closure">Operating On Collections Of Beans</a></li>
+ <li><a href="#bean-property-predicate">Querying Or Filtering Collections Of Beans</a></li>
+ <li><a href="#bean-property-transformer">Transforming Collections Of Beans</a></li>
+ </ul></li>
+<li><a href="#FAQ">Frequently Asked Questions</a>
+ <ul>
+ <li><a href="#FAQ.property">Why Can't BeanUtils Find My Method?</a></li>
+ <li><a href="#FAQ.bc.order">How Do I Set The BeanComparator Order To Be Ascending/Descending?</a></li>
+ </ul></li>
+<li><a href='#Packaging'>Packaging</a></li>
+</ol>
+
+<a name="overview"></a>
+<h1>Overview</h1>
+
+<a name="overview.background"></a>
+<h3>Background</h3>
+
+<p>The <em>JavaBeans</em> name comes from a
+<a href="http://java.sun.com/products/javabeans/">Java API specification</a>
+for a component architecture for the Java language. Writing Java classes that
+conform to the JavaBeans design patterns makes it easier for Java developers
+to understand the functionality provided by your class, as well as allowing
+JavaBeans-aware tools to use Java's <em>introspection</em> capabilities to
+learn about the properties and operations provided by your class, and present
+them in a visually appealing manner in development tools.</p>
+
+<p>The <a href="http://java.sun.com/products/javabeans/docs/spec.html">JavaBeans
+Specification</a> describes the complete set of characteristics that makes
+an arbitrary Java class a JavaBean or not -- and you should consider reading
+this document to be an important part of developing your Java programming
+skills. However, the required characteristics of JavaBeans that are
+important for most development scenarios are listed here:</p>
+<ul>
+<li>The class must be <strong>public</strong>, and provide a
+ <strong>public</strong> constructor that accepts no arguments. This allows
+ tools and applications to dynamically create new instances of your bean,
+ without necessarily knowing what Java class name will be used ahead of
+ time, like this:
+<pre>
+ String className = ...;
+ Class beanClass = Class.forName(className);
+ Object beanInstance = beanClass.newInstance();
+</pre></li>
+<li>As a necessary consequence of having a no-arguments constructor,
+ configuration of your bean's behavior must be accomplished separately
+ from its instantiation. This is typically done by defining a set of
+ <em>properties</em> of your bean, which can be used to modify its behavior
+ or the data that the bean represents. The normal convention for
+ property names is that they start with a lower case letter, and be
+ comprised only of characters that are legal in a Java identifier.</li>
+<li>Typically, each bean property will have a public <em>getter</em> and
+ <em>setter</em> method that are used to retrieve or define the property's
+ value, respectively. The JavaBeans Specification defines a design
+ pattern for these names, using <code>get</code> or <code>set</code> as the
+ prefix for the property name with it's first character capitalized. Thus,
+ you a JavaBean representing an employee might have
+ (among others) properties named <code>firstName</code>,
+ <code>lastName</code>, and <code>hireDate</code>, with method signatures
+ like this:
+<pre>
+ public class Employee {
+ public Employee(); // Zero-arguments constructor
+ public String getFirstName();
+ public void setFirstName(String firstName);
+ public String getLastName();
+ public void setLastName(String lastName);
+ public Date getHireDate();
+ public void setHireDate(Date hireDate);
+ public boolean isManager();
+ public void setManager(boolean manager);
+ public String getFullName();
+ }
+</pre></li>
+<li>As you can see from the above example, there is a special variant allowed
+ for boolean properties -- you can name the <em>getter</em> method with a
+ <code>is</code> prefix instead of a <code>get</code> prefix if that makes
+ for a more understandable method name.</li>
+<li>If you have both a <em>getter</em> and a <em>setter</em> method for a
+ property, the data type returned by the <em>getter</em> must match the
+ data type accepted by the <em>setter</em>. In addition, it is contrary
+ to the JavaBeans specification to have more than one <em>setter</em>
+ with the same name, but different property types.</li>
+<li>It is not required that you provide a <em>getter</em> and a
+ <em>setter</em> for every property. In the example above, the
+ <code>fullName</code> property is read-only, because there is no
+ <em>setter</em> method. It is also possible, but less common, to provide
+ write-only properties.</li>
+<li>It is also possible to create a JavaBean where the <em>getter</em> and
+ <em>setter</em> methods do not match the naming pattern described above.
+ The standard JavaBeans support classes in the Java language, as well as
+ all classes in the BeanUtils package, allow you to describe the actual
+ property method names in a <code>BeanInfo</code> class associated with
+ your bean class. See the JavaBeans Specification for full details.</li>
+<li>The JavaBeans Specification also describes many additional design patterns
+ for event listeners, wiring JavaBeans together into component hierarchies,
+ and other useful features that are beyond the scope of the BeanUtils
+ package.</li>
+</ul>
+
+<p>Using standard Java coding techniques, it is very easy to deal with
+JavaBeans if you know ahead of time which bean classes you will be using, and
+which properties you are interested in:</p>
+<pre>
+ Employee employee = ...;
+ System.out.println("Hello " + employee.getFirstName() + "!");
+</pre>
+
+<a name="overview.dependencies"></a>
+<h3>External Dependencies</h3>
+
+<p>The <em>commons-beanutils</em> package requires that the following
+additional packages be available in the application's class path at runtime:
+</p>
+<ul>
+<li><a href="http://jakarta.apache.org/builds/jakarta-commons/release/commons-collections">
+Collections Package (Jakarta Commons)</a>, version 1.0 or later</li>
+<li><a href="http://jakarta.apache.org/builds/jakarta-commons/release/commons-logging">
+Logging Package (Jakarta Commons)</a>, version 1.0 or later</li>
+</ul>
+
+
+<a name="standard"></a>
+<h1>Standard JavaBeans</h1>
+
+<a name="standard.background"></a>
+<h3>Background</h3>
+
+<p>As described above, the standard facilities of the Java programming language
+make it easy and natural to access the property values of your beans using
+calls to the appropriate getter methods.
+But what happens in more sophisticated environments where you do not
+necessarily know ahead of time which bean class you are going to be using,
+or which property you want to retrieve or modify? The Java language provides
+classes like <code>java.beans.Introspector</code>, which can examine a Java
+class at runtime and identify for you the names of the property getter and
+setter methods, plus the <em>Reflection</em> capabilities to dynamically call
+such a method. However, these APIs can be difficult to use, and expose the
+application developer to many unnecessary details of the underlying structure
+of Java classes. The APIs in the BeanUtils package are intended to simplify
+getting and setting bean properties dynamically, where the objects you are
+accessing -- and the names of the properties you care about -- are determined
+at runtime in your application, rather than as you are writing and compiling
+your application's classes.</p>
+
+<p>This is the set of needs that are satisfied by the static methods of the
+<a href="PropertyUtils.html">PropertyUtils</a>
+class, which are described further in this section. First, however, some
+further definitions will prove to be useful:</p>
+
+<p>The general set of possible property types supported by a JavaBean can be
+broken into three categories -- some of which are supported by the standard
+JavaBeans specification, and some of which are uniquely supported by the
+<em>BeanUtils</em> package:</p>
+<ul>
+<li><strong>Simple</strong> - Simple, or scalar, properties have a single
+ value that may be retrieved or modified. The underlying property type
+ might be a Java language primitive (such as <code>int</code>, a simple
+ object (such as a <code>java.lang.String</code>), or a more complex
+ object whose class is defined either by the Java language, by the
+ application, or by a class library included with the application.</li>
+<li><strong>Indexed</strong> - An indexed property stores an ordered collection
+ of objects (all of the same type) that can be individually accessed by an
+ integer-valued, non-negative index (or subscript). Alternatively, the
+ entire set of values may be set or retrieved using an array.
+ As an extension to the JavaBeans specification, the
+ <em>BeanUtils</em> package considers any property whose underlying data
+ type is <code>java.util.List</code> (or an implementation of List) to be
+ indexed as well.</li>
+<li><strong>Mapped</strong> - As an extension to standard JavaBeans APIs,
+ the <em>BeanUtils</em> package considers any property whose underlying
+ value is a <code>java.util.Map</code> to be "mapped". You can set and
+ retrieve individual values via a String-valued key.</li>
+</ul>
+
+<p>A variety of API methods are provided in the <a href="PropertyUtils.html">
+PropertyUtils</a> class to get and set property values of all of these types.
+In the code fragments below, assume that there are two bean classes defined
+with the following method signatures:</p>
+<pre>
+ public class Employee {
+ public Address getAddress(String type);
+ public void setAddress(String type, Address address);
+ public Employee getSubordinate(int index);
+ public void setSubordinate(int index, Employee subordinate);
+ public String getFirstName();
+ public void setFirstName(String firstName);
+ public String getLastName();
+ public void setLastName(String lastName);
+ }
+</pre>
+
+<a name="standard.basic"></a>
+<h3>Basic Property Access</h3>
+
+<p>Getting and setting <strong>simple</strong> property values is, well,
+simple :-). Check out the following API signatures in the Javadocs:</p>
+
+<ul>
+<li><a href="PropertyUtils.html#getSimpleProperty(java.lang.Object,java.lang.String)">
+ PropertyUtils.getSimpleProperty(Object bean, String name)</a></li>
+<li><a href="PropertyUtils.html#setSimpleProperty(java.lang.Object,java.lang.String,java.lang.Object)">
+ PropertyUtils.setSimpleProperty(Object bean, String name, Object value)</a></li>
+</ul>
+
+<p>Using these methods, you might dynamically manipulate the employee's name
+in an application:</p>
+<pre>
+ Employee employee = ...;
+ String firstName = (String)
+ PropertyUtils.getSimpleProperty(employee, "firstName");
+ String lastName = (String)
+ PropertyUtils.getSimpleProperty(employee, "lastName");
+ ... manipulate the values ...
+ PropertyUtils.setSimpleProperty(employee, "firstName", firstName);
+ PropertyUtils.setSimpleProperty(employee, "lastName", lastName);
+</pre>
+
+<p>For <strong>indexed</strong> properties, you have two choices - you can
+either build a subscript into the "property name" string, using square
+brackets, or you can specify the subscript in a separate argument to the
+method call:</p>
+
+<ul>
+<li><a href="PropertyUtils.html#getIndexedProperty(java.lang.Object,java.lang.String)">
+ PropertyUtils.getIndexedProperty(Object bean, String name)</a></li>
+<li><a href="PropertyUtils.html#getIndexedProperty(java.lang.Object,java.lang.String,int)">
+ PropertyUtils.getIndexedProperty(Object bean, String name, int index)</a></li>
+<li><a href="PropertyUtils.html#setIndexedProperty(java.lang.Object,java.lang.String,java.lang.Object)">
+ PropertyUtils.setIndexedProperty(Object bean, String name, Object value)</a></li>
+<li><a href="PropertyUtils.html#setIndexedProperty(java.lang.Object,java.lang.String,int,java.lang.Object)">
+ PropertyUtils.setIndexedProperty(Object bean, String name, int index, Object value)</a></li>
+</ul>
+
+<p>Only integer constants are allowed when you add a subscript to the property
+name. If you need to calculate the index of the entry you wish to retrieve,
+you can use String concatenation to assemble the property name expression.
+For example, you might do either of the following:</p>
+<pre>
+ Employee employee = ...;
+ int index = ...;
+ String name = "subordinate[" + index + "]";
+ Employee subordinate = (Employee)
+ PropertyUtils.getIndexedProperty(employee, name);
+
+ Employee employee = ...;
+ int index = ...;
+ Employee subordinate = (Employee)
+ PropertyUtils.getIndexedProperty(employee, "subordinate", index);
+</pre>
+
+<p>In a similar manner, there are two possible method signatures for getting
+and setting <strong>mapped</strong> properties. The difference is that the
+extra argument is surrounded by parentheses ("(" and ")") instead of square
+brackets, and it is considered to be a String-value key used to get or set
+the appropriate value from an underlying map.</p>
+
+<ul>
+<li><a href="PropertyUtils.html#getMappedProperty(java.lang.Object,java.lang.String)">
+ PropertyUtils.getMappedProperty(Object bean, String name)</a></li>
+<li><a href="PropertyUtils.html#getMappedProperty(java.lang.Object,java.lang.String,java.lang.String)">
+ PropertyUtils.getMappedProperty(Object bean, String name, String key)</a></li>
+<li><a href="PropertyUtils.html#setMappedProperty(java.lang.Object,java.lang.String,java.lang.Object)">
+ PropertyUtils.setMappedProperty(Object bean, String name, Object value)</a></li>
+<li><a href="PropertyUtils.html#setMappedProperty(java.lang.Object,java.lang.String,java.lang.String,java.lang.Object)">
+ PropertyUtils.setMappedProperty(Object bean, String name, String key, Object value)</a></li>
+</ul>
+
+<p>You can, for example, set the employee's home address in either of these
+two manners:</p>
+
+<pre>
+ Employee employee = ...;
+ Address address = ...;
+ PropertyUtils.setMappedProperty(employee, "address(home)", address);
+
+ Employee employee = ...;
+ Address address = ...;
+ PropertyUtils.setMappedProperty(employee, "address", "home", address);
+</pre>
+
+<a name="standard.nested"></a>
+<h3>Nested Property Access</h3>
+
+<p>In all of the examples above, we have assumed that you wished to retrieve
+the value of a property of the bean being passed as the first argument to a
+PropertyUtils method. However, what if the property value you retrieve is
+really a Java object, and you wish to retrieve a property of <em>that</em>
+object instead?</p>
+
+<p>For example, assume we really wanted the <code>city</code> property of the
+employee's home address. Using standard Java programming techniques for direct
+access to the bean properties, we might write:</p>
+
+<pre>
+ String city = employee.getAddress("home").getCity();
+</pre>
+
+<p>The equivalent mechanism using the PropertyUtils class is called
+<strong>nested</strong> property access. To use this approach, you concatenate
+together the property names of the access path, using "." separators -- very
+similar to the way you can perform nested property access in JavaScript.</p>
+
+<ul>
+<li><a href="PropertyUtils.html#getNestedProperty(java.lang.Object,java.lang.String)">
+ PropertyUtils.getNestedProperty(Object bean, String name)</a></li>
+<li><a href="PropertyUtils.html#setNestedProperty(java.lang.Object,java.lang.String,java.lang.Object)">
+ PropertyUtils.setNestedProperty(Object bean, String name, Object value)</a></li>
+</ul>
+
+<p>The PropertyUtils equivalent to the above Java expression would be:</p>
+
+<pre>
+ String city = (String)
+ PropertyUtils.getNestedProperty(employee, "address(home).city");
+</pre>
+
+<p>Finally, for convenience, PropertyUtils provides method signatures that
+accept any arbitrary combination of simple, indexed, and mapped property
+access, using any arbitrary level of nesting:</p>
+
+<ul>
+<li><a href="PropertyUtils.html#getProperty(java.lang.Object,java.lang.String)">
+ PropertyUtils.getProperty(Object bean, String name)</a></li>
+<li><a href="PropertyUtils.html#setProperty(java.lang.Object,java.lang.String,java.lang.Object)">
+ PropertyUtils.setProperty(Object bean, String name, Object value)</a></li>
+</ul>
+
+<p>which you might use like this:</p>
+
+<pre>
+ Employee employee = ...;
+ String city = (String) PropertyUtils.getProperty(employee,
+ "subordinate[3].address(home).city");
+</pre>
+
+<a name="dynamic"></a>
+<h1>Dynamic Beans (DynaBeans)</h1>
+
+<a name="dynamic.background"></a>
+<h3>Background</h3>
+
+<p>The <a href="PropertyUtils.html">PropertyUtils</a> class described in the
+preceding section is designed to provide dynamic property access on existing
+JavaBean classes, without modifying them in any way. A different use case for
+dynamic property access is when you wish to represent a dynamically calculated
+set of property values as a JavaBean, but <em>without</em> having to actually
+write a Java class to represent these properties. Besides the effort savings
+in not having to create and maintain a separate Java class, this ability also
+means you can deal with situations where the set of properties you care about
+is determined dynamically (think of representing the result set of an SQL
+select as a set of JavaBeans ...).</p>
+
+<p>To support this use case, the <em>BeanUtils</em> package provides the
+<a href="DynaBean.html">DynaBean</a> interface, which must be implemented by a
+bean class actually implementing the interface's methods, and the associated
+<a href="DynaClass.html">DynaClass</a> interface that defines the set of
+properties supported by a particular group of DynaBeans, in much the same way
+that <code>java.lang.Class</code> defines the set of properties supported by
+all instances of a particular JavaBean class.</p>
+
+<p>For example, the <code>Employee</code> class used in the examples above
+might be implemented as a DynaBean, rather than as a standard JavaBean. You
+can access its properties like this:</p>
+
+<pre>
+ DynaBean employee = ...; // Details depend on which
+ // DynaBean implementation you use
+ String firstName = (String) employee.get("firstName");
+ Address homeAddress = (Address) employee.get("address", "home");
+ Object subordinate = employee.get("subordinate", 2);
+</pre>
+
+<p>One very important convenience feature should be noted: <em>the
+PropertyUtils property getter and setter methods understand how to access
+properties in DynaBeans</em>. Therefore, if the bean you pass as the first
+argument to, say, <code>PropertyUtils.getSimpleProperty()</code> is really a
+DynaBean implementation, the call will get converted to the appropriate
+DynaBean getter method transparently. Thus, you can base your application's
+dynamic property access totally on the PropertyUtils APIs, if you wish, and
+use them to access either standard JavaBeans or DynaBeans without having to
+care ahead of time how a particular bean is implemented.</p>
+
+<p>Because DynaBean and DynaClass are interfaces, they may be implemented
+multiple times, in different ways, to address different usage scenarios. The
+following subsections describe the implementations that are provided as a part
+of the standard <em>BeanUtils</em> package, although you are encouraged to
+provide your own custom implementations for cases where the standard
+implementations are not sufficient.</p>
+
+<a name="dynamic.basic"></a>
+<h3><code>BasicDynaBean</code> and <code>BasicDynaClass</code></h3>
+
+<p>The <a href="BasicDynaBean.html">BasicDynaBean</a> and
+<a href="BasicDynaClass.html">BasicDynaClass</a> implementation provides a
+basic set of
+dynamic property capabilities where you want to dynamically define the
+set of properties (described by instances of <a href="DynaProperty.html">
+DynaProperty</a>). You start by defining the DynaClass that establishes
+the set of properties you care about:</p>
+
+<pre>
+ DynaProperty[] props = new DynaProperty[]{
+ new DynaProperty("address", java.util.Map.class),
+ new DynaProperty("subordinate", mypackage.Employee[].class),
+ new DynaProperty("firstName", String.class),
+ new DynaProperty("lastName", String.class)
+ };
+ BasicDynaClass dynaClass = new BasicDynaClass("employee", null, props);
+</pre>
+
+<p>Note that the 'dynaBeanClass' argument (in the constructor of
+<code>BasicDynaClass</code>) can have the value of <code>null</code>. In this
+case, the value of <code>dynaClass.getDynaBeanClass</code> will just be the
+<code>Class</code> for BasicDynaBean.</p>
+
+<p>Next, you use the <code>newInstance()</code> method of this DynaClass to
+create new DynaBean instances that conform to this DynaClass, and populate
+its initial property values (much as you would instantiate a new standard
+JavaBean and then call its property setters):</p>
+
+<pre>
+ DynaBean employee = dynaClass.newInstance();
+ employee.set("address", new HashMap());
+ employee.set("subordinate", new mypackage.Employee[0]);
+ employee.set("firstName", "Fred");
+ employee.set("lastName", "Flintstone");
+</pre>
+
+<p>Note that the DynaBean class was declared to be
+<code>DynaBean</code> instead of <code>BasicDynaBean</code>. In
+general, if you are using DynaBeans, you will not want to care about the
+actual implementation class that is being used -- you only care about
+declaring that it is a <code>DynaBean</code> so that you can use the
+DynaBean APIs.</p>
+
+<p>As stated above, you can pass a DynaBean instance as the first argument
+to a <code>PropertyUtils</code> method that gets and sets properties, and it
+will be interpreted as you expect -- the dynamic properties of the DynaBean
+will be retrieved or modified, instead of underlying properties on the
+actual BasicDynaBean implementation class.</p>
+
+<a name="dynamic.resultSet"></a>
+<h3><code>ResultSetDynaClass</code> (Wraps ResultSet in DynaBeans)</h3>
+
+<p>A very common use case for DynaBean APIs is to wrap other collections of
+"stuff" that do not normally present themselves as JavaBeans. One of the most
+common collections that would be nice to wrap is the
+<code>java.sql.ResultSet</code> that is returned when you ask a JDBC driver
+to perform a SQL SELECT statement. Commons BeanUtils offers a standard
+mechanism for making each row of the result set visible as a DynaBean,
+which you can utilize as shown in this example:</p>
+<pre>
+ Connection conn = ...;
+ Statement stmt = conn.createStatement();
+ ResultSet rs = stmt.executeQuery
+ ("select account_id, name from customers");
+ Iterator rows = (new ResultSetDynaClass(rs)).iterator();
+ while (rows.hasNext()) {
+ DynaBean row = (DynaBean) rows.next();
+ System.out.println("Account number is " +
+ row.get("account_id") +
+ " and name is " + row.get("name"));
+ }
+ rs.close();
+ stmt.close();
+</pre>
+
+
+<a name="dynamic.rowSet"></a>
+<h3><code>RowSetDynaClass</code> (Disconnected ResultSet as DynaBeans)</h3>
+<p>Although <a href="#dynamic.resultSet"><code>ResultSetDynaClass</code></a> is
+a very useful technique for representing the results of an SQL query as a
+series of DynaBeans, an important problem is that the underlying
+<code>ResultSet</code> must remain open throughout the period of time that the
+rows are being processed by your application. This hinders the ability to use
+<code>ResultSetDynaClass</code> as a means of communicating information from
+the model layer to the view layer in a model-view-controller architecture
+such as that provided by the <a href="http://jakarta.apache.org/struts/">Struts
+Framework</a>, because there is no easy mechanism to assure that the result set
+is finally closed (and the underlying <code>Connection</code> returned to its
+connection pool, if you are using one).</p>
+
+<p>The <code>RowSetDynaClass</code> class represents a different approach to
+this problem. When you construct such an instance, the underlying data is
+<em>copied</em> into a set of in-memory DynaBeans that represent the result.
+The advantage of this technique, of course, is that you can immediately close
+the ResultSet (and the corresponding Statement), normally before you even
+process the actual data that was returned. The disadvantage, of course, is
+that you must pay the performance and memory costs of copying the result data,
+and the result data must fit entirely into available heap memory. For many
+environments (particularly in web applications), this tradeoff is usually
+quite beneficial.</p>
+
+<p>As an additional benefit, the <code>RowSetDynaClass</code> class is defined
+to implement <code>java.io.Serializable</code>, so that it (and the
+DynaBeans that correspond to each row of the result) can be conveniently
+serialized and deserialized (as long as the underlying column values are
+also Serializable). Thus, <code>RowSetDynaClass</code> represents a very
+convenient way to transmit the results of an SQL query to a remote Java-based
+client application (such as an applet).</p>
+
+<p>The normal usage pattern for a <code>RowSetDynaClass</code> will look
+something like this:</p>
+<pre>
+ Connection conn = ...; // Acquire connection from pool
+ Statement stmt = conn.createStatement();
+ ResultSet rs = stmt.executeQuery("SELECT ...");
+ RowSetDynaClass rsdc = new RowSetDynaClass(rs);
+ rs.close();
+ stmt.close();
+ ...; // Return connection to pool
+ List rows = rsdc.getRows();
+ ...; // Process the rows as desired
+</pre>
+
+
+<a name="dynamic.wrap"></a>
+<h3><code>WrapDynaBean</code> and <code>WrapDynaClass</code></h3>
+
+<p>OK, you've tried the DynaBeans APIs and they are cool -- very simple
+<code>get()</code> and <code>set()</code> methods provide easy access to all
+of the dynamically defined simple, indexed, and mapped properties of your
+DynaBeans. You'd like to use the DynaBean APIs to access <strong>all</strong>
+of your beans, but you've got a bunch of existing standard JavaBeans classes
+to deal with as well. This is where the
+<a href="WrapDynaBean.html">WrapDynaBean</a> (and its associated
+<a href="WrapDynaClass.html">WrapDynaClass</a>) come into play. As the name
+implies, a WrapDynaBean is used to "wrap" the DynaBean APIs around an
+existing standard JavaBean class. To use it, simply create the wrapper
+like this:</p>
+
+<pre>
+ MyBean bean = ...;
+ DynaBean wrapper = new WrapDynaBean(bean);
+ String firstName = wrapper.get("firstName");
+</pre>
+
+<p>Note that, although appropriate <code>WrapDynaClass</code> instances are
+created internally, you never need to deal with them.</p>
+
+<a name="dynamic.lazy"></a>
+<h3><i>Lazy</i> DynaBeans (<a href="LazyDynaBean.html">LazyDynaBean</a>,
+<a href="LazyDynaMap.html">LazyDynaMap</a> and <a href="LazyDynaClass.html">LazyDynaClass</a>)</h3>
+
+<p>You bought into the DynaBeans because it saves coding all those POJO JavaBeans but
+ you're here because <i>lazy</i> caught your eye and wondered whats that about?
+ What makes these flavors of DynaBean <i>lazy</i> are the following features:</p>
+ <ul>
+ <li><strong><i>Lazy</i> property addition</strong> - lazy beans use a
+ <code><a href="DynaClass.html">DynaClass</a></code> which implements
+ the <code><a href="MutableDynaClass.html">MutableDynaClass</a></code>
+ interface. This provides the ability to add and remove a DynaClass's
+ properties. <i>Lazy</i> beans use this feature to automatically add
+ a property which doesn't exist to the DynaClass when
+ the <code>set(name, value)</code> method is called.</li>
+ <li><strong><i>Lazy</i> List/Array growth</strong> - If an <i>indexed</i> property is not large
+ enough to accomodate the <code>index</code> being set then the <code>List</code> or
+ <code>Array</code> is automatically <i>grown</i> so that it is.</li>
+ <li><strong><i>Lazy</i> List/Array instantiation</strong> - if an <i>indexed</i>
+ property doesn't exist then calling the <a href="DynaBean.html">DynaBean</a>'s
+ <i>indexed</i> property getter/setter methods (i.e. <code>get(name, index)</code> or
+ <code>set(name, index, value)</code>) results in either a new <code>List</code>
+ or <code>Array</code> being instantiated. If the indexed property has not been
+ defined in the DynaClass then it is automatically added and a default <code>List</code>
+ implementation instantiated.</li>
+ <li><strong><i>Lazy</i> Map instantiation</strong> - if a <i>mapped</i>
+ property doesn't exist then calling the <a href="DynaBean.html">DynaBean</a>'s
+ <i>mapped</i> property getter/setter methods (i.e. <code>get(name, key)</code> or
+ <code>set(name, key, value)</code>) results in a new <code>Map</code>
+ being instantiated. If the mapped property has not been defined in the DynaClass
+ then it is automatically added and a default <code>Map</code> implementation
+ instantiated.</li>
+ <li><strong><i>Lazy</i> Bean instantiation</strong> - if a property is defined in
+ the <code>DynaClass</code> as a <code>DynaBean</code> or regular bean and
+ doesn't exist in the <code>DynaBean</code> then <code>LazyDynaBean</code> wiill
+ try to instantiate the bean using a default empty constructor.</li>
+ </ul>
+
+<p><strong><a href="LazyDynaBean.html">LazyDynaBean</a></strong> is the standard <i>lazy</i> bean
+ implementation. By default it is associated with a <a href="LazyDynaClass.html">LazyDynaClass</a>
+ which implements the <a href="MutableDynaClass.html">MutableDynaClass</a> interface - however
+ it can be used with any <code>MutableDynaClass</code> implementation. The question is <i>how do
+ I use it?</i> - well it can be as simple as creating a new bean and then calling the getters/setters...</p>
+
+<pre>
+ DynaBean dynaBean = new LazyDynaBean();
+
+ dynaBean.set("foo", "bar"); // simple
+
+ dynaBean.set("customer", "title", "Mr"); // mapped
+ dynaBean.set("customer", "surname", "Smith"); // mapped
+
+ dynaBean.set("address", 0, addressLine1); // indexed
+ dynaBean.set("address", 1, addressLine2); // indexed
+ dynaBean.set("address", 2, addressLine3); // indexed
+</pre>
+
+<p><strong><a href="LazyDynaMap.html">LazyDynaMap</a></strong> is a <i>light wieght</i> <code>DynaBean</code>
+ facade to a <code>Map</code> with all the usual <i>lazy</i> features. Its <i>light weight</i> because it doesn't
+ have an associated <code>DynaClass</code> containing all the properties. In fact it actually implements
+ the <code>DynaClass</code> interface itself (and <code>MutableDynaClass</code>) and derives all the <i>DynaClass</i>
+ information from the actual contents of the <code>Map</code>. A <code>LazyDynaMap</code> can be created around an
+ existing <code>Map</code> or can instantiate its own <code>Map</code>. After any <code>DynaBean</code>
+ processing has finished the <code>Map</code> can be retrieved and the DynaBean <i>facade</i> discarded.</p>
+
+<p>If you need a new <code>Map</code> then to use....</p>
+
+<pre>
+ DynaBean dynaBean = new LazyDynaMap(); // create DynaBean
+
+ dynaBean.set("foo", "bar"); // simple
+ dynaBean.set("customer", "title", "Mr"); // mapped
+ dynaBean.set("address", 0, addressLine1); // indexed
+
+ Map myMap = dynaBean.getMap() // retrieve the Map
+</pre>
+<p><i>or</i> to use with an existing <code>Map</code> ....</p>
+
+<pre>
+ Map myMap = .... // exisitng Map
+ DynaBean dynaBean = new LazyDynaMap(myMap); // wrap Map in DynaBean
+ dynaBean.set("foo", "bar"); // set properties
+</pre>
+
+<p><strong><a href="LazyDynaClass.html">LazyDynaClass</a></strong> extends <a href="BasicDynaClass.html">BasicDynaClass</a>
+ and implements the <a href="MutableDynaClass.html">MutableDynaClass</a> interface. It can be used with other
+ <code>DynaBean</code> implementations, but it is the default <code>DynaClass</code> used by <code>LazyDynaBean</code>.
+ When using the <code>LazyDynaBean</code> there may be no need to have anything to do with the <code>DynaClass</code>
+ However sometimes there is a requirement to set up the <code>DynaClass</code> first - perhaps to
+ define the type of array for an indexed property, or if using the DynaBean in <i>restricted</i> mode (see note below)
+ is required. Doing so is straight forward...</p>
+
+<p><i>Either</i> create a <code>LazyDynaClass</code> first...
+
+<pre>
+ MutableDynaClass dynaClass = new LazyDynaClass(); // create DynaClass
+
+ dynaClass.add("amount", java.lang.Integer.class); // add property
+ dynaClass.add("orders", OrderBean[].class); // add indexed property
+ dynaClass.add("orders", java.util.TreeMapp.class); // add mapped property
+
+ DynaBean dynaBean = new LazyDynaBean(dynaClass); // Create DynaBean with associated DynaClass
+</pre>
+
+<p><i>or</i> create a <code>LazyDynaBean</code> and get the <code>DynaClass</code>...
+
+<pre>
+ DynaBean dynaBean = new LazyDynaBean(); // Create LazyDynaBean
+ MutableDynaClass dynaClass =
+ (MutableDynaClass)dynaBean.getDynaClass(); // get DynaClass
+
+ dynaClass.add("amount", java.lang.Integer.class); // add property
+ dynaClass.add("myBeans", myPackage.MyBean[].class); // add 'array' indexed property
+ dynaClass.add("myMap", java.util.TreeMapp.class); // add mapped property
+</pre>
+
+<p><strong>NOTE:</strong> One feature of <a href="MutableDynaClass.html">MutableDynaClass</a> is that it
+ has a <i>Restricted</i> property. When the DynaClass is <i>restricted</i> no properties can be added
+ or removed from the <code>DynaClass</code>. Neither the <code>LazyDynaBean</code> or <code>LazyDynaMap</code>
+ will add properties automatically if the <code>DynaClass</code> is <i>restricted</i>.</p>
+
+
+
+<a name="conversion"></a>
+<h1>Data Type Conversions</h1>
+
+<a name="conversion.background"></a>
+<h3>Background</h3>
+
+<p>So far, we've only considered the cases where the data types of the
+dynamically accessed properties are known, and where we can use Java casts
+to perform type conversions. What happens if you want to automatically
+perform type conversions when casting is not possible? The
+<em>BeanUtils</em> package provides a variety of APIs and design patterns
+for performing this task as well.</p>
+
+<a name="conversion.beanutils"></a>
+<h3><code>BeanUtils</code> and <code>ConvertUtils</code> Conversions</h3>
+
+<p>A very common use case (and the situation that caused the initial creation
+of the <em>BeanUtils</em> package) was the desire to convert the set of request
+parameters that were included in a
+<code>javax.servlet.HttpServletRequest</code> received by a web application
+into a set of corresponding property setter calls on an arbitrary JavaBean.
+(This is one of the fundamental services provided by the
+<a href="http://jakarta.apache.org/struts">Struts Framework</a>, which uses
+<em>BeanUtils</em> internally to implement this functionality.)</p>
+
+<p>In an HTTP request, the set of included parameters is made available as a
+series of String (or String array, if there is more than one value for the
+same parameter name) instances, which need to be converted to the underlying
+data type. The <a href="BeanUtils.html">BeanUtils</a> class provides
+property setter methods that accept String values, and automatically convert
+them to appropriate property types for Java primitives (such as
+<code>int</code> or <code>boolean</code>), and property getter methods that
+perform the reverse conversion. Finally, a <code>populate()</code> method
+is provided that accepts a <code>java.util.Map</code> containing a set of
+property values (keyed by property name), and calls all of the appropriate
+setters whenever the underlying bean has a property with the same name as
+one of the request parameters. So, you can perform the all-in-one property
+setting operation like this:</p>
+
+<pre>
+ HttpServletRequest request = ...;
+ MyBean bean = ...;
+ HashMap map = new HashMap();
+ Enumeration names = request.getParameterNames();
+ while (names.hasMoreElements()) {
+ String name = (String) names.nextElement();
+ map.put(name, request.getParameterValues(name));
+ }
+ BeanUtils.populate(bean, map);
+</pre>
+
+<p>The <code>BeanUtils</code> class relies on conversion methods defined in
+the <a href="ConvertUtils.html">ConvertUtils</a> class to perform the actual
+conversions, and these methods are availablve for direct use as well.
+<strong>WARNING</strong> - It is likely that the hard coded use of
+<code>ConvertUtils</code> methods will be deprecated in the future, and
+replaced with a mechanism that allows you to plug in your own implementations
+of the <a href="Converter.html">Converter</a> interface instead. Therefore,
+new code should not be written with reliance on ConvertUtils.</p>
+
+<a name="conversion.defining"></a>
+<h3>Defining Your Own Converters</h3>
+
+<p>The <code>ConvertUtils</code> class supports the ability to define and
+register your own String --> Object conversions for any given Java class.
+Once registered, such converters will be used transparently by all of the
+<code>BeanUtils</code> methods (including <code>populate()</code>). To
+create and register your own converter, follow these steps:</p>
+<ul>
+<li>Write a class that implements the <a href="Converter.html">Converter</a>
+ interface. The <code>convert()</code> method should accept the
+ <code>java.lang.Class</code> object of your application class (i.e.
+ the class that you want to convert to, and a String representing the
+ incoming value to be converted.</li>
+<li>At application startup time, register an instance of your converter class
+ by calling the <code>ConvertUtils.register()</code> method.</li>
+</ul>
+
+<a name="conversion.i18n"></a>
+<h3>Locale Aware Conversions</h3>
+<p>The standard classes in <code>org.apache.commons.beanutils</code> are not
+locale aware. This gives them a cleaner interface and makes then easier to use
+in situations where the locale is not important.</p>
+<p>Extended, locale-aware analogues can be found in
+<code><a href='locale/package-summary.html'>org.apache.commons.beanutils.locale
+</a></code>. These are built along the same
+lines as the basic classes but support localization.</p>
+
+
+<a name="instances"></a>
+<h1>Utility Objects And Static Utility Classes</h1>
+<a name="instances.background"></a>
+<h3>Background</h3>
+<p>
+So far, the examples have covered the static utility classes (<code>BeanUtils</code>,
+<code>ConvertUtils</code> and <code>PropertyUtils</code>). These are easy to use but are
+somewhat inflexible. These all share the same registered converters and the same caches.
+</p>
+<p>
+This functionality can also be accessed through utility objects (in fact, the static utility
+class use worker instances of these classes). For each static utility class, there is a corresponding
+class with the same functionality that can be instantiated:
+</p>
+<p>
+<table cols='2' width='60%'>
+<tr><th>Static Utility Class</th><th>Utility Object</th></tr>
+<tr><td>BeanUtils</td><td>BeanUtilsBean</td></tr>
+<tr><td>ConvertUtils</td><td>ConvertUtilsBean</td></tr>
+<tr><td>PropertyUtils</td><td>PropertyUtilsBean</td></tr>
+</table>
+</p>
+<p>
+Creating an instances allow gives guarenteed control of the caching and registration
+to the code that creates it.
+</p>
+
+<a name="collections"></a>
+<h1>Collections</h1>
+<strong>Note:</strong> these classes depend on
+<a href='http://jakarta.apache.org/commons/collections'>Commons Collection</a> 3.0 or higher
+and are optional. To use them, please ensure that this library is available
+and that either the all-in-one jar is used or the modular
+<code>commons-beanutils-bean-collections.jar</code> is available
+(in addition to <code>commons-beanutils-core.jar</code>, of course :-).
+<a name="bean-comparator"></a>
+<h3>Comparing Beans</h3>
+<p>
+<code>org.apache.commons.beanutils.BeanComparator</code> is a <code>Comparator</code> implementation
+that compares beans based on a shared property value.
+</p>
+<a name="bean-property-closure"></a>
+<h3>Operating On Collections Of Beans</h3>
+<p>
+The <code>Closure</code> interface in <code>commons-collections</code> encapsulates a block of code that
+executes on an arbitrary input Object. <code>Commons-collections</code> contains code that allows
+<code>Closures</code> to be applied to the contents of a Collection. For more details, see the
+<a href='http://jakarta.apache.org/commons/collections.html'>commons-collections</a>
+documentation.
+</p>
+<p>
+<code>BeanPropertyValueChangeClosure</code> is a <code>Closure</code> that sets a specified property
+to a particular value. A typical usage is to combine this with <code>commons-collections</code>
+so that all the beans in a collection can have a particular property set to a particular value.
+</p>
+<p>
+For example, set the activeEmployee property to TRUE for an entire collection:
+ <code><pre>
+ // create the closure
+ BeanPropertyValueChangeClosure closure =
+ new BeanPropertyValueChangeClosure( "activeEmployee", Boolean.TRUE );
+
+ // update the Collection
+ CollectionUtils.forAllDo( peopleCollection, closure );
+ </pre></code>
+</p>
+
+<a name="bean-property-predicate"></a>
+<h3>Querying Or Filtering Collections Of Beans</h3>
+<p>
+The <code>Predicate</code> interface in <code>commons-collections</code> encapsulates an evaluation
+of an input Object that returns either true or false. <code>Commons-collections</code> contains code
+that allows
+<code>Predicates</code> to be applied to be used to filter collections. For more details, see the
+<a href='http://jakarta.apache.org/commons/collections.html'>commons-collections</a>
+documentation.
+</p>
+<p>
+<code>BeanPropertyValueEqualsPredicate</code> is a <code>Predicate</code> that evaluates a
+set property value against a given value. A typical usage is
+(in combination with <code>commons-collections</code>)
+to filter collections on the basis of a property value.
+</p>
+<p>
+For example, to filter a collection to find all beans where active employee is false use:
+<code><pre>
+ BeanPropertyValueEqualsPredicate predicate =
+ new BeanPropertyValueEqualsPredicate( "activeEmployee", Boolean.FALSE );
+
+ // filter the Collection
+ CollectionUtils.filter( peopleCollection, predicate );
+</pre></code>
+</p>
+
+<a href="bean-property-transformer"></a>
+<h3>Transforming Collections Of Beans</h3>
+<p>
+The <code>Transformer</code> interface in <code>commons-collections</code> encapsulates the transformation
+of an input Object into an output object. <code>Commons-collections</code> contains code
+that allows
+<code>Transformers</code> to be applied produce a collection of outputs from a collection of inputs.
+For more details, see the
+<a href='http://jakarta.apache.org/commons/collections.html'>commons-collections</a>
+documentation.
+</p>
+<p>
+<code>BeanToPropertyTransformer</code> is a <code>Transformer</code> implementation
+that transforms a bean into it's property value.
+</p>
+<p>
+For example, to find all cities that are contained in the address of each person property of each bean in
+a collection:
+ <code><pre>
+ // create the transformer
+ BeanToPropertyValueTransformer transformer = new BeanToPropertyValueTransformer( "person.address.city" );
+
+ // transform the Collection
+ Collection peoplesCities = CollectionUtils.collect( peopleCollection, transformer );
+ </pre></code>
+</p>
+
+<a name="FAQ"></a>
+<h1>Frequently Asked Questions</h1>
+
+<a name="FAQ.property"></a>
+<h3>Why Can't BeanUtils Find My Method?</h3>
+<p>The <em>BeanUtils</em> package relies on <em>introspection</em> rather than
+<em>reflection</em>. This means that it will find only
+<a href='http://java.sun.com/products/javabeans'><em>JavaBean</em>
+compliant</a> properties.</p>
+<p>There are some subtleties of this specification that can catch out the unwary:
+<ul>
+<li>A property can have only one set and one get method. Overloading is not allowed.</li>
+<li>The <code>java.beans.Introspector</code> searches widely for a custom <em>BeanInfo</em>
+class. If your class has the same name as another with a custom <em>BeanUtils</em>
+(typically a java API class) then the <code>Introspector</code> may use that instead of
+creating via reflection based on your class. If this happens, the only solution is to
+create your own <em>BeanInfo</em>.</li>
+</ul>
+</p>
+<a name="FAQ.bc.order"></a>
+<h3>How Do I Set The BeanComparator Order To Be Ascending/Descending?</h3>
+<p>
+BeanComparator relies on an internal Comparator to perform the actual
+comparisions. By default,
+<code>org.apache.commons.collections.comparators.ComparableComparator</code>
+is used which imposes a natural order. If you want to change the order,
+then a custom Comparator should be created and passed into the
+appropriate constructor.
+</p>
+<p>
+For example:
+</p>
+<code><pre>
+ import org.apache.commons.collections.comparators.ComparableComparator;
+ import org.apache.commons.collections.comparators.ReverseComparator;
+ import org.apache.commons.beanutils.BeanComparator;
+ ...
+ BeanComparator reversedNaturalOrderBeanComparator
+ = new BeanComparator("propertyName", new ReverseComparator(new ComparableComparator()));
+ Collections.sort(myList, reversedNaturalOrderBeanComparator);
+ ...
+</pre></code>
+
+<a name="Packaging"></a>
+<h1>Packaging</h1>
+<p>
+BeanUtils now comes packaged (into jars) in two different ways:
+</p>
+<ol>
+ <li>an all-in-one jar (<code>commons-beanutils.jar</code>)</li>
+ <li>and as modular component jars:
+ <ul>
+ <li><code>commons-beanutils-core.jar</code> (core classes)</li>
+ <li><code>commons-beanutils-bean-collections.jar</code>
+ (additional dependency on commons-collections 3.0)</li>
+</ol>
+<p>
+Those who want it all should grab the all-in-one <code>commons-beanutils.jar</code>
+(and remember to add any optional dependencies needed)
+whereas those who need minimal dependencies should use <code>commons-beanutils-core.jar</code>
+plus any optional jars they plan to use.
+</p>
+<p>
+The binary distribution contains all these jars.
+</p>
+</body>
+</html>
diff --git a/trunk/src/java/org/apache/commons/collections/ArrayStack.java b/trunk/src/java/org/apache/commons/collections/ArrayStack.java
new file mode 100644
index 0000000..a674c29
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/collections/ArrayStack.java
@@ -0,0 +1,199 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.collections;
+
+import java.util.ArrayList;
+import java.util.EmptyStackException;
+
+/**
+ * An implementation of the {@link java.util.Stack} API that is based on an
+ * <code>ArrayList</code> instead of a <code>Vector</code>, so it is not
+ * synchronized to protect against multi-threaded access. The implementation
+ * is therefore operates faster in environments where you do not need to
+ * worry about multiple thread contention.
+ * <p>
+ * The removal order of an <code>ArrayStack</code> is based on insertion
+ * order: The most recently added element is removed first. The iteration
+ * order is <i>not</i> the same as the removal order. The iterator returns
+ * elements from the bottom up, whereas the {@link #remove()} method removes
+ * them from the top down.
+ * <p>
+ * Unlike <code>Stack</code>, <code>ArrayStack</code> accepts null entries.
+ * <p>
+ * <strong>Note:</strong> this class should be bytecode-identical to the
+ * version in commons collections. This is required to allow backwards
+ * compability with both previous versions of BeanUtils and also allow
+ * coexistance with both collections 2.1 and 3.0.
+ *
+ * @see java.util.Stack
+ * @since Commons Collections 1.0
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 19:50:13 $
+ *
+ * @author Craig R. McClanahan
+ * @author Paul Jack
+ * @author Stephen Colebourne
+ */
+public class ArrayStack extends ArrayList implements Buffer {
+
+ /** Ensure serialization compatibility */
+ private static final long serialVersionUID = 2130079159931574599L;
+
+ /**
+ * Constructs a new empty <code>ArrayStack</code>. The initial size
+ * is controlled by <code>ArrayList</code> and is currently 10.
+ */
+ public ArrayStack() {
+ super();
+ }
+
+ /**
+ * Constructs a new empty <code>ArrayStack</code> with an initial size.
+ *
+ * @param initialSize the initial size to use
+ * @throws IllegalArgumentException if the specified initial size
+ * is negative
+ */
+ public ArrayStack(int initialSize) {
+ super(initialSize);
+ }
+
+ /**
+ * Return <code>true</code> if this stack is currently empty.
+ * <p>
+ * This method exists for compatibility with <code>java.util.Stack</code>.
+ * New users of this class should use <code>isEmpty</code> instead.
+ *
+ * @return true if the stack is currently empty
+ */
+ public boolean empty() {
+ return isEmpty();
+ }
+
+ /**
+ * Returns the top item off of this stack without removing it.
+ *
+ * @return the top item on the stack
+ * @throws EmptyStackException if the stack is empty
+ */
+ public Object peek() throws EmptyStackException {
+ int n = size();
+ if (n <= 0) {
+ throw new EmptyStackException();
+ } else {
+ return get(n - 1);
+ }
+ }
+
+ /**
+ * Returns the n'th item down (zero-relative) from the top of this
+ * stack without removing it.
+ *
+ * @param n the number of items down to go
+ * @return the n'th item on the stack, zero relative
+ * @throws EmptyStackException if there are not enough items on the
+ * stack to satisfy this request
+ */
+ public Object peek(int n) throws EmptyStackException {
+ int m = (size() - n) - 1;
+ if (m < 0) {
+ throw new EmptyStackException();
+ } else {
+ return get(m);
+ }
+ }
+
+ /**
+ * Pops the top item off of this stack and return it.
+ *
+ * @return the top item on the stack
+ * @throws EmptyStackException if the stack is empty
+ */
+ public Object pop() throws EmptyStackException {
+ int n = size();
+ if (n <= 0) {
+ throw new EmptyStackException();
+ } else {
+ return remove(n - 1);
+ }
+ }
+
+ /**
+ * Pushes a new item onto the top of this stack. The pushed item is also
+ * returned. This is equivalent to calling <code>add</code>.
+ *
+ * @param item the item to be added
+ * @return the item just pushed
+ */
+ public Object push(Object item) {
+ add(item);
+ return item;
+ }
+
+ /**
+ * Returns the one-based position of the distance from the top that the
+ * specified object exists on this stack, where the top-most element is
+ * considered to be at distance <code>1</code>. If the object is not
+ * present on the stack, return <code>-1</code> instead. The
+ * <code>equals()</code> method is used to compare to the items
+ * in this stack.
+ *
+ * @param object the object to be searched for
+ * @return the 1-based depth into the stack of the object, or -1 if not found
+ */
+ public int search(Object object) {
+ int i = size() - 1; // Current index
+ int n = 1; // Current distance
+ while (i >= 0) {
+ Object current = get(i);
+ if ((object == null && current == null) ||
+ (object != null && object.equals(current))) {
+ return n;
+ }
+ i--;
+ n++;
+ }
+ return -1;
+ }
+
+ /**
+ * Returns the element on the top of the stack.
+ *
+ * @return the element on the top of the stack
+ * @throws BufferUnderflowException if the stack is empty
+ */
+ public Object get() {
+ int size = size();
+ if (size == 0) {
+ throw new BufferUnderflowException();
+ }
+ return get(size - 1);
+ }
+
+ /**
+ * Removes the element on the top of the stack.
+ *
+ * @return the removed element
+ * @throws BufferUnderflowException if the stack is empty
+ */
+ public Object remove() {
+ int size = size();
+ if (size == 0) {
+ throw new BufferUnderflowException();
+ }
+ return remove(size - 1);
+ }
+
+}
diff --git a/trunk/src/java/org/apache/commons/collections/Buffer.java b/trunk/src/java/org/apache/commons/collections/Buffer.java
new file mode 100644
index 0000000..99aaaa8
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/collections/Buffer.java
@@ -0,0 +1,68 @@
+/*
+ * Copyright 2002-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.collections;
+
+import java.util.Collection;
+
+/**
+ * Defines a collection that allows objects to be removed in some well-defined order.
+ * <p>
+ * The removal order can be based on insertion order (eg, a FIFO queue or a
+ * LIFO stack), on access order (eg, an LRU cache), on some arbitrary comparator
+ * (eg, a priority queue) or on any other well-defined ordering.
+ * <p>
+ * Note that the removal order is not necessarily the same as the iteration
+ * order. A <code>Buffer</code> implementation may have equivalent removal
+ * and iteration orders, but this is not required.
+ * <p>
+ * This interface does not specify any behavior for
+ * {@link Object#equals(Object)} and {@link Object#hashCode} methods. It
+ * is therefore possible for a <code>Buffer</code> implementation to also
+ * also implement {@link java.util.List}, {@link java.util.Set} or
+ * {@link Bag}.
+ * <p>
+ * <strong>Note:</strong> this class should be bytecode-identical to the
+ * version in commons collections. This is required to allow backwards
+ * compability with both previous versions of BeanUtils and also allow
+ * coexistance with both collections 2.1 and 3.0.
+ *
+ * @since Commons Collections 2.1
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 19:50:28 $
+ *
+ * @author Avalon
+ * @author Berin Loritsch
+ * @author Paul Jack
+ * @author Stephen Colebourne
+ */
+public interface Buffer extends Collection {
+
+ /**
+ * Gets and removes the next object from the buffer.
+ *
+ * @return the next object in the buffer, which is also removed
+ * @throws BufferUnderflowException if the buffer is already empty
+ */
+ Object remove();
+
+ /**
+ * Gets the next object from the buffer without removing it.
+ *
+ * @return the next object in the buffer, which is not removed
+ * @throws BufferUnderflowException if the buffer is empty
+ */
+ Object get();
+
+}
diff --git a/trunk/src/java/org/apache/commons/collections/BufferUnderflowException.java b/trunk/src/java/org/apache/commons/collections/BufferUnderflowException.java
new file mode 100644
index 0000000..d35e50d
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/collections/BufferUnderflowException.java
@@ -0,0 +1,76 @@
+/*
+ * Copyright 2002-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.collections;
+
+import java.util.NoSuchElementException;
+
+/**
+ * The BufferUnderflowException is used when the buffer is already empty.
+ * <p>
+ * NOTE: From version 3.0, this exception extends NoSuchElementException.
+ *
+ * @since Commons Collections 2.1
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 19:50:52 $
+ *
+ * @author Avalon
+ * @author Berin Loritsch
+ * @author Jeff Turner
+ * @author Paul Jack
+ * @author Stephen Colebourne
+ */
+public class BufferUnderflowException extends NoSuchElementException {
+
+ /** The root cause throwable */
+ private final Throwable throwable;
+
+ /**
+ * Constructs a new <code>BufferUnderflowException</code>.
+ */
+ public BufferUnderflowException() {
+ super();
+ throwable = null;
+ }
+
+ /**
+ * Construct a new <code>BufferUnderflowException</code>.
+ *
+ * @param message the detail message for this exception
+ */
+ public BufferUnderflowException(String message) {
+ this(message, null);
+ }
+
+ /**
+ * Construct a new <code>BufferUnderflowException</code>.
+ *
+ * @param message the detail message for this exception
+ * @param exception the root cause of the exception
+ */
+ public BufferUnderflowException(String message, Throwable exception) {
+ super(message);
+ throwable = exception;
+ }
+
+ /**
+ * Gets the root cause of the exception.
+ *
+ * @return the root cause
+ */
+ public final Throwable getCause() {
+ return throwable;
+ }
+
+}
diff --git a/trunk/src/java/org/apache/commons/collections/FastHashMap.java b/trunk/src/java/org/apache/commons/collections/FastHashMap.java
new file mode 100644
index 0000000..4398ef9
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/collections/FastHashMap.java
@@ -0,0 +1,714 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.collections;
+
+import java.util.Collection;
+import java.util.ConcurrentModificationException;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.Map;
+import java.util.Set;
+
+/**
+ * <p>A customized implementation of <code>java.util.HashMap</code> designed
+ * to operate in a multithreaded environment where the large majority of
+ * method calls are read-only, instead of structural changes. When operating
+ * in "fast" mode, read calls are non-synchronized and write calls perform the
+ * following steps:</p>
+ * <ul>
+ * <li>Clone the existing collection
+ * <li>Perform the modification on the clone
+ * <li>Replace the existing collection with the (modified) clone
+ * </ul>
+ * <p>When first created, objects of this class default to "slow" mode, where
+ * all accesses of any type are synchronized but no cloning takes place. This
+ * is appropriate for initially populating the collection, followed by a switch
+ * to "fast" mode (by calling <code>setFast(true)</code>) after initialization
+ * is complete.</p>
+ *
+ * <p><strong>NOTE</strong>: If you are creating and accessing a
+ * <code>HashMap</code> only within a single thread, you should use
+ * <code>java.util.HashMap</code> directly (with no synchronization), for
+ * maximum performance.</p>
+ *
+ * <p><strong>NOTE</strong>: <i>This class is not cross-platform.
+ * Using it may cause unexpected failures on some architectures.</i>
+ * It suffers from the same problems as the double-checked locking idiom.
+ * In particular, the instruction that clones the internal collection and the
+ * instruction that sets the internal reference to the clone can be executed
+ * or perceived out-of-order. This means that any read operation might fail
+ * unexpectedly, as it may be reading the state of the internal collection
+ * before the internal collection is fully formed.
+ * For more information on the double-checked locking idiom, see the
+ * <a href="http://www.cs.umd.edu/~pugh/java/memoryModel/DoubleCheckedLocking.html">
+ * Double-Checked Locking Idiom Is Broken Declaration</a>.</p>
+ *
+ * @since Commons Collections 1.0
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 19:51:13 $
+ *
+ * @author Craig R. McClanahan
+ * @author Stephen Colebourne
+ */
+public class FastHashMap extends HashMap {
+
+ /**
+ * The underlying map we are managing.
+ */
+ protected HashMap map = null;
+
+ /**
+ * Are we currently operating in "fast" mode?
+ */
+ protected boolean fast = false;
+
+ // Constructors
+ // ----------------------------------------------------------------------
+
+ /**
+ * Construct an empty map.
+ */
+ public FastHashMap() {
+ super();
+ this.map = new HashMap();
+ }
+
+ /**
+ * Construct an empty map with the specified capacity.
+ *
+ * @param capacity the initial capacity of the empty map
+ */
+ public FastHashMap(int capacity) {
+ super();
+ this.map = new HashMap(capacity);
+ }
+
+ /**
+ * Construct an empty map with the specified capacity and load factor.
+ *
+ * @param capacity the initial capacity of the empty map
+ * @param factor the load factor of the new map
+ */
+ public FastHashMap(int capacity, float factor) {
+ super();
+ this.map = new HashMap(capacity, factor);
+ }
+
+ /**
+ * Construct a new map with the same mappings as the specified map.
+ *
+ * @param map the map whose mappings are to be copied
+ */
+ public FastHashMap(Map map) {
+ super();
+ this.map = new HashMap(map);
+ }
+
+
+ // Property access
+ // ----------------------------------------------------------------------
+
+ /**
+ * Returns true if this map is operating in fast mode.
+ *
+ * @return true if this map is operating in fast mode
+ */
+ public boolean getFast() {
+ return (this.fast);
+ }
+
+ /**
+ * Sets whether this map is operating in fast mode.
+ *
+ * @param fast true if this map should operate in fast mode
+ */
+ public void setFast(boolean fast) {
+ this.fast = fast;
+ }
+
+
+ // Map access
+ // ----------------------------------------------------------------------
+ // These methods can forward straight to the wrapped Map in 'fast' mode.
+ // (because they are query methods)
+
+ /**
+ * Return the value to which this map maps the specified key. Returns
+ * <code>null</code> if the map contains no mapping for this key, or if
+ * there is a mapping with a value of <code>null</code>. Use the
+ * <code>containsKey()</code> method to disambiguate these cases.
+ *
+ * @param key the key whose value is to be returned
+ * @return the value mapped to that key, or null
+ */
+ public Object get(Object key) {
+ if (fast) {
+ return (map.get(key));
+ } else {
+ synchronized (map) {
+ return (map.get(key));
+ }
+ }
+ }
+
+ /**
+ * Return the number of key-value mappings in this map.
+ *
+ * @return the current size of the map
+ */
+ public int size() {
+ if (fast) {
+ return (map.size());
+ } else {
+ synchronized (map) {
+ return (map.size());
+ }
+ }
+ }
+
+ /**
+ * Return <code>true</code> if this map contains no mappings.
+ *
+ * @return is the map currently empty
+ */
+ public boolean isEmpty() {
+ if (fast) {
+ return (map.isEmpty());
+ } else {
+ synchronized (map) {
+ return (map.isEmpty());
+ }
+ }
+ }
+
+ /**
+ * Return <code>true</code> if this map contains a mapping for the
+ * specified key.
+ *
+ * @param key the key to be searched for
+ * @return true if the map contains the key
+ */
+ public boolean containsKey(Object key) {
+ if (fast) {
+ return (map.containsKey(key));
+ } else {
+ synchronized (map) {
+ return (map.containsKey(key));
+ }
+ }
+ }
+
+ /**
+ * Return <code>true</code> if this map contains one or more keys mapping
+ * to the specified value.
+ *
+ * @param value the value to be searched for
+ * @return true if the map contains the value
+ */
+ public boolean containsValue(Object value) {
+ if (fast) {
+ return (map.containsValue(value));
+ } else {
+ synchronized (map) {
+ return (map.containsValue(value));
+ }
+ }
+ }
+
+ // Map modification
+ // ----------------------------------------------------------------------
+ // These methods perform special behaviour in 'fast' mode.
+ // The map is cloned, updated and then assigned back.
+ // See the comments at the top as to why this won't always work.
+
+ /**
+ * Associate the specified value with the specified key in this map.
+ * If the map previously contained a mapping for this key, the old
+ * value is replaced and returned.
+ *
+ * @param key the key with which the value is to be associated
+ * @param value the value to be associated with this key
+ * @return the value previously mapped to the key, or null
+ */
+ public Object put(Object key, Object value) {
+ if (fast) {
+ synchronized (this) {
+ HashMap temp = (HashMap) map.clone();
+ Object result = temp.put(key, value);
+ map = temp;
+ return (result);
+ }
+ } else {
+ synchronized (map) {
+ return (map.put(key, value));
+ }
+ }
+ }
+
+ /**
+ * Copy all of the mappings from the specified map to this one, replacing
+ * any mappings with the same keys.
+ *
+ * @param in the map whose mappings are to be copied
+ */
+ public void putAll(Map in) {
+ if (fast) {
+ synchronized (this) {
+ HashMap temp = (HashMap) map.clone();
+ temp.putAll(in);
+ map = temp;
+ }
+ } else {
+ synchronized (map) {
+ map.putAll(in);
+ }
+ }
+ }
+
+ /**
+ * Remove any mapping for this key, and return any previously
+ * mapped value.
+ *
+ * @param key the key whose mapping is to be removed
+ * @return the value removed, or null
+ */
+ public Object remove(Object key) {
+ if (fast) {
+ synchronized (this) {
+ HashMap temp = (HashMap) map.clone();
+ Object result = temp.remove(key);
+ map = temp;
+ return (result);
+ }
+ } else {
+ synchronized (map) {
+ return (map.remove(key));
+ }
+ }
+ }
+
+ /**
+ * Remove all mappings from this map.
+ */
+ public void clear() {
+ if (fast) {
+ synchronized (this) {
+ map = new HashMap();
+ }
+ } else {
+ synchronized (map) {
+ map.clear();
+ }
+ }
+ }
+
+ // Basic object methods
+ // ----------------------------------------------------------------------
+
+ /**
+ * Compare the specified object with this list for equality. This
+ * implementation uses exactly the code that is used to define the
+ * list equals function in the documentation for the
+ * <code>Map.equals</code> method.
+ *
+ * @param o the object to be compared to this list
+ * @return true if the two maps are equal
+ */
+ public boolean equals(Object o) {
+ // Simple tests that require no synchronization
+ if (o == this) {
+ return (true);
+ } else if (!(o instanceof Map)) {
+ return (false);
+ }
+ Map mo = (Map) o;
+
+ // Compare the two maps for equality
+ if (fast) {
+ if (mo.size() != map.size()) {
+ return (false);
+ }
+ Iterator i = map.entrySet().iterator();
+ while (i.hasNext()) {
+ Map.Entry e = (Map.Entry) i.next();
+ Object key = e.getKey();
+ Object value = e.getValue();
+ if (value == null) {
+ if (!(mo.get(key) == null && mo.containsKey(key))) {
+ return (false);
+ }
+ } else {
+ if (!value.equals(mo.get(key))) {
+ return (false);
+ }
+ }
+ }
+ return (true);
+
+ } else {
+ synchronized (map) {
+ if (mo.size() != map.size()) {
+ return (false);
+ }
+ Iterator i = map.entrySet().iterator();
+ while (i.hasNext()) {
+ Map.Entry e = (Map.Entry) i.next();
+ Object key = e.getKey();
+ Object value = e.getValue();
+ if (value == null) {
+ if (!(mo.get(key) == null && mo.containsKey(key))) {
+ return (false);
+ }
+ } else {
+ if (!value.equals(mo.get(key))) {
+ return (false);
+ }
+ }
+ }
+ return (true);
+ }
+ }
+ }
+
+ /**
+ * Return the hash code value for this map. This implementation uses
+ * exactly the code that is used to define the list hash function in the
+ * documentation for the <code>Map.hashCode</code> method.
+ *
+ * @return suitable integer hash code
+ */
+ public int hashCode() {
+ if (fast) {
+ int h = 0;
+ Iterator i = map.entrySet().iterator();
+ while (i.hasNext()) {
+ h += i.next().hashCode();
+ }
+ return (h);
+ } else {
+ synchronized (map) {
+ int h = 0;
+ Iterator i = map.entrySet().iterator();
+ while (i.hasNext()) {
+ h += i.next().hashCode();
+ }
+ return (h);
+ }
+ }
+ }
+
+ /**
+ * Return a shallow copy of this <code>FastHashMap</code> instance.
+ * The keys and values themselves are not copied.
+ *
+ * @return a clone of this map
+ */
+ public Object clone() {
+ FastHashMap results = null;
+ if (fast) {
+ results = new FastHashMap(map);
+ } else {
+ synchronized (map) {
+ results = new FastHashMap(map);
+ }
+ }
+ results.setFast(getFast());
+ return (results);
+ }
+
+ // Map views
+ // ----------------------------------------------------------------------
+
+ /**
+ * Return a collection view of the mappings contained in this map. Each
+ * element in the returned collection is a <code>Map.Entry</code>.
+ */
+ public Set entrySet() {
+ return new EntrySet();
+ }
+
+ /**
+ * Return a set view of the keys contained in this map.
+ */
+ public Set keySet() {
+ return new KeySet();
+ }
+
+ /**
+ * Return a collection view of the values contained in this map.
+ */
+ public Collection values() {
+ return new Values();
+ }
+
+ // Map view inner classes
+ // ----------------------------------------------------------------------
+
+ /**
+ * Abstract collection implementation shared by keySet(), values() and entrySet().
+ */
+ private abstract class CollectionView implements Collection {
+
+ public CollectionView() {
+ }
+
+ protected abstract Collection get(Map map);
+ protected abstract Object iteratorNext(Map.Entry entry);
+
+
+ public void clear() {
+ if (fast) {
+ synchronized (FastHashMap.this) {
+ map = new HashMap();
+ }
+ } else {
+ synchronized (map) {
+ get(map).clear();
+ }
+ }
+ }
+
+ public boolean remove(Object o) {
+ if (fast) {
+ synchronized (FastHashMap.this) {
+ HashMap temp = (HashMap) map.clone();
+ boolean r = get(temp).remove(o);
+ map = temp;
+ return r;
+ }
+ } else {
+ synchronized (map) {
+ return get(map).remove(o);
+ }
+ }
+ }
+
+ public boolean removeAll(Collection o) {
+ if (fast) {
+ synchronized (FastHashMap.this) {
+ HashMap temp = (HashMap) map.clone();
+ boolean r = get(temp).removeAll(o);
+ map = temp;
+ return r;
+ }
+ } else {
+ synchronized (map) {
+ return get(map).removeAll(o);
+ }
+ }
+ }
+
+ public boolean retainAll(Collection o) {
+ if (fast) {
+ synchronized (FastHashMap.this) {
+ HashMap temp = (HashMap) map.clone();
+ boolean r = get(temp).retainAll(o);
+ map = temp;
+ return r;
+ }
+ } else {
+ synchronized (map) {
+ return get(map).retainAll(o);
+ }
+ }
+ }
+
+ public int size() {
+ if (fast) {
+ return get(map).size();
+ } else {
+ synchronized (map) {
+ return get(map).size();
+ }
+ }
+ }
+
+
+ public boolean isEmpty() {
+ if (fast) {
+ return get(map).isEmpty();
+ } else {
+ synchronized (map) {
+ return get(map).isEmpty();
+ }
+ }
+ }
+
+ public boolean contains(Object o) {
+ if (fast) {
+ return get(map).contains(o);
+ } else {
+ synchronized (map) {
+ return get(map).contains(o);
+ }
+ }
+ }
+
+ public boolean containsAll(Collection o) {
+ if (fast) {
+ return get(map).containsAll(o);
+ } else {
+ synchronized (map) {
+ return get(map).containsAll(o);
+ }
+ }
+ }
+
+ public Object[] toArray(Object[] o) {
+ if (fast) {
+ return get(map).toArray(o);
+ } else {
+ synchronized (map) {
+ return get(map).toArray(o);
+ }
+ }
+ }
+
+ public Object[] toArray() {
+ if (fast) {
+ return get(map).toArray();
+ } else {
+ synchronized (map) {
+ return get(map).toArray();
+ }
+ }
+ }
+
+
+ public boolean equals(Object o) {
+ if (o == this) return true;
+ if (fast) {
+ return get(map).equals(o);
+ } else {
+ synchronized (map) {
+ return get(map).equals(o);
+ }
+ }
+ }
+
+ public int hashCode() {
+ if (fast) {
+ return get(map).hashCode();
+ } else {
+ synchronized (map) {
+ return get(map).hashCode();
+ }
+ }
+ }
+
+ public boolean add(Object o) {
+ throw new UnsupportedOperationException();
+ }
+
+ public boolean addAll(Collection c) {
+ throw new UnsupportedOperationException();
+ }
+
+ public Iterator iterator() {
+ return new CollectionViewIterator();
+ }
+
+ private class CollectionViewIterator implements Iterator {
+
+ private Map expected;
+ private Map.Entry lastReturned = null;
+ private Iterator iterator;
+
+ public CollectionViewIterator() {
+ this.expected = map;
+ this.iterator = expected.entrySet().iterator();
+ }
+
+ public boolean hasNext() {
+ if (expected != map) {
+ throw new ConcurrentModificationException();
+ }
+ return iterator.hasNext();
+ }
+
+ public Object next() {
+ if (expected != map) {
+ throw new ConcurrentModificationException();
+ }
+ lastReturned = (Map.Entry)iterator.next();
+ return iteratorNext(lastReturned);
+ }
+
+ public void remove() {
+ if (lastReturned == null) {
+ throw new IllegalStateException();
+ }
+ if (fast) {
+ synchronized (FastHashMap.this) {
+ if (expected != map) {
+ throw new ConcurrentModificationException();
+ }
+ FastHashMap.this.remove(lastReturned.getKey());
+ lastReturned = null;
+ expected = map;
+ }
+ } else {
+ iterator.remove();
+ lastReturned = null;
+ }
+ }
+ }
+ }
+
+ /**
+ * Set implementation over the keys of the FastHashMap
+ */
+ private class KeySet extends CollectionView implements Set {
+
+ protected Collection get(Map map) {
+ return map.keySet();
+ }
+
+ protected Object iteratorNext(Map.Entry entry) {
+ return entry.getKey();
+ }
+
+ }
+
+ /**
+ * Collection implementation over the values of the FastHashMap
+ */
+ private class Values extends CollectionView {
+
+ protected Collection get(Map map) {
+ return map.values();
+ }
+
+ protected Object iteratorNext(Map.Entry entry) {
+ return entry.getValue();
+ }
+ }
+
+ /**
+ * Set implementation over the entries of the FastHashMap
+ */
+ private class EntrySet extends CollectionView implements Set {
+
+ protected Collection get(Map map) {
+ return map.entrySet();
+ }
+
+ protected Object iteratorNext(Map.Entry entry) {
+ return entry;
+ }
+
+ }
+
+}
diff --git a/trunk/src/java/org/apache/commons/collections/package.html b/trunk/src/java/org/apache/commons/collections/package.html
new file mode 100644
index 0000000..1f128fc
--- /dev/null
+++ b/trunk/src/java/org/apache/commons/collections/package.html
@@ -0,0 +1,13 @@
+<html>
+<head>
+<title>Package Documentation for org.apache.commons.collections Package</title>
+</head>
+<body bgcolor="white">
+<p>
+This package contains a small number of collections classes that are needed for beanutils
+to function. The dependencies will be deprecated in the 1.7.0 release and the need for these
+classes will be removed in the future. These classes are identical to those in both 2.x and 3.x
+series of commons-collections releases.
+</p>
+</body>
+</html>
diff --git a/trunk/src/java/overview.html b/trunk/src/java/overview.html
new file mode 100644
index 0000000..2bf0fb7
--- /dev/null
+++ b/trunk/src/java/overview.html
@@ -0,0 +1,17 @@
+<html>
+<head>
+<title>Overview Documentation for COMMONS-BEANUTILS</title>
+</head>
+<body bgcolor="white">
+<p>The <em>Bean Introspection Utilities</em> component of the Jakarta Commons
+subproject offers low-level utility classes that assist in getting and setting
+property values on Java classes that follow the naming design patterns outlined
+in the JavaBeans Specification, as well as mechanisms for dynamically defining
+and accessing bean properties.</p>
+
+<p>See the
+<a href="org/apache/commons/beanutils/package-summary.html#package_description">
+Package Description</a> for the <code>org.apache.commons.beanutils</code>
+package for more information.</p>
+</body>
+</html>
diff --git a/trunk/src/media/logo.xcf b/trunk/src/media/logo.xcf
new file mode 100644
index 0000000..246d769
Binary files /dev/null and b/trunk/src/media/logo.xcf differ
diff --git a/trunk/src/test/org/apache/commons/beanutils/A.java b/trunk/src/test/org/apache/commons/beanutils/A.java
new file mode 100644
index 0000000..716986e
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/A.java
@@ -0,0 +1,34 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+import java.io.OutputStream;
+
+/**
+ * <p>Class used in MethodUtils test</p>
+ *
+ */
+public class A {
+
+ boolean called = false;
+
+ public void foo(OutputStream os)
+ {
+ called = true;
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/AbstractChild.java b/trunk/src/test/org/apache/commons/beanutils/AbstractChild.java
new file mode 100644
index 0000000..d301116
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/AbstractChild.java
@@ -0,0 +1,33 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+public class AbstractChild implements Child {
+
+ private String name;
+
+ protected void setName(String name)
+ {
+ this.name = name;
+ }
+
+ public String getName()
+ {
+ return name;
+ }
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/AbstractParent.java b/trunk/src/test/org/apache/commons/beanutils/AbstractParent.java
new file mode 100644
index 0000000..0687a3f
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/AbstractParent.java
@@ -0,0 +1,53 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+public abstract class AbstractParent {
+
+ private Child child;
+
+ public Child getChild()
+ {
+ return child;
+ }
+
+ /**
+ * Method which matches signature but which has wrong parameters
+ */
+ public String testAddChild(String badParameter) {
+ return null;
+ }
+
+ /**
+ * Method which matches signature but which has wrong parameters
+ */
+ public String testAddChild2(String ignore, String badParameter) {
+ return null;
+ }
+
+ public String testAddChild(Child child) {
+ this.child = child;
+ return child.getName();
+ }
+
+
+ public String testAddChild2(String ignore, Child child) {
+ this.child = child;
+ return child.getName();
+ }
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/AlphaBean.java b/trunk/src/test/org/apache/commons/beanutils/AlphaBean.java
new file mode 100644
index 0000000..50bf3c5
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/AlphaBean.java
@@ -0,0 +1,41 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+public class AlphaBean extends AbstractParent implements Child {
+
+ private String name;
+
+ public AlphaBean() {}
+
+ public AlphaBean(String name) {
+ setName(name);
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public void setName(String name) {
+ this.name = name;
+ }
+
+ /**
+ * Used for testing that correct exception is thrown.
+ */
+ public void bogus(String badParameter){}
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/BasicDynaBeanTestCase.java b/trunk/src/test/org/apache/commons/beanutils/BasicDynaBeanTestCase.java
new file mode 100644
index 0000000..716ebbc
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/BasicDynaBeanTestCase.java
@@ -0,0 +1,1030 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.io.ByteArrayInputStream;
+import java.io.ByteArrayOutputStream;
+import java.io.ObjectInputStream;
+import java.io.ObjectOutputStream;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * <p>Test Case for the <code>BasicDynaBean</code> implementation class.
+ * These tests were based on the ones in <code>PropertyUtilsTestCase</code>
+ * because the two classes provide similar levels of functionality.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.10 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class BasicDynaBeanTestCase extends TestCase {
+
+
+ // ---------------------------------------------------- Instance Variables
+
+
+ /**
+ * The basic test bean for each test.
+ */
+ protected DynaBean bean = null;
+
+
+ /**
+ * The set of property names we expect to have returned when calling
+ * <code>getDynaProperties()</code>. You should update this list
+ * when new properties are added to TestBean.
+ */
+ protected final static String[] properties = {
+ "booleanProperty",
+ "booleanSecond",
+ "doubleProperty",
+ "floatProperty",
+ "intArray",
+ "intIndexed",
+ "intProperty",
+ "listIndexed",
+ "longProperty",
+ "mappedProperty",
+ "mappedIntProperty",
+ "nullProperty",
+ "shortProperty",
+ "stringArray",
+ "stringIndexed",
+ "stringProperty",
+ };
+
+
+ // ---------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public BasicDynaBeanTestCase(String name) {
+
+ super(name);
+
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+
+ // Instantiate a new DynaBean instance
+ DynaClass dynaClass = createDynaClass();
+ bean = dynaClass.newInstance();
+
+ // Initialize the DynaBean's property values (like TestBean)
+ bean.set("booleanProperty", new Boolean(true));
+ bean.set("booleanSecond", new Boolean(true));
+ bean.set("doubleProperty", new Double(321.0));
+ bean.set("floatProperty", new Float((float) 123.0));
+ int intArray[] = { 0, 10, 20, 30, 40 };
+ bean.set("intArray", intArray);
+ int intIndexed[] = { 0, 10, 20, 30, 40 };
+ bean.set("intIndexed", intIndexed);
+ bean.set("intProperty", new Integer(123));
+ List listIndexed = new ArrayList();
+ listIndexed.add("String 0");
+ listIndexed.add("String 1");
+ listIndexed.add("String 2");
+ listIndexed.add("String 3");
+ listIndexed.add("String 4");
+ bean.set("listIndexed", listIndexed);
+ bean.set("longProperty", new Long((long) 321));
+ HashMap mappedProperty = new HashMap();
+ mappedProperty.put("First Key", "First Value");
+ mappedProperty.put("Second Key", "Second Value");
+ bean.set("mappedProperty", mappedProperty);
+ HashMap mappedIntProperty = new HashMap();
+ mappedIntProperty.put("One", new Integer(1));
+ mappedIntProperty.put("Two", new Integer(2));
+ bean.set("mappedIntProperty", mappedIntProperty);
+ // Property "nullProperty" is not initialized, so it should return null
+ bean.set("shortProperty", new Short((short) 987));
+ String stringArray[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+ bean.set("stringArray", stringArray);
+ String stringIndexed[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+ bean.set("stringIndexed", stringIndexed);
+ bean.set("stringProperty", "This is a string");
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+
+ return (new TestSuite(BasicDynaBeanTestCase.class));
+
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+
+ bean = null;
+
+ }
+
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ /**
+ * Corner cases on getDynaProperty invalid arguments.
+ */
+ public void testGetDescriptorArguments() {
+
+ try {
+ DynaProperty descriptor =
+ bean.getDynaClass().getDynaProperty("unknown");
+ assertNull("Unknown property descriptor should be null",
+ descriptor);
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of returning null");
+ }
+
+ try {
+ bean.getDynaClass().getDynaProperty(null);
+ fail("Should throw IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException");
+ }
+
+ }
+
+
+ /**
+ * Positive getDynaProperty on property <code>booleanProperty</code>.
+ */
+ public void testGetDescriptorBoolean() {
+
+ testGetDescriptorBase("booleanProperty", Boolean.TYPE);
+
+ }
+
+
+ /**
+ * Positive getDynaProperty on property <code>doubleProperty</code>.
+ */
+ public void testGetDescriptorDouble() {
+
+ testGetDescriptorBase("doubleProperty", Double.TYPE);
+
+ }
+
+
+ /**
+ * Positive getDynaProperty on property <code>floatProperty</code>.
+ */
+ public void testGetDescriptorFloat() {
+
+ testGetDescriptorBase("floatProperty", Float.TYPE);
+
+ }
+
+
+ /**
+ * Positive getDynaProperty on property <code>intProperty</code>.
+ */
+ public void testGetDescriptorInt() {
+
+ testGetDescriptorBase("intProperty", Integer.TYPE);
+
+ }
+
+
+ /**
+ * Positive getDynaProperty on property <code>longProperty</code>.
+ */
+ public void testGetDescriptorLong() {
+
+ testGetDescriptorBase("longProperty", Long.TYPE);
+
+ }
+
+
+ /**
+ * Positive getDynaProperty on property <code>booleanSecond</code>
+ * that uses an "is" method as the getter.
+ */
+ public void testGetDescriptorSecond() {
+
+ testGetDescriptorBase("booleanSecond", Boolean.TYPE);
+
+ }
+
+
+ /**
+ * Positive getDynaProperty on property <code>shortProperty</code>.
+ */
+ public void testGetDescriptorShort() {
+
+ testGetDescriptorBase("shortProperty", Short.TYPE);
+
+ }
+
+
+ /**
+ * Positive getDynaProperty on property <code>stringProperty</code>.
+ */
+ public void testGetDescriptorString() {
+
+ testGetDescriptorBase("stringProperty", String.class);
+
+ }
+
+
+ /**
+ * Positive test for getDynaPropertys(). Each property name
+ * listed in <code>properties</code> should be returned exactly once.
+ */
+ public void testGetDescriptors() {
+
+ DynaProperty pd[] = bean.getDynaClass().getDynaProperties();
+ assertNotNull("Got descriptors", pd);
+ int count[] = new int[properties.length];
+ for (int i = 0; i < pd.length; i++) {
+ String name = pd[i].getName();
+ for (int j = 0; j < properties.length; j++) {
+ if (name.equals(properties[j]))
+ count[j]++;
+ }
+ }
+ for (int j = 0; j < properties.length; j++) {
+ if (count[j] < 0)
+ fail("Missing property " + properties[j]);
+ else if (count[j] > 1)
+ fail("Duplicate property " + properties[j]);
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getIndexedProperty invalid arguments.
+ */
+ public void testGetIndexedArguments() {
+
+ try {
+ bean.get("intArray", -1);
+ fail("Should throw IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+
+ }
+
+
+ /**
+ * Positive and negative tests on getIndexedProperty valid arguments.
+ */
+ public void testGetIndexedValues() {
+
+ Object value = null;
+
+ for (int i = 0; i < 5; i++) {
+
+ try {
+ value = bean.get("intArray", i);
+ assertNotNull("intArray returned value " + i, value);
+ assertTrue("intArray returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intArray returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intArray " + i + " threw " + t);
+ }
+
+ try {
+ value = bean.get("intIndexed", i);
+ assertNotNull("intIndexed returned value " + i, value);
+ assertTrue("intIndexed returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intIndexed returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value = bean.get("listIndexed", i);
+ assertNotNull("listIndexed returned value " + i, value);
+ assertTrue("list returned String " + i,
+ value instanceof String);
+ assertEquals("listIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("listIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value = bean.get("stringArray", i);
+ assertNotNull("stringArray returned value " + i, value);
+ assertTrue("stringArray returned String " + i,
+ value instanceof String);
+ assertEquals("stringArray returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringArray " + i + " threw " + t);
+ }
+
+ try {
+ value = bean.get("stringIndexed", i);
+ assertNotNull("stringIndexed returned value " + i, value);
+ assertTrue("stringIndexed returned String " + i,
+ value instanceof String);
+ assertEquals("stringIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringIndexed " + i + " threw " + t);
+ }
+
+ }
+
+
+ }
+
+
+ /**
+ * Corner cases on getMappedProperty invalid arguments.
+ */
+ public void testGetMappedArguments() {
+
+
+ try {
+ Object value = bean.get("mappedProperty", "unknown");
+ assertNull("Should not return a value", value);
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of returning null");
+ }
+
+
+ }
+
+
+ /**
+ * Positive and negative tests on getMappedProperty valid arguments.
+ */
+ public void testGetMappedValues() {
+
+ Object value = null;
+
+ try {
+ value = bean.get("mappedProperty", "First Key");
+ assertEquals("Can find first value", "First Value", value);
+ } catch (Throwable t) {
+ fail("Finding first value threw " + t);
+ }
+
+ try {
+ value = bean.get("mappedProperty", "Second Key");
+ assertEquals("Can find second value", "Second Value", value);
+ } catch (Throwable t) {
+ fail("Finding second value threw " + t);
+ }
+
+ try {
+ value = bean.get("mappedProperty", "Third Key");
+ assertNull("Can not find third value", value);
+ } catch (Throwable t) {
+ fail("Finding third value threw " + t);
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getSimpleProperty invalid arguments.
+ */
+ public void testGetSimpleArguments() {
+
+ try {
+ bean.get(null);
+ fail("Should throw IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a boolean property.
+ */
+ public void testGetSimpleBoolean() {
+
+ try {
+ Object value = bean.get("booleanProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Boolean));
+ assertTrue("Got correct value",
+ ((Boolean) value).booleanValue() == true);
+ } catch (Throwable e) {
+ fail("Exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a double property.
+ */
+ public void testGetSimpleDouble() {
+
+ try {
+ Object value = bean.get("doubleProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Double));
+ assertEquals("Got correct value",
+ ((Double) value).doubleValue(),
+ (double) 321.0,
+ (double) 0.005);
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a float property.
+ */
+ public void testGetSimpleFloat() {
+
+ try {
+ Object value = bean.get("floatProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Float));
+ assertEquals("Got correct value",
+ ((Float) value).floatValue(),
+ (float) 123.0,
+ (float) 0.005);
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a int property.
+ */
+ public void testGetSimpleInt() {
+
+ try {
+ Object value = bean.get("intProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Integer));
+ assertEquals("Got correct value",
+ ((Integer) value).intValue(),
+ (int) 123);
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a long property.
+ */
+ public void testGetSimpleLong() {
+
+ try {
+ Object value = bean.get("longProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Long));
+ assertEquals("Got correct value",
+ ((Long) value).longValue(),
+ (long) 321);
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a short property.
+ */
+ public void testGetSimpleShort() {
+
+ try {
+ Object value = bean.get("shortProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Short));
+ assertEquals("Got correct value",
+ ((Short) value).shortValue(),
+ (short) 987);
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a String property.
+ */
+ public void testGetSimpleString() {
+
+ try {
+ Object value = bean.get("stringProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof String));
+ assertEquals("Got correct value",
+ (String) value,
+ "This is a string");
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * Test <code>contains()</code> method for mapped properties.
+ */
+ public void testMappedContains() {
+
+ try {
+ assertTrue("Can see first key",
+ bean.contains("mappedProperty", "First Key"));
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+
+ try {
+ assertTrue("Can not see unknown key",
+ !bean.contains("mappedProperty", "Unknown Key"));
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * Test <code>remove()</code> method for mapped properties.
+ */
+ public void testMappedRemove() {
+
+ try {
+ assertTrue("Can see first key",
+ bean.contains("mappedProperty", "First Key"));
+ bean.remove("mappedProperty", "First Key");
+ assertTrue("Can not see first key",
+ !bean.contains("mappedProperty", "First Key"));
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ try {
+ assertTrue("Can not see unknown key",
+ !bean.contains("mappedProperty", "Unknown Key"));
+ bean.remove("mappedProperty", "Unknown Key");
+ assertTrue("Can not see unknown key",
+ !bean.contains("mappedProperty", "Unknown Key"));
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * Test serialization and deserialization.
+ */
+ public void testSerialization() {
+
+ // Serialize the test bean
+ ByteArrayOutputStream baos = new ByteArrayOutputStream();
+ try {
+ ObjectOutputStream oos = new ObjectOutputStream(baos);
+ oos.writeObject(bean);
+ oos.flush();
+ oos.close();
+ } catch (Exception e) {
+ fail("Exception during serialization: " + e);
+ }
+
+ // Deserialize the test bean
+ try {
+ bean = null;
+ ByteArrayInputStream bais =
+ new ByteArrayInputStream(baos.toByteArray());
+ ObjectInputStream ois = new ObjectInputStream(bais);
+ bean = (DynaBean) ois.readObject();
+ bais.close();
+ } catch (Exception e) {
+ fail("Exception during deserialization: " + e);
+ }
+
+ // Confirm property values
+ testGetDescriptorArguments();
+ testGetDescriptorBoolean();
+ testGetDescriptorDouble();
+ testGetDescriptorFloat();
+ testGetDescriptorInt();
+ testGetDescriptorLong();
+ testGetDescriptorSecond();
+ testGetDescriptorShort();
+ testGetDescriptorString();
+ testGetDescriptors();
+ testGetIndexedArguments();
+ testGetIndexedValues();
+ testGetMappedArguments();
+ testGetMappedValues();
+ testGetSimpleArguments();
+ testGetSimpleBoolean();
+ testGetSimpleDouble();
+ testGetSimpleFloat();
+ testGetSimpleInt();
+ testGetSimpleLong();
+ testGetSimpleShort();
+ testGetSimpleString();
+ testMappedContains();
+ testMappedRemove();
+
+ // Ensure that we can create a new instance of the same DynaClass
+ try {
+ bean = bean.getDynaClass().newInstance();
+ } catch (Exception e) {
+ fail("Exception creating new instance: " + e);
+ }
+ testGetDescriptorArguments();
+ testGetDescriptorBoolean();
+ testGetDescriptorDouble();
+ testGetDescriptorFloat();
+ testGetDescriptorInt();
+ testGetDescriptorLong();
+ testGetDescriptorSecond();
+ testGetDescriptorShort();
+ testGetDescriptorString();
+ testGetDescriptors();
+
+ }
+
+
+ /**
+ * Corner cases on setIndexedProperty invalid arguments.
+ */
+ public void testSetIndexedArguments() {
+
+ try {
+ bean.set("intArray", -1, new Integer(0));
+ fail("Should throw IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+ }
+
+
+ /**
+ * Positive and negative tests on setIndexedProperty valid arguments.
+ */
+ public void testSetIndexedValues() {
+
+ Object value = null;
+
+ try {
+ bean.set("intArray", 0, new Integer(1));
+ value = (Integer) bean.get("intArray", 0);
+ assertNotNull("Returned new value 0", value);
+ assertTrue("Returned Integer new value 0",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 0", 1,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ bean.set("intIndexed", 1, new Integer(11));
+ value = (Integer) bean.get("intIndexed", 1);
+ assertNotNull("Returned new value 1", value);
+ assertTrue("Returned Integer new value 1",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 1", 11,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ bean.set("listIndexed", 2, "New Value 2");
+ value = (String) bean.get("listIndexed", 2);
+ assertNotNull("Returned new value 2", value);
+ assertTrue("Returned String new value 2",
+ value instanceof String);
+ assertEquals("Returned correct new value 2", "New Value 2",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ bean.set("stringArray", 3, "New Value 3");
+ value = (String) bean.get("stringArray", 3);
+ assertNotNull("Returned new value 3", value);
+ assertTrue("Returned String new value 3",
+ value instanceof String);
+ assertEquals("Returned correct new value 3", "New Value 3",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ bean.set("stringIndexed", 4, "New Value 4");
+ value = (String) bean.get("stringIndexed", 4);
+ assertNotNull("Returned new value 4", value);
+ assertTrue("Returned String new value 4",
+ value instanceof String);
+ assertEquals("Returned correct new value 4", "New Value 4",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+
+ }
+
+
+ /**
+ * Positive and negative tests on setMappedProperty valid arguments.
+ */
+ public void testSetMappedValues() {
+
+ try {
+ bean.set("mappedProperty", "First Key", "New First Value");
+ assertEquals("Can replace old value",
+ "New First Value",
+ (String) bean.get("mappedProperty", "First Key"));
+ } catch (Throwable t) {
+ fail("Finding fourth value threw " + t);
+ }
+
+ try {
+ bean.set("mappedProperty", "Fourth Key", "Fourth Value");
+ assertEquals("Can set new value",
+ "Fourth Value",
+ (String) bean.get("mappedProperty", "Fourth Key"));
+ } catch (Throwable t) {
+ fail("Finding fourth value threw " + t);
+ }
+
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a boolean property.
+ */
+ public void testSetSimpleBoolean() {
+
+ try {
+ boolean oldValue =
+ ((Boolean) bean.get("booleanProperty")).booleanValue();
+ boolean newValue = !oldValue;
+ bean.set("booleanProperty", new Boolean(newValue));
+ assertTrue("Matched new value",
+ newValue ==
+ ((Boolean) bean.get("booleanProperty")).booleanValue());
+ } catch (Throwable e) {
+ fail("Exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a double property.
+ */
+ public void testSetSimpleDouble() {
+
+ try {
+ double oldValue =
+ ((Double) bean.get("doubleProperty")).doubleValue();
+ double newValue = oldValue + 1.0;
+ bean.set("doubleProperty", new Double(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Double) bean.get("doubleProperty")).doubleValue(),
+ (double) 0.005);
+ } catch (Throwable e) {
+ fail("Exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a float property.
+ */
+ public void testSetSimpleFloat() {
+
+ try {
+ float oldValue =
+ ((Float) bean.get("floatProperty")).floatValue();
+ float newValue = oldValue + (float) 1.0;
+ bean.set("floatProperty", new Float(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Float) bean.get("floatProperty")).floatValue(),
+ (float) 0.005);
+ } catch (Throwable e) {
+ fail("Exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a int property.
+ */
+ public void testSetSimpleInt() {
+
+ try {
+ int oldValue =
+ ((Integer) bean.get("intProperty")).intValue();
+ int newValue = oldValue + 1;
+ bean.set("intProperty", new Integer(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Integer) bean.get("intProperty")).intValue());
+ } catch (Throwable e) {
+ fail("Exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a long property.
+ */
+ public void testSetSimpleLong() {
+
+ try {
+ long oldValue =
+ ((Long) bean.get("longProperty")).longValue();
+ long newValue = oldValue + 1;
+ bean.set("longProperty", new Long(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Long) bean.get("longProperty")).longValue());
+ } catch (Throwable e) {
+ fail("Exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a short property.
+ */
+ public void testSetSimpleShort() {
+
+ try {
+ short oldValue =
+ ((Short) bean.get("shortProperty")).shortValue();
+ short newValue = (short) (oldValue + 1);
+ bean.set("shortProperty", new Short(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Short) bean.get("shortProperty")).shortValue());
+ } catch (Throwable e) {
+ fail("Exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a String property.
+ */
+ public void testSetSimpleString() {
+
+ try {
+ String oldValue = (String) bean.get("stringProperty");
+ String newValue = oldValue + " Extra Value";
+ bean.set("stringProperty", newValue);
+ assertEquals("Matched new value",
+ newValue,
+ (String) bean.get("stringProperty"));
+ } catch (Throwable e) {
+ fail("Exception: " + e);
+ }
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * Create and return a <code>DynaClass</code> instance for our test
+ * <code>DynaBean</code>.
+ */
+ protected DynaClass createDynaClass() {
+
+ int intArray[] = new int[0];
+ String stringArray[] = new String[0];
+
+ DynaClass dynaClass = new BasicDynaClass
+ ("TestDynaClass", null,
+ new DynaProperty[]{
+ new DynaProperty("booleanProperty", Boolean.TYPE),
+ new DynaProperty("booleanSecond", Boolean.TYPE),
+ new DynaProperty("doubleProperty", Double.TYPE),
+ new DynaProperty("floatProperty", Float.TYPE),
+ new DynaProperty("intArray", intArray.getClass()),
+ new DynaProperty("intIndexed", intArray.getClass()),
+ new DynaProperty("intProperty", Integer.TYPE),
+ new DynaProperty("listIndexed", List.class),
+ new DynaProperty("longProperty", Long.TYPE),
+ new DynaProperty("mappedProperty", Map.class),
+ new DynaProperty("mappedIntProperty", Map.class),
+ new DynaProperty("nullProperty", String.class),
+ new DynaProperty("shortProperty", Short.TYPE),
+ new DynaProperty("stringArray", stringArray.getClass()),
+ new DynaProperty("stringIndexed", stringArray.getClass()),
+ new DynaProperty("stringProperty", String.class),
+ });
+ return (dynaClass);
+
+ }
+
+
+ /**
+ * Base for testGetDescriptorXxxxx() series of tests.
+ *
+ * @param name Name of the property to be retrieved
+ * @param type Expected class type of this property
+ */
+ protected void testGetDescriptorBase(String name, Class type) {
+
+ try {
+ DynaProperty descriptor =
+ bean.getDynaClass().getDynaProperty(name);
+ assertNotNull("Got descriptor", descriptor);
+ assertEquals("Got correct type", type, descriptor.getType());
+ } catch (Throwable t) {
+ fail("Threw an exception: " + t);
+ }
+
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/BeanUtilsBenchCase.java b/trunk/src/test/org/apache/commons/beanutils/BeanUtilsBenchCase.java
new file mode 100644
index 0000000..25397c6
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/BeanUtilsBenchCase.java
@@ -0,0 +1,373 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.Map;
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * JUnit Test Case containing microbenchmarks for BeanUtils.
+ */
+
+public class BeanUtilsBenchCase extends TestCase {
+
+
+ // ------------------------------------------------------------ Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public BeanUtilsBenchCase(String name) {
+
+ super(name);
+
+ }
+
+
+ // ------------------------------------------------------ Instance Variables
+
+
+ // Basic loop counter
+ private long counter = 100000;
+
+ // DynaClass for inDyna and outDyna
+ private DynaClass dynaClass = null;
+
+ // Input objects that have identical sets of properties and values.
+ private BenchBean inBean = null;
+ private DynaBean inDyna = null;
+ private Map inMap = null; // Map of Objects requiring no conversion
+ private Map inStrs = null; // Map of Strings requiring conversion
+
+ // Output objects that have identical sets of properties.
+ private BenchBean outBean = null;
+ private DynaBean outDyna = null;
+
+ // BeanUtilsBean instance to be used
+ private BeanUtilsBean bu = null;
+
+
+ // ---------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+
+ // Set up loop counter (if property specified)
+ String prop = System.getProperty("counter");
+ if (prop != null) {
+ counter = Long.parseLong(prop);
+ }
+
+ // Set up DynaClass for our DynaBean instances
+ dynaClass = new BasicDynaClass
+ ("BenchDynaClass", null,
+ new DynaProperty[]{
+ new DynaProperty("booleanProperty", Boolean.TYPE),
+ new DynaProperty("byteProperty", Byte.TYPE),
+ new DynaProperty("doubleProperty", Double.TYPE),
+ new DynaProperty("floatProperty", Float.TYPE),
+ new DynaProperty("intProperty", Integer.TYPE),
+ new DynaProperty("longProperty", Long.TYPE),
+ new DynaProperty("shortProperty", Short.TYPE),
+ new DynaProperty("stringProperty", String.class),
+ });
+
+ // Create input instances
+ inBean = new BenchBean();
+ inMap = new HashMap();
+ inMap.put("booleanProperty", new Boolean(inBean.getBooleanProperty()));
+ inMap.put("byteProperty", new Byte(inBean.getByteProperty()));
+ inMap.put("doubleProperty", new Double(inBean.getDoubleProperty()));
+ inMap.put("floatProperty", new Float(inBean.getFloatProperty()));
+ inMap.put("intProperty", new Integer(inBean.getIntProperty()));
+ inMap.put("longProperty", new Long(inBean.getLongProperty()));
+ inMap.put("shortProperty", new Short(inBean.getShortProperty()));
+ inMap.put("stringProperty", inBean.getStringProperty());
+ inDyna = dynaClass.newInstance();
+ Iterator inKeys = inMap.keySet().iterator();
+ while (inKeys.hasNext()) {
+ String inKey = (String) inKeys.next();
+ inDyna.set(inKey, inMap.get(inKey));
+ }
+ inStrs = new HashMap();
+ inKeys = inMap.keySet().iterator();
+ while (inKeys.hasNext()) {
+ String inKey = (String) inKeys.next();
+ inStrs.put(inKey, inMap.get(inKey).toString());
+ }
+
+ // Create output instances
+ outBean = new BenchBean();
+ outDyna = dynaClass.newInstance();
+ Iterator outKeys = inMap.keySet().iterator();
+ while (outKeys.hasNext()) {
+ String outKey = (String) outKeys.next();
+ outDyna.set(outKey, inMap.get(outKey));
+ }
+
+ // Set up BeanUtilsBean instance we will use
+ bu = BeanUtilsBean.getInstance();
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+
+ return (new TestSuite(BeanUtilsBenchCase.class));
+
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+
+ dynaClass = null;
+ inBean = null;
+ inDyna = null;
+ inMap = null;
+ outBean = null;
+ outDyna = null;
+ bu = null;
+
+ }
+
+
+
+ // ------------------------------------------------- Individual Test Methods
+
+
+ // Time copyProperties() from a bean
+ public void testCopyPropertiesBean() throws Exception {
+
+ long start;
+ long stop;
+
+ // Bean->Bean
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outBean, inBean);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outBean, inBean);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.copyProperties(bean,bean), count=" + counter +
+ ", time=" + (stop - start));
+
+ // Bean->Dyna
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outDyna, inBean);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outDyna, inBean);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.copyProperties(dyna,bean), count=" + counter +
+ ", time=" + (stop - start));
+
+ }
+
+
+ // Time copyProperties() from a DynaBean
+ public void testCopyPropertiesDyna() throws Exception {
+
+ long start;
+ long stop;
+
+ // Dyna->Bean
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outBean, inDyna);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outBean, inDyna);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.copyProperties(bean,dyna), count=" + counter +
+ ", time=" + (stop - start));
+
+ // Dyna->Dyna
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outDyna, inDyna);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outDyna, inDyna);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.copyProperties(dyna,dyna), count=" + counter +
+ ", time=" + (stop - start));
+
+ }
+
+
+ // Time copyProperties() from a Map of Objects
+ public void testCopyPropertiesMap() throws Exception {
+
+ long start;
+ long stop;
+
+ // Map->Bean
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outBean, inMap);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outBean, inMap);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.copyProperties(bean, map), count=" + counter +
+ ", time=" + (stop - start));
+
+ // Map->Dyna
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outDyna, inMap);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outDyna, inMap);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.copyProperties(dyna, map), count=" + counter +
+ ", time=" + (stop - start));
+
+ }
+
+
+ // Time copyProperties() from a Map of Strings
+ public void testCopyPropertiesStrs() throws Exception {
+
+ long start;
+ long stop;
+
+ // Strs->Bean
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outBean, inStrs);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outBean, inStrs);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.copyProperties(bean,strs), count=" + counter +
+ ", time=" + (stop - start));
+
+ // Strs->Dyna
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outDyna, inStrs);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.copyProperties(outDyna, inStrs);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.copyProperties(dyna,strs), count=" + counter +
+ ", time=" + (stop - start));
+
+ }
+
+
+ // Time populate() from a Map of Objects
+ public void testPopulateMap() throws Exception {
+
+ long start;
+ long stop;
+
+ // Map->Bean
+ for (long i = 0; i < counter; i++) {
+ bu.populate(outBean, inMap);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.populate(outBean, inMap);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.populate(bean, map), count=" + counter +
+ ", time=" + (stop - start));
+
+ // Map->Dyna
+ for (long i = 0; i < counter; i++) {
+ bu.populate(outDyna, inMap);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.populate(outDyna, inMap);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.populate(dyna, map), count=" + counter +
+ ", time=" + (stop - start));
+
+ }
+
+
+ // Time populate() from a Map of Strings
+ // NOTE - This simulates what Struts does when processing form beans
+ public void testPopulateStrs() throws Exception {
+
+ long start;
+ long stop;
+
+ // Strs->Bean
+ for (long i = 0; i < counter; i++) {
+ bu.populate(outBean, inStrs);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.populate(outBean, inStrs);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.populate(bean,strs), count=" + counter +
+ ", time=" + (stop - start));
+
+ // Strs->Dyna
+ for (long i = 0; i < counter; i++) {
+ bu.populate(outDyna, inStrs);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ bu.populate(outDyna, inStrs);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("BU.populate(dyna,strs), count=" + counter +
+ ", time=" + (stop - start));
+
+ }
+
+
+ // --------------------------------------------------------- Support Methods
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/BeanUtilsTestCase.java b/trunk/src/test/org/apache/commons/beanutils/BeanUtilsTestCase.java
new file mode 100644
index 0000000..0ccbd8a
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/BeanUtilsTestCase.java
@@ -0,0 +1,1341 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.Map;
+
+import junit.framework.Test;
+import junit.framework.TestCase;
+import junit.framework.TestSuite;
+
+
+/**
+ * <p>
+ * Test Case for the BeanUtils class. The majority of these tests use
+ * instances of the TestBean class, so be sure to update the tests if you
+ * change the characteristics of that class.
+ * </p>
+ *
+ * <p>
+ * Template for this stolen from Craigs PropertyUtilsTestCase
+ * </p>
+ *
+ * <p>
+ * Note that the tests are dependant upon the static aspects
+ * (such as array sizes...) of the TestBean.java class, so ensure
+ * than all changes to TestBean are reflected here.
+ * </p>
+ *
+ * <p>
+ * So far, this test case has tests for the following methods of the
+ * <code>BeanUtils</code> class:
+ * </p>
+ * <ul>
+ * <li>getArrayProperty(Object bean, String name)</li>
+ * </ul>
+ *
+ * @author <a href="mailto:geirm at optonline.net">Geir Magnusson Jr.</a>
+ * @version $Revision: 1.29 $
+ */
+
+public class BeanUtilsTestCase extends TestCase {
+
+ // ---------------------------------------------------- Instance Variables
+
+ /**
+ * The test bean for each test.
+ */
+ protected TestBean bean = null;
+
+
+ /**
+ * The set of properties that should be described.
+ */
+ protected String describes[] =
+ { "booleanProperty",
+ "booleanSecond",
+ "byteProperty",
+ "doubleProperty",
+ "dupProperty",
+ "floatProperty",
+ "intArray",
+ // "intIndexed",
+ "longProperty",
+ "listIndexed",
+ "longProperty",
+ // "mappedProperty",
+ // "mappedIntProperty",
+ "nested",
+ "nullProperty",
+ "readOnlyProperty",
+ "shortProperty",
+ "stringArray",
+ // "stringIndexed",
+ "stringProperty"
+ };
+
+
+ // ---------------------------------------------------------- Constructors
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public BeanUtilsTestCase(String name) {
+ super(name);
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() {
+ bean = new TestBean();
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(BeanUtilsTestCase.class));
+ }
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ bean = null;
+ }
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ /**
+ * Test the copyProperties() method from a DynaBean.
+ */
+ public void testCopyPropertiesDynaBean() {
+
+ // Set up an origin bean with customized properties
+ DynaClass dynaClass = DynaBeanUtilsTestCase.createDynaClass();
+ DynaBean orig = null;
+ try {
+ orig = dynaClass.newInstance();
+ } catch (Exception e) {
+ fail("newInstance(): " + e);
+ }
+ orig.set("booleanProperty", Boolean.FALSE);
+ orig.set("byteProperty", new Byte((byte) 111));
+ orig.set("doubleProperty", new Double(333.33));
+ orig.set("dupProperty",
+ new String[] { "New 0", "New 1", "New 2" });
+ orig.set("intArray", new int[] { 100, 200, 300 });
+ orig.set("intProperty", new Integer(333));
+ orig.set("longProperty", new Long(3333));
+ orig.set("shortProperty", new Short((short) 33));
+ orig.set("stringArray", new String[] { "New 0", "New 1" });
+ orig.set("stringProperty", "Custom string");
+
+ // Copy the origin bean to our destination test bean
+ try {
+ BeanUtils.copyProperties(bean, orig);
+ } catch (Exception e) {
+ fail("Threw exception: " + e);
+ }
+
+ // Validate the results for scalar properties
+ assertEquals("Copied boolean property",
+ false,
+ bean.getBooleanProperty());
+ assertEquals("Copied byte property",
+ (byte) 111,
+ bean.getByteProperty());
+ assertEquals("Copied double property",
+ 333.33,
+ bean.getDoubleProperty(),
+ 0.005);
+ assertEquals("Copied int property",
+ 333,
+ bean.getIntProperty());
+ assertEquals("Copied long property",
+ (long) 3333,
+ bean.getLongProperty());
+ assertEquals("Copied short property",
+ (short) 33,
+ bean.getShortProperty());
+ assertEquals("Copied string property",
+ "Custom string",
+ bean.getStringProperty());
+
+ // Validate the results for array properties
+ String dupProperty[] = bean.getDupProperty();
+ assertNotNull("dupProperty present", dupProperty);
+ assertEquals("dupProperty length", 3, dupProperty.length);
+ assertEquals("dupProperty[0]", "New 0", dupProperty[0]);
+ assertEquals("dupProperty[1]", "New 1", dupProperty[1]);
+ assertEquals("dupProperty[2]", "New 2", dupProperty[2]);
+ int intArray[] = bean.getIntArray();
+ assertNotNull("intArray present", intArray);
+ assertEquals("intArray length", 3, intArray.length);
+ assertEquals("intArray[0]", 100, intArray[0]);
+ assertEquals("intArray[1]", 200, intArray[1]);
+ assertEquals("intArray[2]", 300, intArray[2]);
+ String stringArray[] = bean.getStringArray();
+ assertNotNull("stringArray present", stringArray);
+ assertEquals("stringArray length", 2, stringArray.length);
+ assertEquals("stringArray[0]", "New 0", stringArray[0]);
+ assertEquals("stringArray[1]", "New 1", stringArray[1]);
+
+ }
+
+
+ /**
+ * Test copyProperties() when the origin is a a <code>Map</code>.
+ */
+ public void testCopyPropertiesMap() {
+
+ Map map = new HashMap();
+ map.put("booleanProperty", "false");
+ map.put("byteProperty", "111");
+ map.put("doubleProperty", "333.0");
+ map.put("dupProperty", new String[] { "New 0", "New 1", "New 2" });
+ map.put("floatProperty", "222.0");
+ map.put("intArray", new String[] { "0", "100", "200" });
+ map.put("intProperty", "111");
+ map.put("longProperty", "444");
+ map.put("shortProperty", "555");
+ map.put("stringProperty", "New String Property");
+
+ try {
+ BeanUtils.copyProperties(bean, map);
+ } catch (Throwable t) {
+ fail("Threw " + t.toString());
+ }
+
+ // Scalar properties
+ assertEquals("booleanProperty", false,
+ bean.getBooleanProperty());
+ assertEquals("byteProperty", (byte) 111,
+ bean.getByteProperty());
+ assertEquals("doubleProperty", 333.0,
+ bean.getDoubleProperty(), 0.005);
+ assertEquals("floatProperty", (float) 222.0,
+ bean.getFloatProperty(), (float) 0.005);
+ assertEquals("longProperty", 111,
+ bean.getIntProperty());
+ assertEquals("longProperty", (long) 444,
+ bean.getLongProperty());
+ assertEquals("shortProperty", (short) 555,
+ bean.getShortProperty());
+ assertEquals("stringProperty", "New String Property",
+ bean.getStringProperty());
+
+ // Indexed Properties
+ String dupProperty[] = bean.getDupProperty();
+ assertNotNull("dupProperty present", dupProperty);
+ assertEquals("dupProperty length", 3, dupProperty.length);
+ assertEquals("dupProperty[0]", "New 0", dupProperty[0]);
+ assertEquals("dupProperty[1]", "New 1", dupProperty[1]);
+ assertEquals("dupProperty[2]", "New 2", dupProperty[2]);
+ int intArray[] = bean.getIntArray();
+ assertNotNull("intArray present", intArray);
+ assertEquals("intArray length", 3, intArray.length);
+ assertEquals("intArray[0]", 0, intArray[0]);
+ assertEquals("intArray[1]", 100, intArray[1]);
+ assertEquals("intArray[2]", 200, intArray[2]);
+
+ }
+
+
+ /**
+ * Test the copyProperties() method from a standard JavaBean.
+ */
+ public void testCopyPropertiesStandard() {
+
+ // Set up an origin bean with customized properties
+ TestBean orig = new TestBean();
+ orig.setBooleanProperty(false);
+ orig.setByteProperty((byte) 111);
+ orig.setDoubleProperty(333.33);
+ orig.setDupProperty(new String[] { "New 0", "New 1", "New 2" });
+ orig.setIntArray(new int[] { 100, 200, 300 });
+ orig.setIntProperty(333);
+ orig.setLongProperty(3333);
+ orig.setShortProperty((short) 33);
+ orig.setStringArray(new String[] { "New 0", "New 1" });
+ orig.setStringProperty("Custom string");
+
+ // Copy the origin bean to our destination test bean
+ try {
+ BeanUtils.copyProperties(bean, orig);
+ } catch (Exception e) {
+ fail("Threw exception: " + e);
+ }
+
+ // Validate the results for scalar properties
+ assertEquals("Copied boolean property",
+ false,
+ bean.getBooleanProperty());
+ assertEquals("Copied byte property",
+ (byte) 111,
+ bean.getByteProperty());
+ assertEquals("Copied double property",
+ 333.33,
+ bean.getDoubleProperty(),
+ 0.005);
+ assertEquals("Copied int property",
+ 333,
+ bean.getIntProperty());
+ assertEquals("Copied long property",
+ (long) 3333,
+ bean.getLongProperty());
+ assertEquals("Copied short property",
+ (short) 33,
+ bean.getShortProperty());
+ assertEquals("Copied string property",
+ "Custom string",
+ bean.getStringProperty());
+
+ // Validate the results for array properties
+ String dupProperty[] = bean.getDupProperty();
+ assertNotNull("dupProperty present", dupProperty);
+ assertEquals("dupProperty length", 3, dupProperty.length);
+ assertEquals("dupProperty[0]", "New 0", dupProperty[0]);
+ assertEquals("dupProperty[1]", "New 1", dupProperty[1]);
+ assertEquals("dupProperty[2]", "New 2", dupProperty[2]);
+ int intArray[] = bean.getIntArray();
+ assertNotNull("intArray present", intArray);
+ assertEquals("intArray length", 3, intArray.length);
+ assertEquals("intArray[0]", 100, intArray[0]);
+ assertEquals("intArray[1]", 200, intArray[1]);
+ assertEquals("intArray[2]", 300, intArray[2]);
+ String stringArray[] = bean.getStringArray();
+ assertNotNull("stringArray present", stringArray);
+ assertEquals("stringArray length", 2, stringArray.length);
+ assertEquals("stringArray[0]", "New 0", stringArray[0]);
+ assertEquals("stringArray[1]", "New 1", stringArray[1]);
+
+ }
+
+
+ /**
+ * Test the describe() method.
+ */
+ public void testDescribe() {
+
+ Map map = null;
+ try {
+ map = BeanUtils.describe(bean);
+ } catch (Exception e) {
+ fail("Threw exception " + e);
+ }
+
+ // Verify existence of all the properties that should be present
+ for (int i = 0; i < describes.length; i++) {
+ assertTrue("Property '" + describes[i] + "' is present",
+ map.containsKey(describes[i]));
+ }
+ assertTrue("Property 'writeOnlyProperty' is not present",
+ !map.containsKey("writeOnlyProperty"));
+
+ // Verify the values of scalar properties
+ assertEquals("Value of 'booleanProperty'",
+ "true",
+ (String) map.get("booleanProperty"));
+ assertEquals("Value of 'byteProperty'",
+ "121",
+ (String) map.get("byteProperty"));
+ assertEquals("Value of 'doubleProperty'",
+ "321.0",
+ (String) map.get("doubleProperty"));
+ assertEquals("Value of 'floatProperty'",
+ "123.0",
+ (String) map.get("floatProperty"));
+ assertEquals("Value of 'intProperty'",
+ "123",
+ (String) map.get("intProperty"));
+ assertEquals("Value of 'longProperty'",
+ "321",
+ (String) map.get("longProperty"));
+ assertEquals("Value of 'shortProperty'",
+ "987",
+ (String) map.get("shortProperty"));
+ assertEquals("Value of 'stringProperty'",
+ "This is a string",
+ (String) map.get("stringProperty"));
+
+ }
+
+
+ /**
+ * tests the string and int arrays of TestBean
+ */
+ public void testGetArrayProperty() {
+ try {
+ String arr[] = BeanUtils.getArrayProperty(bean, "stringArray");
+ String comp[] = bean.getStringArray();
+
+ assertTrue("String array length = " + comp.length,
+ (comp.length == arr.length));
+
+ arr = BeanUtils.getArrayProperty(bean, "intArray");
+ int iarr[] = bean.getIntArray();
+
+ assertTrue("String array length = " + iarr.length,
+ (iarr.length == arr.length));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * tests getting an indexed property
+ */
+ public void testGetIndexedProperty1() {
+ try {
+ String val = BeanUtils.getIndexedProperty(bean, "intIndexed[3]");
+ String comp = String.valueOf(bean.getIntIndexed(3));
+ assertTrue("intIndexed[3] == " + comp, val.equals(comp));
+
+ val = BeanUtils.getIndexedProperty(bean, "stringIndexed[3]");
+ comp = bean.getStringIndexed(3);
+ assertTrue("stringIndexed[3] == " + comp, val.equals(comp));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * tests getting an indexed property
+ */
+ public void testGetIndexedProperty2() {
+ try {
+ String val = BeanUtils.getIndexedProperty(bean, "intIndexed", 3);
+ String comp = String.valueOf(bean.getIntIndexed(3));
+
+ assertTrue("intIndexed,3 == " + comp, val.equals(comp));
+
+ val = BeanUtils.getIndexedProperty(bean, "stringIndexed", 3);
+ comp = bean.getStringIndexed(3);
+
+ assertTrue("stringIndexed,3 == " + comp, val.equals(comp));
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * tests getting a nested property
+ */
+ public void testGetNestedProperty() {
+ try {
+ String val = BeanUtils.getNestedProperty(bean, "nested.stringProperty");
+ String comp = bean.getNested().getStringProperty();
+ assertTrue("nested.StringProperty == " + comp,
+ val.equals(comp));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * tests getting a 'whatever' property
+ */
+ public void testGetGeneralProperty() {
+ try {
+ String val = BeanUtils.getProperty(bean, "nested.intIndexed[2]");
+ String comp = String.valueOf(bean.getIntIndexed(2));
+
+ assertTrue("nested.intIndexed[2] == " + comp,
+ val.equals(comp));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * tests getting a 'whatever' property
+ */
+ public void testGetSimpleProperty() {
+ try {
+ String val = BeanUtils.getSimpleProperty(bean, "shortProperty");
+ String comp = String.valueOf(bean.getShortProperty());
+
+ assertTrue("shortProperty == " + comp,
+ val.equals(comp));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * Test populate() method on individual array elements.
+ */
+ public void testPopulateArrayElements() {
+
+ try {
+
+ HashMap map = new HashMap();
+ map.put("intIndexed[0]", "100");
+ map.put("intIndexed[2]", "120");
+ map.put("intIndexed[4]", "140");
+
+ BeanUtils.populate(bean, map);
+
+ assertEquals("intIndexed[0] is 100",
+ 100, bean.getIntIndexed(0));
+ assertEquals("intIndexed[1] is 10",
+ 10, bean.getIntIndexed(1));
+ assertEquals("intIndexed[2] is 120",
+ 120, bean.getIntIndexed(2));
+ assertEquals("intIndexed[3] is 30",
+ 30, bean.getIntIndexed(3));
+ assertEquals("intIndexed[4] is 140",
+ 140, bean.getIntIndexed(4));
+
+ map.clear();
+ map.put("stringIndexed[1]", "New String 1");
+ map.put("stringIndexed[3]", "New String 3");
+
+ BeanUtils.populate(bean, map);
+
+ assertEquals("stringIndexed[0] is \"String 0\"",
+ "String 0", bean.getStringIndexed(0));
+ assertEquals("stringIndexed[1] is \"New String 1\"",
+ "New String 1", bean.getStringIndexed(1));
+ assertEquals("stringIndexed[2] is \"String 2\"",
+ "String 2", bean.getStringIndexed(2));
+ assertEquals("stringIndexed[3] is \"New String 3\"",
+ "New String 3", bean.getStringIndexed(3));
+ assertEquals("stringIndexed[4] is \"String 4\"",
+ "String 4", bean.getStringIndexed(4));
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * Test populate() method on array properties as a whole.
+ */
+ public void testPopulateArrayProperties() {
+
+ try {
+
+ HashMap map = new HashMap();
+ int intArray[] = new int[] { 123, 456, 789 };
+ map.put("intArray", intArray);
+ String stringArray[] = new String[]
+ { "New String 0", "New String 1" };
+ map.put("stringArray", stringArray);
+
+ BeanUtils.populate(bean, map);
+
+ intArray = bean.getIntArray();
+ assertNotNull("intArray is present", intArray);
+ assertEquals("intArray length",
+ 3, intArray.length);
+ assertEquals("intArray[0]", 123, intArray[0]);
+ assertEquals("intArray[1]", 456, intArray[1]);
+ assertEquals("intArray[2]", 789, intArray[2]);
+ stringArray = bean.getStringArray();
+ assertNotNull("stringArray is present", stringArray);
+ assertEquals("stringArray length", 2, stringArray.length);
+ assertEquals("stringArray[0]", "New String 0", stringArray[0]);
+ assertEquals("stringArray[1]", "New String 1", stringArray[1]);
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * Test populate() on mapped properties.
+ */
+ public void testPopulateMapped() {
+
+ try {
+
+ HashMap map = new HashMap();
+ map.put("mappedProperty(First Key)", "New First Value");
+ map.put("mappedProperty(Third Key)", "New Third Value");
+
+ BeanUtils.populate(bean, map);
+
+ assertEquals("mappedProperty(First Key)",
+ "New First Value",
+ bean.getMappedProperty("First Key"));
+ assertEquals("mappedProperty(Second Key)",
+ "Second Value",
+ bean.getMappedProperty("Second Key"));
+ assertEquals("mappedProperty(Third Key)",
+ "New Third Value",
+ bean.getMappedProperty("Third Key"));
+ assertNull("mappedProperty(Fourth Key",
+ bean.getMappedProperty("Fourth Key"));
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * Test populate() method on nested properties.
+ */
+ public void testPopulateNested() {
+
+ try {
+
+ HashMap map = new HashMap();
+ map.put("nested.booleanProperty", "false");
+ // booleanSecond is left at true
+ map.put("nested.doubleProperty", "432.0");
+ // floatProperty is left at 123.0
+ map.put("nested.intProperty", "543");
+ // longProperty is left at 321
+ map.put("nested.shortProperty", "654");
+ // stringProperty is left at "This is a string"
+ map.put("nested.writeOnlyProperty", "New writeOnlyProperty value");
+
+ BeanUtils.populate(bean, map);
+
+ assertTrue("booleanProperty is false",
+ !bean.getNested().getBooleanProperty());
+ assertTrue("booleanSecond is true",
+ bean.getNested().isBooleanSecond());
+ assertEquals("doubleProperty is 432.0",
+ (double) 432.0,
+ bean.getNested().getDoubleProperty(),
+ (double) 0.005);
+ assertEquals("floatProperty is 123.0",
+ (float) 123.0,
+ bean.getNested().getFloatProperty(),
+ (float) 0.005);
+ assertEquals("intProperty is 543",
+ 543, bean.getNested().getIntProperty());
+ assertEquals("longProperty is 321",
+ (long) 321, bean.getNested().getLongProperty());
+ assertEquals("shortProperty is 654",
+ (short) 654, bean.getNested().getShortProperty());
+ assertEquals("stringProperty is \"This is a string\"",
+ "This is a string",
+ bean.getNested().getStringProperty());
+ assertEquals("writeOnlyProperty is \"New writeOnlyProperty value\"",
+ "New writeOnlyProperty value",
+ bean.getNested().getWriteOnlyPropertyValue());
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * Test populate() method on scalar properties.
+ */
+ public void testPopulateScalar() {
+
+ try {
+
+ bean.setNullProperty("Non-null value");
+
+ HashMap map = new HashMap();
+ map.put("booleanProperty", "false");
+ // booleanSecond is left at true
+ map.put("byteProperty", "111");
+ map.put("doubleProperty", "432.0");
+ // floatProperty is left at 123.0
+ map.put("intProperty", "543");
+ map.put("longProperty", "");
+ map.put("nullProperty", null);
+ map.put("shortProperty", "654");
+ // stringProperty is left at "This is a string"
+ map.put("writeOnlyProperty", "New writeOnlyProperty value");
+ map.put("readOnlyProperty", "New readOnlyProperty value");
+
+ BeanUtils.populate(bean, map);
+
+ assertTrue("booleanProperty is false", !bean.getBooleanProperty());
+ assertTrue("booleanSecond is true", bean.isBooleanSecond());
+ assertEquals("byteProperty is 111",
+ (byte) 111, bean.getByteProperty());
+ assertEquals("doubleProperty is 432.0",
+ (double) 432.0, bean.getDoubleProperty(),
+ (double) 0.005);
+ assertEquals("floatProperty is 123.0",
+ (float) 123.0, bean.getFloatProperty(),
+ (float) 0.005);
+ assertEquals("intProperty is 543",
+ 543, bean.getIntProperty());
+ assertEquals("longProperty is 0",
+ (long) 0, bean.getLongProperty());
+ assertNull("nullProperty is null",
+ bean.getNullProperty());
+ assertEquals("shortProperty is 654",
+ (short) 654, bean.getShortProperty());
+ assertEquals("stringProperty is \"This is a string\"",
+ "This is a string", bean.getStringProperty());
+ assertEquals("writeOnlyProperty is \"New writeOnlyProperty value\"",
+ "New writeOnlyProperty value",
+ bean.getWriteOnlyPropertyValue());
+ assertEquals("readOnlyProperty is \"Read Only String Property\"",
+ "Read Only String Property",
+ bean.getReadOnlyProperty());
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * Test calling setProperty() with null property values.
+ */
+ public void testSetPropertyNullValues() throws Exception {
+
+ Object oldValue = null;
+ Object newValue = null;
+
+ // Scalar value into array
+ oldValue = PropertyUtils.getSimpleProperty(bean, "stringArray");
+ BeanUtils.setProperty(bean, "stringArray", (String) null);
+ newValue = PropertyUtils.getSimpleProperty(bean, "stringArray");
+ assertNotNull("stringArray is not null", newValue);
+ assertTrue("stringArray of correct type",
+ newValue instanceof String[]);
+ assertEquals("stringArray length",
+ 1, ((String[]) newValue).length);
+ assertTrue("stringArray[0] is null",
+ ((String[]) newValue)[0] == null);
+ PropertyUtils.setProperty(bean, "stringArray", oldValue);
+
+ // Indexed value into array
+ oldValue = PropertyUtils.getSimpleProperty(bean, "stringArray");
+ BeanUtils.setProperty(bean, "stringArray[2]", (String) null);
+ newValue = PropertyUtils.getSimpleProperty(bean, "stringArray");
+ assertNotNull("stringArray is not null", newValue);
+ assertTrue("stringArray of correct type",
+ newValue instanceof String[]);
+ assertEquals("stringArray length",
+ 5, ((String[]) newValue).length);
+ assertTrue("stringArray[2] is null",
+ ((String[]) newValue)[2] == null);
+ PropertyUtils.setProperty(bean, "stringArray", oldValue);
+
+ // Value into scalar
+ BeanUtils.setProperty(bean, "stringProperty", null);
+ assertTrue("stringProperty is now null",
+ BeanUtils.getProperty(bean, "stringProperty") == null);
+
+ }
+
+
+ /**
+ * Test converting to and from primitive wrapper types.
+ */
+ public void testSetPropertyOnPrimitiveWrappers() throws Exception {
+
+ BeanUtils.setProperty(bean,"intProperty", new Integer(1));
+ assertEquals(1,bean.getIntProperty());
+ BeanUtils.setProperty(bean,"stringProperty", new Integer(1));
+ assertEquals(1, Integer.parseInt(bean.getStringProperty()));
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on byte.
+ */
+ public void testSetPropertyByte() throws Exception {
+
+ BeanUtils.setProperty(bean, "byteProperty", new Byte((byte) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+/*
+ BeanUtils.setProperty(bean, "byteProperty", new Double((double) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+ BeanUtils.setProperty(bean, "byteProperty", new Float((float) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+*/
+ BeanUtils.setProperty(bean, "byteProperty", new Integer((int) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+ BeanUtils.setProperty(bean, "byteProperty", new Long((long) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+ BeanUtils.setProperty(bean, "byteProperty", new Short((short) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on double.
+ */
+ public void testSetPropertyDouble() throws Exception {
+
+ BeanUtils.setProperty(bean, "doubleProperty", new Byte((byte) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Double((double) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Float((float) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Integer((int) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Long((long) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Short((short) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on float.
+ */
+ public void testSetPropertyFloat() throws Exception {
+
+ BeanUtils.setProperty(bean, "floatProperty", new Byte((byte) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Double((double) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Float((float) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Integer((int) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Long((long) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Short((short) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on int.
+ */
+ public void testSetPropertyInteger() throws Exception {
+
+ BeanUtils.setProperty(bean, "longProperty", new Byte((byte) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+/*
+ BeanUtils.setProperty(bean, "longProperty", new Double((double) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+ BeanUtils.setProperty(bean, "longProperty", new Float((float) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+*/
+ BeanUtils.setProperty(bean, "longProperty", new Integer((int) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+ BeanUtils.setProperty(bean, "longProperty", new Long((long) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+ BeanUtils.setProperty(bean, "longProperty", new Short((short) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on long.
+ */
+ public void testSetPropertyLong() throws Exception {
+
+ BeanUtils.setProperty(bean, "longProperty", new Byte((byte) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+/*
+ BeanUtils.setProperty(bean, "longProperty", new Double((double) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+ BeanUtils.setProperty(bean, "longProperty", new Float((float) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+*/
+ BeanUtils.setProperty(bean, "longProperty", new Integer((int) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+ BeanUtils.setProperty(bean, "longProperty", new Long((long) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+ BeanUtils.setProperty(bean, "longProperty", new Short((short) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+
+ }
+
+
+ /**
+ * Test setting a null property value.
+ */
+ public void testSetPropertyNull() throws Exception {
+
+ bean.setNullProperty("non-null value");
+ BeanUtils.setProperty(bean, "nullProperty", null);
+ assertNull("nullProperty is null", bean.getNullProperty());
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on short.
+ */
+ public void testSetPropertyShort() throws Exception {
+
+ BeanUtils.setProperty(bean, "shortProperty", new Byte((byte) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+/*
+ BeanUtils.setProperty(bean, "shortProperty", new Double((double) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+ BeanUtils.setProperty(bean, "shortProperty", new Float((float) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+*/
+ BeanUtils.setProperty(bean, "shortProperty", new Integer((int) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+ BeanUtils.setProperty(bean, "shortProperty", new Long((long) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+ BeanUtils.setProperty(bean, "shortProperty", new Short((short) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on byte.
+ */
+ public void testCopyPropertyByte() throws Exception {
+
+ BeanUtils.copyProperty(bean, "byteProperty", new Byte((byte) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+ BeanUtils.copyProperty(bean, "byteProperty", new Double((double) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+ BeanUtils.copyProperty(bean, "byteProperty", new Float((float) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+ BeanUtils.copyProperty(bean, "byteProperty", new Integer((int) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+ BeanUtils.copyProperty(bean, "byteProperty", new Long((long) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+ BeanUtils.copyProperty(bean, "byteProperty", new Short((short) 123));
+ assertEquals((byte) 123, bean.getByteProperty());
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on double.
+ */
+ public void testCopyPropertyDouble() throws Exception {
+
+ BeanUtils.copyProperty(bean, "doubleProperty", new Byte((byte) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "doubleProperty", new Double((double) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "doubleProperty", new Float((float) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "doubleProperty", new Integer((int) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "doubleProperty", new Long((long) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "doubleProperty", new Short((short) 123));
+ assertEquals((double) 123, bean.getDoubleProperty(), 0.005);
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on float.
+ */
+ public void testCopyPropertyFloat() throws Exception {
+
+ BeanUtils.copyProperty(bean, "floatProperty", new Byte((byte) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "floatProperty", new Double((double) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "floatProperty", new Float((float) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "floatProperty", new Integer((int) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "floatProperty", new Long((long) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+ BeanUtils.copyProperty(bean, "floatProperty", new Short((short) 123));
+ assertEquals((float) 123, bean.getFloatProperty(), 0.005);
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on int.
+ */
+ public void testCopyPropertyInteger() throws Exception {
+
+ BeanUtils.copyProperty(bean, "longProperty", new Byte((byte) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Double((double) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Float((float) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Integer((int) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Long((long) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Short((short) 123));
+ assertEquals((int) 123, bean.getIntProperty());
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on long.
+ */
+ public void testCopyPropertyLong() throws Exception {
+
+ BeanUtils.copyProperty(bean, "longProperty", new Byte((byte) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Double((double) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Float((float) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Integer((int) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Long((long) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+ BeanUtils.copyProperty(bean, "longProperty", new Short((short) 123));
+ assertEquals((long) 123, bean.getLongProperty());
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on short.
+ */
+ public void testCopyPropertyShort() throws Exception {
+
+ BeanUtils.copyProperty(bean, "shortProperty", new Byte((byte) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+ BeanUtils.copyProperty(bean, "shortProperty", new Double((double) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+ BeanUtils.copyProperty(bean, "shortProperty", new Float((float) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+ BeanUtils.copyProperty(bean, "shortProperty", new Integer((int) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+ BeanUtils.copyProperty(bean, "shortProperty", new Long((long) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+ BeanUtils.copyProperty(bean, "shortProperty", new Short((short) 123));
+ assertEquals((short) 123, bean.getShortProperty());
+
+ }
+
+
+ /**
+ * Test copying a property using a nested indexed array expression,
+ * with and without conversions.
+ */
+ public void testCopyPropertyNestedIndexedArray() throws Exception {
+
+ int origArray[] = { 0, 10, 20, 30, 40 };
+ int intArray[] = { 0, 0, 0 };
+ bean.getNested().setIntArray(intArray);
+ int intChanged[] = { 0, 0, 0 };
+
+ // No conversion required
+ BeanUtils.copyProperty(bean, "nested.intArray[1]", new Integer(1));
+ checkIntArray(bean.getIntArray(), origArray);
+ intChanged[1] = 1;
+ checkIntArray(bean.getNested().getIntArray(), intChanged);
+
+ // Widening conversion required
+ BeanUtils.copyProperty(bean, "nested.intArray[1]", new Byte((byte) 2));
+ checkIntArray(bean.getIntArray(), origArray);
+ intChanged[1] = 2;
+ checkIntArray(bean.getNested().getIntArray(), intChanged);
+
+ // Narrowing conversion required
+ BeanUtils.copyProperty(bean, "nested.intArray[1]", new Long((long) 3));
+ checkIntArray(bean.getIntArray(), origArray);
+ intChanged[1] = 3;
+ checkIntArray(bean.getNested().getIntArray(), intChanged);
+
+ // String conversion required
+ BeanUtils.copyProperty(bean, "nested.intArray[1]", "4");
+ checkIntArray(bean.getIntArray(), origArray);
+ intChanged[1] = 4;
+ checkIntArray(bean.getNested().getIntArray(), intChanged);
+
+ }
+
+
+ /**
+ * Test copying a property using a nested mapped map property.
+ */
+ public void testCopyPropertyNestedMappedMap() throws Exception {
+
+ Map origMap = new HashMap();
+ origMap.put("First Key", "First Value");
+ origMap.put("Second Key", "Second Value");
+ Map changedMap = new HashMap();
+ changedMap.put("First Key", "First Value");
+ changedMap.put("Second Key", "Second Value");
+
+ // No conversion required
+ BeanUtils.copyProperty(bean, "nested.mapProperty(Second Key)",
+ "New Second Value");
+ checkMap(bean.getMapProperty(), origMap);
+ changedMap.put("Second Key", "New Second Value");
+ checkMap(bean.getNested().getMapProperty(), changedMap);
+
+ }
+
+
+ /**
+ * Test copying a property using a nested simple expression, with and
+ * without conversions.
+ */
+ public void testCopyPropertyNestedSimple() throws Exception {
+
+ bean.setIntProperty(0);
+ bean.getNested().setIntProperty(0);
+
+ // No conversion required
+ BeanUtils.copyProperty(bean, "nested.intProperty", new Integer(1));
+ assertNotNull(bean.getNested());
+ assertEquals(0, bean.getIntProperty());
+ assertEquals(1, bean.getNested().getIntProperty());
+
+ // Widening conversion required
+ BeanUtils.copyProperty(bean, "nested.intProperty", new Byte((byte) 2));
+ assertNotNull(bean.getNested());
+ assertEquals(0, bean.getIntProperty());
+ assertEquals(2, bean.getNested().getIntProperty());
+
+ // Narrowing conversion required
+ BeanUtils.copyProperty(bean, "nested.intProperty", new Long((long) 3));
+ assertNotNull(bean.getNested());
+ assertEquals(0, bean.getIntProperty());
+ assertEquals(3, bean.getNested().getIntProperty());
+
+ // String conversion required
+ BeanUtils.copyProperty(bean, "nested.intProperty", "4");
+ assertNotNull(bean.getNested());
+ assertEquals(0, bean.getIntProperty());
+ assertEquals(4, bean.getNested().getIntProperty());
+
+ }
+
+
+ /**
+ * Test copying a null property value.
+ */
+ public void testCopyPropertyNull() throws Exception {
+
+ bean.setNullProperty("non-null value");
+ BeanUtils.copyProperty(bean, "nullProperty", null);
+ assertNull("nullProperty is null", bean.getNullProperty());
+
+ }
+
+
+ /**
+ * Test copying a new value to a write-only property, with and without
+ * conversions.
+ */
+ public void testCopyPropertyWriteOnly() throws Exception {
+
+ bean.setWriteOnlyProperty("Original value");
+
+ // No conversion required
+ BeanUtils.copyProperty(bean, "writeOnlyProperty", "New value");
+ assertEquals("New value", bean.getWriteOnlyPropertyValue());
+
+ // Integer->String conversion required
+ BeanUtils.copyProperty(bean, "writeOnlyProperty", new Integer(123));
+ assertEquals("123", bean.getWriteOnlyPropertyValue());
+
+ }
+
+
+ /**
+ * Test setting a new value to a write-only property, with and without
+ * conversions.
+ */
+ public void testSetPropertyWriteOnly() throws Exception {
+
+ bean.setWriteOnlyProperty("Original value");
+
+ // No conversion required
+ BeanUtils.setProperty(bean, "writeOnlyProperty", "New value");
+ assertEquals("New value", bean.getWriteOnlyPropertyValue());
+
+ // Integer->String conversion required
+ BeanUtils.setProperty(bean, "writeOnlyProperty", new Integer(123));
+ assertEquals("123", bean.getWriteOnlyPropertyValue());
+
+ }
+
+ /** Tests that separate instances can register separate instances */
+ public void testSeparateInstances() throws Exception {
+ BeanUtilsBean utilsOne = new BeanUtilsBean(
+ new ConvertUtilsBean(),
+ new PropertyUtilsBean());
+ BeanUtilsBean utilsTwo = new BeanUtilsBean(
+ new ConvertUtilsBean(),
+ new PropertyUtilsBean());
+
+
+ TestBean bean = new TestBean();
+
+ // Make sure what we're testing works
+ bean.setBooleanProperty(false);
+ utilsOne.setProperty(bean, "booleanProperty", "true");
+ assertEquals("Set property failed (1)", bean.getBooleanProperty(), true);
+
+ bean.setBooleanProperty(false);
+ utilsTwo.setProperty(bean, "booleanProperty", "true");
+ assertEquals("Set property failed (2)", bean.getBooleanProperty(), true);
+
+ // now change the registered conversion
+
+ utilsOne.getConvertUtils().register(new ThrowExceptionConverter(), Boolean.TYPE);
+ try {
+
+ bean.setBooleanProperty(false);
+ utilsOne.setProperty(bean, "booleanProperty", "true");
+ fail("Registered conversion not used.");
+
+ } catch (PassTestException e) { /* Do nothing */ }
+
+ // make sure that this conversion has no been registered in the other instance
+ try {
+
+ bean.setBooleanProperty(false);
+ utilsTwo.setProperty(bean, "booleanProperty", "true");
+ assertEquals("Set property failed (3)", bean.getBooleanProperty(), true);
+
+ } catch (PassTestException e) {
+ fail("Registed converter is used by other instances");
+ }
+ }
+
+ public void testArrayPropertyConversion() throws Exception {
+ BeanUtilsBean beanUtils = new BeanUtilsBean(
+ new ConvertUtilsBean(),
+ new PropertyUtilsBean());
+ beanUtils.getConvertUtils().register(
+ new Converter () {
+ public Object convert(Class type, Object value) {
+ return "Spam, spam, spam, spam!";
+ }
+ },
+ String.class);
+
+ TestBean bean = new TestBean();
+ String [] results = beanUtils.getArrayProperty(bean, "intArray");
+
+ int[] values = bean.getIntArray();
+ assertEquals(
+ "Converted array size not equal to property array size.",
+ results.length,
+ values.length);
+ for (int i=0, size=values.length ; i<size; i++) {
+ assertEquals(
+ "Value " + i + " incorrectly converted ",
+ "Spam, spam, spam, spam!",
+ results[i]);
+ }
+ }
+
+ // Ensure that the actual int[] matches the expected int[]
+ protected void checkIntArray(int actual[], int expected[]) {
+ assertNotNull("actual array not null", actual);
+ assertEquals("actual array length", expected.length, actual.length);
+ for (int i = 0; i < actual.length; i++) {
+ assertEquals("actual array value[" + i + "]",
+ expected[i], actual[i]);
+ }
+ }
+
+
+ // Ensure that the actual Map matches the expected Map
+ protected void checkMap(Map actual, Map expected) {
+ assertNotNull("actual map not null", actual);
+ assertEquals("actual map size", expected.size(), actual.size());
+ Iterator keys = expected.keySet().iterator();
+ while (keys.hasNext()) {
+ Object key = keys.next();
+ assertEquals("actual map value(" + key + ")",
+ expected.get(key), actual.get(key));
+ }
+ }
+
+ public void testMappedProperty() throws Exception {
+ MappedPropertyTestBean bean = new MappedPropertyTestBean();
+
+ BeanUtils.setProperty(bean, "mapproperty(this.that.the-other)", "some.dotty.value");
+
+ assertEquals(
+ "Mapped property set correctly",
+ "some.dotty.value",
+ bean.getMapproperty("this.that.the-other"));
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/BeanWithInnerBean.java b/trunk/src/test/org/apache/commons/beanutils/BeanWithInnerBean.java
new file mode 100644
index 0000000..006eee9
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/BeanWithInnerBean.java
@@ -0,0 +1,45 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import java.util.Properties;
+
+/**
+ * Bean with inner bean.
+ *
+ * @author John E. Conlon
+ * @author Robert Burrell Donkin
+ */
+public class BeanWithInnerBean {
+ private InnerBean innerBean = new InnerBean();
+
+ public BeanWithInnerBean() {}
+ public InnerBean getInnerBean(){
+ return innerBean;
+ }
+
+ public class InnerBean {
+ private Properties fish = new Properties();
+
+ public String getFish(String key){
+ return fish.getProperty(key);
+ }
+ public void setFish(String key, String value){
+ fish.setProperty(key, value);
+ }
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/BeanificationTestCase.java b/trunk/src/test/org/apache/commons/beanutils/BeanificationTestCase.java
new file mode 100644
index 0000000..78bc83e
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/BeanificationTestCase.java
@@ -0,0 +1,541 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import java.util.*;
+
+import java.lang.ref.WeakReference;
+import java.lang.ref.ReferenceQueue;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+import org.apache.commons.collections.ReferenceMap;
+import org.apache.commons.logging.LogFactory;
+
+/**
+ * <p>
+ * Test Case for changes made during Beanutils Beanification
+ * </p>
+ *
+ * @author Robert Burrell Donkin
+ * @author Juozas Baliuka
+ * @version $Revision: 1.5 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class BeanificationTestCase extends TestCase {
+
+ // ---------------------------------------------------- Constants
+
+ /** Maximum number of iterations before our test fails */
+ public static final int MAX_GC_ITERATIONS = 50;
+
+ // ---------------------------------------------------- Instance Variables
+
+
+ // ---------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public BeanificationTestCase(String name) {
+ super(name);
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() {
+
+ ConvertUtils.deregister();
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(BeanificationTestCase.class));
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ ; // No action required
+ }
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+ /** Test of the methodology we'll use for some of the later tests */
+ public void testMemoryTestMethodology() throws Exception {
+ // test methodology
+ // many thanks to Juozas Baliuka for suggesting this method
+ ClassLoader loader = new ClassLoader() {};
+ WeakReference reference = new WeakReference(loader);
+ Class myClass = loader.loadClass("org.apache.commons.beanutils.BetaBean");
+
+ assertNotNull("Weak reference released early", reference.get());
+
+ // dereference class loader and class:
+ loader = null;
+ myClass = null;
+
+ int iterations = 0;
+ int bytz = 2;
+ while(true) {
+ System.gc();
+ if(iterations++ > MAX_GC_ITERATIONS){
+ fail("Max iterations reached before resource released.");
+ }
+ if( reference.get() == null ) {
+ break;
+
+ } else {
+ // create garbage:
+ byte[] b = new byte[bytz];
+ bytz = bytz * 2;
+ }
+ }
+ }
+
+ /** Tests whether classloaders and beans are released from memory by the map used by beanutils */
+ public void testMemoryLeak2() throws Exception {
+ // tests when the map used by beanutils has the right behaviour
+
+ if (isPre14JVM()) {
+ System.out.println("WARNING: CANNOT TEST MEMORY LEAK ON PRE1.4 JVM");
+ return;
+ }
+
+ // many thanks to Juozas Baliuka for suggesting this methodology
+ TestClassLoader loader = new TestClassLoader();
+ ReferenceQueue queue = new ReferenceQueue();
+ WeakReference loaderReference = new WeakReference(loader, queue);
+ Integer test = new Integer(1);
+
+ WeakReference testReference = new WeakReference(test, queue);
+ //Map map = new ReferenceMap(ReferenceMap.WEAK, ReferenceMap.HARD, true);
+ Map map = new WeakHashMap();
+ map.put(loader, test);
+
+ assertEquals("In map", test, map.get(loader));
+ assertNotNull("Weak reference released early (1)", loaderReference.get());
+ assertNotNull("Weak reference released early (2)", testReference.get());
+
+ // dereference strong references
+ loader = null;
+ test = null;
+
+ int iterations = 0;
+ int bytz = 2;
+ while(true) {
+ System.gc();
+ if(iterations++ > MAX_GC_ITERATIONS){
+ fail("Max iterations reached before resource released.");
+ }
+ map.isEmpty();
+
+ if(
+ loaderReference.get() == null &&
+ testReference.get() == null) {
+ break;
+
+ } else {
+ // create garbage:
+ byte[] b = new byte[bytz];
+ bytz = bytz * 2;
+ }
+ }
+ }
+
+ /** Tests whether classloaders and beans are released from memory */
+ public void testMemoryLeak() throws Exception {
+ if (isPre14JVM()) {
+ System.out.println("WARNING: CANNOT TEST MEMORY LEAK ON PRE1.4 JVM");
+ return;
+ }
+
+ // many thanks to Juozas Baliuka for suggesting this methodology
+ TestClassLoader loader = new TestClassLoader();
+ WeakReference loaderReference = new WeakReference(loader);
+ BeanUtilsBean.getInstance();
+
+ class GetBeanUtilsBeanThread extends Thread {
+
+ BeanUtilsBean beanUtils;
+ ConvertUtilsBean convertUtils;
+ PropertyUtilsBean propertyUtils;
+
+ GetBeanUtilsBeanThread() {}
+
+ public void run() {
+ beanUtils = BeanUtilsBean.getInstance();
+ convertUtils = ConvertUtilsBean.getInstance();
+ propertyUtils = PropertyUtilsBean.getInstance();
+ // XXX Log keeps a reference around!
+ LogFactory.releaseAll();
+ }
+
+ public String toString() {
+ return "GetBeanUtilsBeanThread";
+ }
+ }
+
+
+ GetBeanUtilsBeanThread thread = new GetBeanUtilsBeanThread();
+ WeakReference threadWeakReference = new WeakReference(thread);
+ thread.setContextClassLoader(loader);
+
+ thread.start();
+ thread.join();
+
+ WeakReference beanUtilsReference = new WeakReference(thread.beanUtils);
+ WeakReference propertyUtilsReference = new WeakReference(thread.propertyUtils);
+ WeakReference convertUtilsReference = new WeakReference(thread.convertUtils);
+
+ assertNotNull("Weak reference released early (1)", loaderReference.get());
+ assertNotNull("Weak reference released early (2)", beanUtilsReference.get());
+ assertNotNull("Weak reference released early (3)", propertyUtilsReference.get());
+ assertNotNull("Weak reference released early (4)", convertUtilsReference.get());
+
+ // dereference strong references
+ loader = null;
+ thread.setContextClassLoader(null);
+ thread = null;
+
+ int iterations = 0;
+ int bytz = 2;
+ while(true) {
+ BeanUtilsBean.getInstance();
+ System.gc();
+ if(iterations++ > MAX_GC_ITERATIONS){
+ fail("Max iterations reached before resource released.");
+ }
+
+ if(
+ loaderReference.get() == null &&
+ beanUtilsReference.get() == null &&
+ propertyUtilsReference.get() == null &&
+ convertUtilsReference.get() == null) {
+ break;
+
+ } else {
+ // create garbage:
+ byte[] b = new byte[bytz];
+ bytz = bytz * 2;
+ }
+ }
+ }
+
+ /**
+ * Tests whether difference instances are loaded by different
+ * context classloaders.
+ */
+ public void testGetByContextClassLoader() throws Exception {
+
+ class GetBeanUtilsBeanThread extends Thread {
+
+ private Signal signal;
+
+ GetBeanUtilsBeanThread(Signal signal) {
+ this.signal = signal;
+ }
+
+ public void run() {
+ signal.setSignal(2);
+ signal.setBean(BeanUtilsBean.getInstance());
+ signal.setConvertUtils(ConvertUtilsBean.getInstance());
+ signal.setPropertyUtils(PropertyUtilsBean.getInstance());
+ }
+
+ public String toString() {
+ return "GetBeanUtilsBeanThread";
+ }
+ }
+
+ Signal signal = new Signal();
+ signal.setSignal(1);
+
+ GetBeanUtilsBeanThread thread = new GetBeanUtilsBeanThread(signal);
+ thread.setContextClassLoader(new TestClassLoader());
+
+ thread.start();
+ thread.join();
+
+ assertEquals("Signal not set by test thread", 2, signal.getSignal());
+ assertTrue(
+ "Different BeanUtilsBean instances per context classloader",
+ BeanUtilsBean.getInstance() != signal.getBean());
+ assertTrue(
+ "Different ConvertUtilsBean instances per context classloader",
+ ConvertUtilsBean.getInstance() != signal.getConvertUtils());
+ assertTrue(
+ "Different PropertyUtilsBean instances per context classloader",
+ PropertyUtilsBean.getInstance() != signal.getPropertyUtils());
+ }
+
+
+ /**
+ * Tests whether difference instances are loaded by different
+ * context classloaders.
+ */
+ public void testContextClassLoaderLocal() throws Exception {
+
+ class CCLLTesterThread extends Thread {
+
+ private Signal signal;
+ private ContextClassLoaderLocal ccll;
+
+ CCLLTesterThread(Signal signal, ContextClassLoaderLocal ccll) {
+ this.signal = signal;
+ this.ccll = ccll;
+ }
+
+ public void run() {
+ ccll.set(new Integer(1789));
+ signal.setSignal(2);
+ signal.setMarkerObject(ccll.get());
+ }
+
+ public String toString() {
+ return "CCLLTesterThread";
+ }
+ }
+
+ ContextClassLoaderLocal ccll = new ContextClassLoaderLocal();
+ ccll.set(new Integer(1776));
+ assertEquals("Start thread sets value", new Integer(1776), ccll.get());
+
+ Signal signal = new Signal();
+ signal.setSignal(1);
+
+ CCLLTesterThread thread = new CCLLTesterThread(signal, ccll);
+ thread.setContextClassLoader(new TestClassLoader());
+
+ thread.start();
+ thread.join();
+
+ assertEquals("Signal not set by test thread", 2, signal.getSignal());
+ assertEquals("Second thread preserves value", new Integer(1776), ccll.get());
+ assertEquals("Second thread gets value it set", new Integer(1789), signal.getMarkerObject());
+ }
+
+ /** Tests whether calls are independent for different classloaders */
+ public void testContextClassloaderIndependence() throws Exception {
+
+ class TestIndependenceThread extends Thread {
+ private Signal signal;
+ private PrimitiveBean bean;
+
+ TestIndependenceThread(Signal signal, PrimitiveBean bean) {
+ this.signal = signal;
+ this.bean = bean;
+ }
+
+ public void run() {
+ try {
+ signal.setSignal(3);
+ ConvertUtils.register(new Converter() {
+ public Object convert(Class type, Object value) {
+ return new Integer(9);
+ }
+ }, Integer.TYPE);
+ BeanUtils.setProperty(bean, "int", new Integer(1));
+ } catch (Exception e) {
+ e.printStackTrace();
+ signal.setException(e);
+ }
+ }
+
+ public String toString() {
+ return "TestIndependenceThread";
+ }
+ }
+
+ PrimitiveBean bean = new PrimitiveBean();
+ BeanUtils.setProperty(bean, "int", new Integer(1));
+ assertEquals("Wrong property value (1)", 1, bean.getInt());
+
+ ConvertUtils.register(new Converter() {
+ public Object convert(Class type, Object value) {
+ return new Integer(5);
+ }
+ }, Integer.TYPE);
+ BeanUtils.setProperty(bean, "int", new Integer(1));
+ assertEquals("Wrong property value(2)", 5, bean.getInt());
+
+ Signal signal = new Signal();
+ signal.setSignal(1);
+ TestIndependenceThread thread = new TestIndependenceThread(signal, bean);
+ thread.setContextClassLoader(new TestClassLoader());
+
+ thread.start();
+ thread.join();
+
+ assertNull("Exception thrown by test thread:" + signal.getException(), signal.getException());
+ assertEquals("Signal not set by test thread", 3, signal.getSignal());
+ assertEquals("Wrong property value(3)", 9, bean.getInt());
+
+ }
+
+ /** Tests whether different threads can set beanutils instances correctly */
+ public void testBeanUtilsBeanSetInstance() throws Exception {
+
+ class SetInstanceTesterThread extends Thread {
+
+ private Signal signal;
+ private BeanUtilsBean bean;
+
+ SetInstanceTesterThread(Signal signal, BeanUtilsBean bean) {
+ this.signal = signal;
+ this.bean = bean;
+ }
+
+ public void run() {
+ BeanUtilsBean.setInstance(bean);
+ signal.setSignal(21);
+ signal.setBean(BeanUtilsBean.getInstance());
+ }
+
+ public String toString() {
+ return "SetInstanceTesterThread";
+ }
+ }
+
+ Signal signal = new Signal();
+ signal.setSignal(1);
+
+ BeanUtilsBean beanOne = new BeanUtilsBean();
+ BeanUtilsBean beanTwo = new BeanUtilsBean();
+
+ SetInstanceTesterThread thread = new SetInstanceTesterThread(signal, beanTwo);
+ thread.setContextClassLoader(new TestClassLoader());
+
+ BeanUtilsBean.setInstance(beanOne);
+ assertEquals("Start thread gets right instance", beanOne, BeanUtilsBean.getInstance());
+
+ thread.start();
+ thread.join();
+
+ assertEquals("Signal not set by test thread", 21, signal.getSignal());
+ assertEquals("Second thread preserves value", beanOne, BeanUtilsBean.getInstance());
+ assertEquals("Second thread gets value it set", beanTwo, signal.getBean());
+ }
+
+ /** Tests whether the unset method works*/
+ public void testContextClassLoaderUnset() throws Exception {
+ BeanUtilsBean beanOne = new BeanUtilsBean();
+ ContextClassLoaderLocal ccll = new ContextClassLoaderLocal();
+ ccll.set(beanOne);
+ assertEquals("Start thread gets right instance", beanOne, ccll.get());
+ ccll.unset();
+ assertTrue("Unset works", !beanOne.equals(ccll.get()));
+ }
+
+ private boolean isPre14JVM() {
+ // some pre 1.4 JVM have buggy WeakHashMap implementations
+ // this is used to test for those JVM
+ String version = System.getProperty("java.specification.version");
+ StringTokenizer tokenizer = new StringTokenizer(version,".");
+ if (tokenizer.nextToken().equals("1")) {
+ String minorVersion = tokenizer.nextToken();
+ if (minorVersion.equals("0")) return true;
+ if (minorVersion.equals("1")) return true;
+ if (minorVersion.equals("2")) return true;
+ if (minorVersion.equals("3")) return true;
+ }
+ return false;
+ }
+
+ // ---- Auxillary classes
+
+ class TestClassLoader extends ClassLoader {
+ public String toString() {
+ return "TestClassLoader";
+ }
+ }
+
+ class Signal {
+ private Exception e;
+ private int signal = 0;
+ private BeanUtilsBean bean;
+ private PropertyUtilsBean propertyUtils;
+ private ConvertUtilsBean convertUtils;
+ private Object marker;
+
+ public Exception getException() {
+ return e;
+ }
+
+ public void setException(Exception e) {
+ this.e = e;
+ }
+
+ public int getSignal() {
+ return signal;
+ }
+
+ public void setSignal(int signal) {
+ this.signal = signal;
+ }
+
+ public Object getMarkerObject() {
+ return marker;
+ }
+
+ public void setMarkerObject(Object marker) {
+ this.marker = marker;
+ }
+
+ public BeanUtilsBean getBean() {
+ return bean;
+ }
+
+ public void setBean(BeanUtilsBean bean) {
+ this.bean = bean;
+ }
+
+ public PropertyUtilsBean getPropertyUtils() {
+ return propertyUtils;
+ }
+
+ public void setPropertyUtils(PropertyUtilsBean propertyUtils) {
+ this.propertyUtils = propertyUtils;
+ }
+
+ public ConvertUtilsBean getConvertUtils() {
+ return convertUtils;
+ }
+
+ public void setConvertUtils(ConvertUtilsBean convertUtils) {
+ this.convertUtils = convertUtils;
+ }
+ }
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/BenchBean.java b/trunk/src/test/org/apache/commons/beanutils/BenchBean.java
new file mode 100644
index 0000000..d16aa12
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/BenchBean.java
@@ -0,0 +1,146 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+/**
+ * Plain old java bean (POJO) for microbenchmarks.
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class BenchBean {
+
+
+ // -------------------------------------------------------------- Properties
+
+
+ /**
+ * A boolean property.
+ */
+ private boolean booleanProperty = true;
+
+ public boolean getBooleanProperty() {
+ return (booleanProperty);
+ }
+
+ public void setBooleanProperty(boolean booleanProperty) {
+ this.booleanProperty = booleanProperty;
+ }
+
+
+ /**
+ * A byte property.
+ */
+ private byte byteProperty = (byte) 121;
+
+ public byte getByteProperty() {
+ return (this.byteProperty);
+ }
+
+ public void setByteProperty(byte byteProperty) {
+ this.byteProperty = byteProperty;
+ }
+
+
+ /**
+ * A double property.
+ */
+ private double doubleProperty = 321.0;
+
+ public double getDoubleProperty() {
+ return (this.doubleProperty);
+ }
+
+ public void setDoubleProperty(double doubleProperty) {
+ this.doubleProperty = doubleProperty;
+ }
+
+
+ /**
+ * A float property.
+ */
+ private float floatProperty = (float) 123.0;
+
+ public float getFloatProperty() {
+ return (this.floatProperty);
+ }
+
+ public void setFloatProperty(float floatProperty) {
+ this.floatProperty = floatProperty;
+ }
+
+
+ /**
+ * An integer property.
+ */
+ private int intProperty = 123;
+
+ public int getIntProperty() {
+ return (this.intProperty);
+ }
+
+ public void setIntProperty(int intProperty) {
+ this.intProperty = intProperty;
+ }
+
+
+ /**
+ * A long property.
+ */
+ private long longProperty = 321;
+
+ public long getLongProperty() {
+ return (this.longProperty);
+ }
+
+ public void setLongProperty(long longProperty) {
+ this.longProperty = longProperty;
+ }
+
+
+ /**
+ * A short property.
+ */
+ private short shortProperty = (short) 987;
+
+ public short getShortProperty() {
+ return (this.shortProperty);
+ }
+
+ public void setShortProperty(short shortProperty) {
+ this.shortProperty = shortProperty;
+ }
+
+
+ /**
+ * A String property.
+ */
+ private String stringProperty = "This is a string";
+
+ public String getStringProperty() {
+ return (this.stringProperty);
+ }
+
+ public void setStringProperty(String stringProperty) {
+ this.stringProperty = stringProperty;
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/BetaBean.java b/trunk/src/test/org/apache/commons/beanutils/BetaBean.java
new file mode 100644
index 0000000..059f782
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/BetaBean.java
@@ -0,0 +1,38 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+public class BetaBean extends AbstractChild {
+
+ private String secret = "utah";
+
+ public String getSecret() {
+ return secret;
+ }
+
+ public void setNoGetterProperty(String secret) {
+ this.secret = secret;
+ }
+
+ public void setNoGetterMappedProperty(String secret, String key) {
+ this.secret = "MAP:" + secret;
+ }
+
+ public BetaBean(String name) {
+ setName(name);
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/Child.java b/trunk/src/test/org/apache/commons/beanutils/Child.java
new file mode 100644
index 0000000..7fb49ac
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/Child.java
@@ -0,0 +1,23 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+public interface Child {
+
+ public String getName();
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/ConstructorUtilsTestCase.java b/trunk/src/test/org/apache/commons/beanutils/ConstructorUtilsTestCase.java
new file mode 100644
index 0000000..63fed48
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/ConstructorUtilsTestCase.java
@@ -0,0 +1,257 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.Constructor;
+import java.lang.reflect.Modifier;
+
+import junit.framework.Test;
+import junit.framework.TestCase;
+import junit.framework.TestSuite;
+
+
+/**
+ * <p> Test case for <code>ConstructorUtils</code> </p>
+ *
+ */
+public class ConstructorUtilsTestCase extends TestCase {
+
+ // ---------------------------------------------------------- Constructors
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public ConstructorUtilsTestCase(String name) {
+ super(name);
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+ super.setUp();
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(ConstructorUtilsTestCase.class));
+ }
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() throws Exception {
+ super.tearDown();
+ }
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+ public void testInvokeConstructor() throws Exception {
+ {
+ Object obj = ConstructorUtils.invokeConstructor(TestBean.class,"TEST");
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals("TEST",((TestBean)obj).getStringProperty());
+ }
+ {
+ Object obj = ConstructorUtils.invokeConstructor(TestBean.class,new Float(17.3f));
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals(17.3f,((TestBean)obj).getFloatProperty(),0.0f);
+ }
+ }
+
+ public void testInvokeConstructorWithArgArray() throws Exception {
+ Object[] args = { new Float(17.3f), "TEST" };
+ Object obj = ConstructorUtils.invokeConstructor(TestBean.class,args);
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals(17.3f,((TestBean)obj).getFloatProperty(),0.0f);
+ assertEquals("TEST",((TestBean)obj).getStringProperty());
+ }
+
+ public void testInvokeConstructorWithTypeArray() throws Exception {
+ {
+ Object[] args = { Boolean.TRUE, "TEST" };
+ Class[] types = { Boolean.TYPE, String.class };
+ Object obj = ConstructorUtils.invokeConstructor(TestBean.class,args,types);
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals(true,((TestBean)obj).getBooleanProperty());
+ assertEquals("TEST",((TestBean)obj).getStringProperty());
+ }
+ {
+ Object[] args = { Boolean.TRUE, "TEST" };
+ Class[] types = { Boolean.class, String.class };
+ Object obj = ConstructorUtils.invokeConstructor(TestBean.class,args,types);
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals(true,((TestBean)obj).isBooleanSecond());
+ assertEquals("TEST",((TestBean)obj).getStringProperty());
+ }
+ }
+
+ public void testInvokeExactConstructor() throws Exception {
+ {
+ Object obj = ConstructorUtils.invokeExactConstructor(TestBean.class,"TEST");
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals("TEST",((TestBean)obj).getStringProperty());
+ }
+ {
+ try {
+ Object obj = ConstructorUtils.invokeExactConstructor(TestBean.class,new Float(17.3f));
+ fail("Expected NoSuchMethodException");
+ } catch(NoSuchMethodException e) {
+ // expected
+ }
+ }
+ {
+ Object obj = ConstructorUtils.invokeExactConstructor(TestBean.class,Boolean.TRUE);
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals(true,((TestBean)obj).isBooleanSecond());
+ }
+ }
+
+ public void testInvokeExactConstructorWithArgArray() throws Exception {
+ {
+ Object[] args = { new Float(17.3f), "TEST" };
+ try {
+ Object obj = ConstructorUtils.invokeExactConstructor(TestBean.class,args);
+ fail("Expected NoSuchMethodException");
+ } catch(NoSuchMethodException e) {
+ // expected
+ }
+ }
+ {
+ Object[] args = { Boolean.TRUE, "TEST" };
+ Object obj = ConstructorUtils.invokeExactConstructor(TestBean.class,args);
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals(true,((TestBean)obj).isBooleanSecond());
+ assertEquals("TEST",((TestBean)obj).getStringProperty());
+ }
+ }
+
+ public void testInvokeExactConstructorWithTypeArray() throws Exception {
+ {
+ Object[] args = { Boolean.TRUE, "TEST" };
+ Class[] types = { Boolean.TYPE, String.class };
+ Object obj = ConstructorUtils.invokeExactConstructor(TestBean.class,args,types);
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals(true,((TestBean)obj).getBooleanProperty());
+ assertEquals("TEST",((TestBean)obj).getStringProperty());
+ }
+ {
+ Object[] args = { Boolean.TRUE, "TEST" };
+ Class[] types = { Boolean.class, String.class };
+ Object obj = ConstructorUtils.invokeExactConstructor(TestBean.class,args,types);
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals(true,((TestBean)obj).isBooleanSecond());
+ assertEquals("TEST",((TestBean)obj).getStringProperty());
+ }
+ {
+ Object[] args = { new Float(17.3f), "TEST" };
+ Class[] types = { Float.TYPE, String.class };
+ Object obj = ConstructorUtils.invokeExactConstructor(TestBean.class,args,types);
+ assertNotNull(obj);
+ assertTrue(obj instanceof TestBean);
+ assertEquals(17.3f,((TestBean)obj).getFloatProperty(),0.0f);
+ assertEquals("TEST",((TestBean)obj).getStringProperty());
+ }
+ {
+ Object[] args = { new Float(17.3f), "TEST" };
+ Class[] types = { Float.class, String.class };
+ try {
+ Object obj = ConstructorUtils.invokeExactConstructor(TestBean.class,args,types);
+ fail("Expected NoSuchMethodException");
+ } catch(NoSuchMethodException e) {
+ // expected
+ }
+ }
+ }
+
+ public void testGetAccessibleConstructor() throws Exception {
+ {
+ Constructor ctor = ConstructorUtils.getAccessibleConstructor(TestBean.class,String.class);
+ assertNotNull(ctor);
+ assertTrue(Modifier.isPublic(ctor.getModifiers()));
+ }
+ {
+ Constructor ctor = ConstructorUtils.getAccessibleConstructor(TestBean.class,Integer.class);
+ assertNotNull(ctor);
+ assertTrue(Modifier.isPublic(ctor.getModifiers()));
+ }
+ {
+ Constructor ctor = ConstructorUtils.getAccessibleConstructor(TestBean.class,Integer.TYPE);
+ assertNull(ctor);
+ }
+ }
+
+ public void testGetAccessibleConstructorWithTypeArray() throws Exception {
+ {
+ Class[] types = { Boolean.TYPE, String.class };
+ Constructor ctor = ConstructorUtils.getAccessibleConstructor(TestBean.class,types);
+ assertNotNull(ctor);
+ assertTrue(Modifier.isPublic(ctor.getModifiers()));
+ }
+ {
+ Class[] types = { Boolean.TYPE, Boolean.TYPE, String.class };
+ Constructor ctor = ConstructorUtils.getAccessibleConstructor(TestBean.class,types);
+ assertNull(ctor);
+ }
+ }
+
+ public void testGetAccessibleConstructorWithConstructorArg() throws Exception {
+ {
+ Class[] types = { Integer.class };
+ Constructor c1 = TestBean.class.getConstructor(types);
+ Constructor ctor = ConstructorUtils.getAccessibleConstructor(c1);
+ assertNotNull(ctor);
+ assertTrue(Modifier.isPublic(ctor.getModifiers()));
+ }
+ {
+ Class[] types = { Integer.class };
+ Constructor c1 = TestBean.class.getDeclaredConstructor(types);
+ Constructor ctor = ConstructorUtils.getAccessibleConstructor(c1);
+ assertNotNull(ctor);
+ assertTrue(Modifier.isPublic(ctor.getModifiers()));
+ }
+ {
+ Class[] types = { Integer.TYPE };
+ Constructor c1 = TestBean.class.getDeclaredConstructor(types);
+ Constructor ctor = ConstructorUtils.getAccessibleConstructor(c1);
+ assertNull(ctor);
+ }
+ }
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/ConvertUtilsTestCase.java b/trunk/src/test/org/apache/commons/beanutils/ConvertUtilsTestCase.java
new file mode 100644
index 0000000..02537db
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/ConvertUtilsTestCase.java
@@ -0,0 +1,654 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.sql.Date;
+import java.sql.Time;
+import java.sql.Timestamp;
+import java.util.HashMap;
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+import org.apache.commons.beanutils.converters.BooleanConverter;
+
+
+/**
+ * <p>
+ * Test Case for the ConvertUtils class.
+ * </p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.10 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class ConvertUtilsTestCase extends TestCase {
+
+ // ---------------------------------------------------- Instance Variables
+
+
+ // ---------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public ConvertUtilsTestCase(String name) {
+ super(name);
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() {
+
+ ConvertUtils.deregister();
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(ConvertUtilsTestCase.class));
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ ; // No action required
+ }
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ /**
+ * Negative String to primitive integer array tests.
+ */
+ public void testNegativeIntegerArray() {
+
+ Object value = null;
+ int intArray[] = new int[0];
+
+ value = ConvertUtils.convert((String) null, intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = ConvertUtils.convert("a", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = ConvertUtils.convert("{ a }", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = ConvertUtils.convert("1a3", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = ConvertUtils.convert("{ 1a3 }", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = ConvertUtils.convert("0,1a3", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = ConvertUtils.convert("{ 0, 1a3 }", intArray.getClass());
+ checkIntegerArray(value, intArray);
+
+
+ }
+
+
+ /**
+ * Negative scalar conversion tests. These rely on the standard
+ * default value conversions in ConvertUtils.
+ */
+ public void testNegativeScalar() {
+
+ Object value = null;
+
+ value = ConvertUtils.convert("foo", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("foo", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("foo", Byte.TYPE);
+ assertTrue(value instanceof Byte);
+ assertEquals(((Byte) value).byteValue(), (byte) 0);
+
+ value = ConvertUtils.convert("foo", Byte.class);
+ assertTrue(value instanceof Byte);
+ assertEquals(((Byte) value).byteValue(), (byte) 0);
+
+ try {
+ value = ConvertUtils.convert
+ ("org.apache.commons.beanutils.Undefined", Class.class);
+ fail("Should have thrown conversion exception");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ value = ConvertUtils.convert("foo", Double.TYPE);
+ assertTrue(value instanceof Double);
+ assertEquals(((Double) value).doubleValue(), (double) 0.0,
+ (double) 0.005);
+
+ value = ConvertUtils.convert("foo", Double.class);
+ assertTrue(value instanceof Double);
+ assertEquals(((Double) value).doubleValue(), (double) 0.0,
+ (double) 0.005);
+
+ value = ConvertUtils.convert("foo", Float.TYPE);
+ assertTrue(value instanceof Float);
+ assertEquals(((Float) value).floatValue(), (float) 0.0,
+ (float) 0.005);
+
+ value = ConvertUtils.convert("foo", Float.class);
+ assertTrue(value instanceof Float);
+ assertEquals(((Float) value).floatValue(), (float) 0.0,
+ (float) 0.005);
+
+ value = ConvertUtils.convert("foo", Integer.TYPE);
+ assertTrue(value instanceof Integer);
+ assertEquals(((Integer) value).intValue(), (int) 0);
+
+ value = ConvertUtils.convert("foo", Integer.class);
+ assertTrue(value instanceof Integer);
+ assertEquals(((Integer) value).intValue(), (int) 0);
+
+ value = ConvertUtils.convert("foo", Byte.TYPE);
+ assertTrue(value instanceof Byte);
+ assertEquals(((Byte) value).byteValue(), (byte) 0);
+
+ value = ConvertUtils.convert("foo", Long.class);
+ assertTrue(value instanceof Long);
+ assertEquals(((Long) value).longValue(), (long) 0);
+
+ value = ConvertUtils.convert("foo", Short.TYPE);
+ assertTrue(value instanceof Short);
+ assertEquals(((Short) value).shortValue(), (short) 0);
+
+ value = ConvertUtils.convert("foo", Short.class);
+ assertTrue(value instanceof Short);
+ assertEquals(((Short) value).shortValue(), (short) 0);
+
+ }
+
+
+ /**
+ * Negative String to String array tests.
+ */
+ public void testNegativeStringArray() {
+
+ Object value = null;
+ String stringArray[] = new String[0];
+
+ value = ConvertUtils.convert((String) null, stringArray.getClass());
+ checkStringArray(value, stringArray);
+
+ }
+
+
+ /**
+ * Test conversion of object to string for arrays.
+ */
+ public void testObjectToStringArray() {
+
+ int intArray0[] = new int[0];
+ int intArray1[] = { 123 };
+ int intArray2[] = { 123, 456 };
+ String stringArray0[] = new String[0];
+ String stringArray1[] = { "abc" };
+ String stringArray2[] = { "abc", "def" };
+
+ assertEquals("intArray0", null,
+ ConvertUtils.convert(intArray0));
+ assertEquals("intArray1", "123",
+ ConvertUtils.convert(intArray1));
+ assertEquals("intArray2", "123",
+ ConvertUtils.convert(intArray2));
+
+ assertEquals("stringArray0", null,
+ ConvertUtils.convert(stringArray0));
+ assertEquals("stringArray1", "abc",
+ ConvertUtils.convert(stringArray1));
+ assertEquals("stringArray2", "abc",
+ ConvertUtils.convert(stringArray2));
+
+ }
+
+
+ /**
+ * Test conversion of object to string for scalars.
+ */
+ public void testObjectToStringScalar() {
+
+ assertEquals("Boolean->String", "false",
+ ConvertUtils.convert(Boolean.FALSE));
+ assertEquals("Boolean->String", "true",
+ ConvertUtils.convert(Boolean.TRUE));
+ assertEquals("Byte->String", "123",
+ ConvertUtils.convert(new Byte((byte) 123)));
+ assertEquals("Character->String", "a",
+ ConvertUtils.convert(new Character('a')));
+ assertEquals("Double->String", "123.0",
+ ConvertUtils.convert(new Double((double) 123.0)));
+ assertEquals("Float->String", "123.0",
+ ConvertUtils.convert(new Float((float) 123.0)));
+ assertEquals("Integer->String", "123",
+ ConvertUtils.convert(new Integer((int) 123)));
+ assertEquals("Long->String", "123",
+ ConvertUtils.convert(new Long((long) 123)));
+ assertEquals("Short->String", "123",
+ ConvertUtils.convert(new Short((short) 123)));
+ assertEquals("String->String", "abc",
+ ConvertUtils.convert("abc"));
+ assertEquals("String->String null", null,
+ ConvertUtils.convert(null));
+
+ }
+
+
+ /**
+ * Positive array conversion tests.
+ */
+ public void testPositiveArray() {
+
+ String values1[] = { "10", "20", "30" };
+ Object value = ConvertUtils.convert(values1, Integer.TYPE);
+ int shape[] = new int[0];
+ assertEquals(shape.getClass(), value.getClass());
+ int results1[] = (int[]) value;
+ assertEquals(results1[0], 10);
+ assertEquals(results1[1], 20);
+ assertEquals(results1[2], 30);
+
+ String values2[] = { "100", "200", "300" };
+ value = ConvertUtils.convert(values2, shape.getClass());
+ assertEquals(shape.getClass(), value.getClass());
+ int results2[] = (int[]) value;
+ assertEquals(results2[0], 100);
+ assertEquals(results2[1], 200);
+ assertEquals(results2[2], 300);
+
+ }
+
+
+ /**
+ * Positive String to primitive integer array tests.
+ */
+ public void testPositiveIntegerArray() {
+
+ Object value = null;
+ int intArray[] = new int[0];
+ int intArray1[] = new int[] { 0 };
+ int intArray2[] = new int[] { 0, 10 };
+
+ value = ConvertUtils.convert("{ }", intArray.getClass());
+ checkIntegerArray(value, intArray);
+
+ value = ConvertUtils.convert("0", intArray.getClass());
+ checkIntegerArray(value, intArray1);
+ value = ConvertUtils.convert(" 0 ", intArray.getClass());
+ checkIntegerArray(value, intArray1);
+ value = ConvertUtils.convert("{ 0 }", intArray.getClass());
+ checkIntegerArray(value, intArray1);
+
+ value = ConvertUtils.convert("0,10", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = ConvertUtils.convert("0 10", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = ConvertUtils.convert("{0,10}", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = ConvertUtils.convert("{0 10}", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = ConvertUtils.convert("{ 0, 10 }", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = ConvertUtils.convert("{ 0 10 }", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+
+ }
+
+
+ /**
+ * Positive scalar conversion tests.
+ */
+ public void testPositiveScalar() {
+
+ Object value = null;
+
+ value = ConvertUtils.convert("true", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = ConvertUtils.convert("true", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = ConvertUtils.convert("yes", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = ConvertUtils.convert("yes", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = ConvertUtils.convert("y", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = ConvertUtils.convert("y", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = ConvertUtils.convert("on", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = ConvertUtils.convert("on", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = ConvertUtils.convert("false", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("false", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("no", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("no", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("n", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("n", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("off", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("off", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = ConvertUtils.convert("123", Byte.TYPE);
+ assertTrue(value instanceof Byte);
+ assertEquals(((Byte) value).byteValue(), (byte) 123);
+
+ value = ConvertUtils.convert("123", Byte.class);
+ assertTrue(value instanceof Byte);
+ assertEquals(((Byte) value).byteValue(), (byte) 123);
+
+ value = ConvertUtils.convert("a", Character.TYPE);
+ assertTrue(value instanceof Character);
+ assertEquals(((Character) value).charValue(), 'a');
+
+ value = ConvertUtils.convert("a", Character.class);
+ assertTrue(value instanceof Character);
+ assertEquals(((Character) value).charValue(), 'a');
+
+ value = ConvertUtils.convert("java.lang.String", Class.class);
+ assertTrue(value instanceof Class);
+ assertEquals(String.class, (Class) value);
+
+ value = ConvertUtils.convert("123.456", Double.TYPE);
+ assertTrue(value instanceof Double);
+ assertEquals(((Double) value).doubleValue(), (double) 123.456,
+ (double) 0.005);
+
+ value = ConvertUtils.convert("123.456", Double.class);
+ assertTrue(value instanceof Double);
+ assertEquals(((Double) value).doubleValue(), (double) 123.456,
+ (double) 0.005);
+
+ value = ConvertUtils.convert("123.456", Float.TYPE);
+ assertTrue(value instanceof Float);
+ assertEquals(((Float) value).floatValue(), (float) 123.456,
+ (float) 0.005);
+
+ value = ConvertUtils.convert("123.456", Float.class);
+ assertTrue(value instanceof Float);
+ assertEquals(((Float) value).floatValue(), (float) 123.456,
+ (float) 0.005);
+
+ value = ConvertUtils.convert("123", Integer.TYPE);
+ assertTrue(value instanceof Integer);
+ assertEquals(((Integer) value).intValue(), (int) 123);
+
+ value = ConvertUtils.convert("123", Integer.class);
+ assertTrue(value instanceof Integer);
+ assertEquals(((Integer) value).intValue(), (int) 123);
+
+ value = ConvertUtils.convert("123", Long.TYPE);
+ assertTrue(value instanceof Long);
+ assertEquals(((Long) value).longValue(), (long) 123);
+
+ value = ConvertUtils.convert("123", Long.class);
+ assertTrue(value instanceof Long);
+ assertEquals(((Long) value).longValue(), (long) 123);
+
+ value = ConvertUtils.convert("123", Short.TYPE);
+ assertTrue(value instanceof Short);
+ assertEquals(((Short) value).shortValue(), (short) 123);
+
+ value = ConvertUtils.convert("123", Short.class);
+ assertTrue(value instanceof Short);
+ assertEquals(((Short) value).shortValue(), (short) 123);
+
+ String input = null;
+
+ input = "2002-03-17";
+ value = ConvertUtils.convert(input, Date.class);
+ assertTrue(value instanceof Date);
+ assertEquals(input, value.toString());
+
+ input = "20:30:40";
+ value = ConvertUtils.convert(input, Time.class);
+ assertTrue(value instanceof Time);
+ assertEquals(input, value.toString());
+
+ input = "2002-03-17 20:30:40.0";
+ value = ConvertUtils.convert(input, Timestamp.class);
+ assertTrue(value instanceof Timestamp);
+ assertEquals(input, value.toString());
+
+ }
+
+
+ /**
+ * Positive String to String array tests.
+ */
+ public void testPositiveStringArray() {
+
+ Object value = null;
+ String stringArray[] = new String[0];
+ String stringArray1[] = new String[]
+ { "abc" };
+ String stringArray2[] = new String[]
+ { "abc", "de,f" };
+
+ value = ConvertUtils.convert("", stringArray.getClass());
+ checkStringArray(value, stringArray);
+ value = ConvertUtils.convert(" ", stringArray.getClass());
+ checkStringArray(value, stringArray);
+ value = ConvertUtils.convert("{}", stringArray.getClass());
+ checkStringArray(value, stringArray);
+ value = ConvertUtils.convert("{ }", stringArray.getClass());
+ checkStringArray(value, stringArray);
+
+ value = ConvertUtils.convert("abc", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = ConvertUtils.convert("{abc}", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = ConvertUtils.convert("\"abc\"", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = ConvertUtils.convert("{\"abc\"}", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = ConvertUtils.convert("'abc'", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = ConvertUtils.convert("{'abc'}", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+
+ value = ConvertUtils.convert("abc 'de,f'",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = ConvertUtils.convert("{abc, 'de,f'}",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = ConvertUtils.convert("\"abc\",\"de,f\"",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = ConvertUtils.convert("{\"abc\" 'de,f'}",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = ConvertUtils.convert("'abc' 'de,f'",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = ConvertUtils.convert("{'abc', \"de,f\"}",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+
+
+ }
+
+ public void testSeparateConvertInstances() throws Exception {
+ ConvertUtilsBean utilsOne = new ConvertUtilsBean();
+ ConvertUtilsBean utilsTwo = new ConvertUtilsBean();
+
+ // make sure that the test work ok before anything's changed
+ Object
+ value = utilsOne.convert("true", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(
+ "Standard conversion failed (1)",
+ ((Boolean) value).booleanValue(),
+ true);
+
+ value = utilsTwo.convert("true", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(
+ "Standard conversion failed (2)",
+ ((Boolean) value).booleanValue(),
+ true);
+
+ // now register a test
+
+ utilsOne.register(new ThrowExceptionConverter(), Boolean.TYPE);
+ try {
+
+ utilsOne.convert("true", Boolean.TYPE);
+ fail("Register converter failed.");
+
+ } catch (PassTestException e) { /* This shows that the registration has worked */ }
+
+ try {
+ // nothing should have changed
+ value = utilsTwo.convert("true", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(
+ "Standard conversion failed (3)",
+ ((Boolean) value).booleanValue(),
+ true);
+
+ } catch (PassTestException e) {
+ // This is a failure since utilsTwo should still have
+ // standard converters registered
+ fail("Registering a converter for an instance should not effect another instance.");
+ }
+
+ // nothing we'll test deregister
+ utilsOne.deregister();
+ value = utilsOne.convert("true", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals("Instance deregister failed.", ((Boolean) value).booleanValue(), true);
+
+ value = utilsTwo.convert("true", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(
+ "Standard conversion failed (4)",
+ ((Boolean) value).booleanValue(),
+ true);
+ }
+
+ public void testDeregisteringSingleConverter() throws Exception {
+ ConvertUtils convertUtils = new ConvertUtils();
+
+ // make sure that the test work ok before anything's changed
+ Object
+ value = convertUtils.convert("true", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(
+ "Standard conversion failed (1)",
+ ((Boolean) value).booleanValue(),
+ true);
+
+ // we'll test deregister
+ convertUtils.deregister(Boolean.TYPE);
+ assertNull("Converter should be null",convertUtils.lookup(Boolean.TYPE));
+
+ }
+
+ // -------------------------------------------------------- Private Methods
+
+
+ private void checkIntegerArray(Object value, int intArray[]) {
+
+ assertNotNull("Returned value is not null", value);
+ assertEquals("Returned value is int[]",
+ intArray.getClass(), value.getClass());
+ int results[] = (int[]) value;
+ assertEquals("Returned array length", intArray.length, results.length);
+ for (int i = 0; i < intArray.length; i++) {
+ assertEquals("Returned array value " + i,
+ intArray[i], results[i]);
+ }
+
+ }
+
+
+ private void checkStringArray(Object value, String stringArray[]) {
+
+ assertNotNull("Returned value is not null", value);
+ assertEquals("Returned value is String[]",
+ stringArray.getClass(), value.getClass());
+ String results[] = (String[]) value;
+ assertEquals("Returned array length",
+ stringArray.length, results.length);
+ for (int i = 0; i < stringArray.length; i++) {
+ assertEquals("Returned array value " + i,
+ stringArray[i], results[i]);
+ }
+
+ }
+
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/DynaBeanUtilsTestCase.java b/trunk/src/test/org/apache/commons/beanutils/DynaBeanUtilsTestCase.java
new file mode 100644
index 0000000..714ac12
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/DynaBeanUtilsTestCase.java
@@ -0,0 +1,1266 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Map;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * Test case for BeanUtils when the underlying bean is actually a DynaBean.
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.22 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class DynaBeanUtilsTestCase extends TestCase {
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The basic test bean for each test.
+ */
+ protected DynaBean bean = null;
+
+
+ /**
+ * The nested bean pointed at by the "nested" property.
+ */
+ protected TestBean nested = null;
+
+
+ /**
+ * The set of properties that should be described.
+ */
+ protected String describes[] =
+ { "booleanProperty",
+ "booleanSecond",
+ "byteProperty",
+ "doubleProperty",
+ "dupProperty",
+ "floatProperty",
+ "intArray",
+ "intIndexed",
+ "intProperty",
+ "listIndexed",
+ "longProperty",
+ "mapProperty",
+ "mappedProperty",
+ "mappedIntProperty",
+ "nested",
+ "nullProperty",
+ // "readOnlyProperty",
+ "shortProperty",
+ "stringArray",
+ "stringIndexed",
+ "stringProperty"
+ };
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public DynaBeanUtilsTestCase(String name) {
+
+ super(name);
+
+ }
+
+
+ // --------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+
+ // Instantiate a new DynaBean instance
+ DynaClass dynaClass = createDynaClass();
+ bean = dynaClass.newInstance();
+
+ // Initialize the DynaBean's property values (like TestBean)
+ bean.set("booleanProperty", new Boolean(true));
+ bean.set("booleanSecond", new Boolean(true));
+ bean.set("byteProperty", new Byte((byte) 121));
+ bean.set("doubleProperty", new Double(321.0));
+ bean.set("floatProperty", new Float((float) 123.0));
+ String dupProperty[] = { "Dup 0", "Dup 1", "Dup 2", "Dup 3", "Dup 4"};
+ bean.set("dupProperty", dupProperty);
+ int intArray[] = { 0, 10, 20, 30, 40 };
+ bean.set("intArray", intArray);
+ int intIndexed[] = { 0, 10, 20, 30, 40 };
+ bean.set("intIndexed", intIndexed);
+ bean.set("intProperty", new Integer(123));
+ List listIndexed = new ArrayList();
+ listIndexed.add("String 0");
+ listIndexed.add("String 1");
+ listIndexed.add("String 2");
+ listIndexed.add("String 3");
+ listIndexed.add("String 4");
+ bean.set("listIndexed", listIndexed);
+ bean.set("longProperty", new Long((long) 321));
+ HashMap mapProperty = new HashMap();
+ mapProperty.put("First Key", "First Value");
+ mapProperty.put("Second Key", "Second Value");
+ bean.set("mapProperty", mapProperty);
+ HashMap mappedProperty = new HashMap();
+ mappedProperty.put("First Key", "First Value");
+ mappedProperty.put("Second Key", "Second Value");
+ bean.set("mappedProperty", mappedProperty);
+ HashMap mappedIntProperty = new HashMap();
+ mappedIntProperty.put("One", new Integer(1));
+ mappedIntProperty.put("Two", new Integer(2));
+ bean.set("mappedIntProperty", mappedIntProperty);
+ nested = new TestBean();
+ bean.set("nested", nested);
+ // Property "nullProperty" is not initialized, so it should return null
+ bean.set("shortProperty", new Short((short) 987));
+ String stringArray[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+ bean.set("stringArray", stringArray);
+ String stringIndexed[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+ bean.set("stringIndexed", stringIndexed);
+ bean.set("stringProperty", "This is a string");
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+
+ return (new TestSuite(DynaBeanUtilsTestCase.class));
+
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+
+ bean = null;
+ nested = null;
+
+ }
+
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+ /**
+ * Test the cloneBean() method from a DynaBean.
+ */
+ public void testCloneDynaBean() {
+
+ // Set up an origin bean with customized properties
+ DynaClass dynaClass = DynaBeanUtilsTestCase.createDynaClass();
+ DynaBean orig = null;
+ try {
+ orig = dynaClass.newInstance();
+ } catch (Exception e) {
+ fail("newInstance(): " + e);
+ }
+ orig.set("booleanProperty", Boolean.FALSE);
+ orig.set("byteProperty", new Byte((byte)111));
+ orig.set("doubleProperty", new Double(333.33));
+ orig.set("dupProperty", new String[] { "New 0", "New 1", "New 2" });
+ orig.set("intArray", new int[] { 100, 200, 300 });
+ orig.set("intProperty", new Integer(333));
+ orig.set("longProperty", new Long(3333));
+ orig.set("shortProperty", new Short((short) 33));
+ orig.set("stringArray", new String[] { "New 0", "New 1" });
+ orig.set("stringProperty", "Custom string");
+
+ // Copy the origin bean to our destination test bean
+ DynaBean clonedBean = null;
+ try {
+ clonedBean = (DynaBean) BeanUtils.cloneBean(orig);
+ } catch (Exception e) {
+ fail("Threw exception: " + e);
+ }
+
+ // Validate the results for scalar properties
+ assertEquals("Cloned boolean property",
+ false,
+ ((Boolean) clonedBean.get("booleanProperty")).booleanValue());
+ assertEquals("Cloned byte property",
+ (byte) 111,
+ ((Byte) clonedBean.get("byteProperty")).byteValue());
+ assertEquals("Cloned double property",
+ 333.33,
+ ((Double) clonedBean.get("doubleProperty")).doubleValue(),
+ 0.005);
+ assertEquals("Cloned int property",
+ 333,
+ ((Integer) clonedBean.get("intProperty")).intValue());
+ assertEquals("Cloned long property",
+ (long) 3333,
+ ((Long) clonedBean.get("longProperty")).longValue());
+ assertEquals("Cloned short property",
+ (short) 33,
+ ((Short) clonedBean.get("shortProperty")).shortValue());
+ assertEquals("Cloned string property",
+ "Custom string",
+ (String) clonedBean.get("stringProperty"));
+
+ // Validate the results for array properties
+ String dupProperty[] = (String[]) clonedBean.get("dupProperty");
+ assertNotNull("dupProperty present", dupProperty);
+ assertEquals("dupProperty length", 3, dupProperty.length);
+ assertEquals("dupProperty[0]", "New 0", dupProperty[0]);
+ assertEquals("dupProperty[1]", "New 1", dupProperty[1]);
+ assertEquals("dupProperty[2]", "New 2", dupProperty[2]);
+ int intArray[] = (int[]) clonedBean.get("intArray");
+ assertNotNull("intArray present", intArray);
+ assertEquals("intArray length", 3, intArray.length);
+ assertEquals("intArray[0]", 100, intArray[0]);
+ assertEquals("intArray[1]", 200, intArray[1]);
+ assertEquals("intArray[2]", 300, intArray[2]);
+ String stringArray[] = (String[]) clonedBean.get("stringArray");
+ assertNotNull("stringArray present", stringArray);
+ assertEquals("stringArray length", 2, stringArray.length);
+ assertEquals("stringArray[0]", "New 0", stringArray[0]);
+ assertEquals("stringArray[1]", "New 1", stringArray[1]);
+
+ }
+
+ /**
+ * Test the copyProperties() method from a DynaBean.
+ */
+ public void testCopyPropertiesDynaBean() {
+
+ // Set up an origin bean with customized properties
+ DynaClass dynaClass = DynaBeanUtilsTestCase.createDynaClass();
+ DynaBean orig = null;
+ try {
+ orig = dynaClass.newInstance();
+ } catch (Exception e) {
+ fail("newInstance(): " + e);
+ }
+ orig.set("booleanProperty", Boolean.FALSE);
+ orig.set("byteProperty", new Byte((byte)111));
+ orig.set("doubleProperty", new Double(333.33));
+ orig.set("dupProperty", new String[] { "New 0", "New 1", "New 2" });
+ orig.set("intArray", new int[] { 100, 200, 300 });
+ orig.set("intProperty", new Integer(333));
+ orig.set("longProperty", new Long(3333));
+ orig.set("shortProperty", new Short((short) 33));
+ orig.set("stringArray", new String[] { "New 0", "New 1" });
+ orig.set("stringProperty", "Custom string");
+
+ // Copy the origin bean to our destination test bean
+ try {
+ BeanUtils.copyProperties(bean, orig);
+ } catch (Exception e) {
+ fail("Threw exception: " + e);
+ }
+
+ // Validate the results for scalar properties
+ assertEquals("Copied boolean property",
+ false,
+ ((Boolean) bean.get("booleanProperty")).booleanValue());
+ assertEquals("Copied byte property",
+ (byte) 111,
+ ((Byte) bean.get("byteProperty")).byteValue());
+ assertEquals("Copied double property",
+ 333.33,
+ ((Double) bean.get("doubleProperty")).doubleValue(),
+ 0.005);
+ assertEquals("Copied int property",
+ 333,
+ ((Integer) bean.get("intProperty")).intValue());
+ assertEquals("Copied long property",
+ (long) 3333,
+ ((Long) bean.get("longProperty")).longValue());
+ assertEquals("Copied short property",
+ (short) 33,
+ ((Short) bean.get("shortProperty")).shortValue());
+ assertEquals("Copied string property",
+ "Custom string",
+ (String) bean.get("stringProperty"));
+
+ // Validate the results for array properties
+ String dupProperty[] = (String[]) bean.get("dupProperty");
+ assertNotNull("dupProperty present", dupProperty);
+ assertEquals("dupProperty length", 3, dupProperty.length);
+ assertEquals("dupProperty[0]", "New 0", dupProperty[0]);
+ assertEquals("dupProperty[1]", "New 1", dupProperty[1]);
+ assertEquals("dupProperty[2]", "New 2", dupProperty[2]);
+ int intArray[] = (int[]) bean.get("intArray");
+ assertNotNull("intArray present", intArray);
+ assertEquals("intArray length", 3, intArray.length);
+ assertEquals("intArray[0]", 100, intArray[0]);
+ assertEquals("intArray[1]", 200, intArray[1]);
+ assertEquals("intArray[2]", 300, intArray[2]);
+ String stringArray[] = (String[]) bean.get("stringArray");
+ assertNotNull("stringArray present", stringArray);
+ assertEquals("stringArray length", 2, stringArray.length);
+ assertEquals("stringArray[0]", "New 0", stringArray[0]);
+ assertEquals("stringArray[1]", "New 1", stringArray[1]);
+
+ }
+
+
+ /**
+ * Test copyProperties() when the origin is a a <code>Map</code>.
+ */
+ public void testCopyPropertiesMap() {
+
+ Map map = new HashMap();
+ map.put("booleanProperty", "false");
+ map.put("byteProperty", "111");
+ map.put("doubleProperty", "333.0");
+ map.put("dupProperty", new String[] { "New 0", "New 1", "New 2" });
+ map.put("floatProperty", "222.0");
+ map.put("intArray", new String[] { "0", "100", "200" });
+ map.put("intProperty", "111");
+ map.put("longProperty", "444");
+ map.put("shortProperty", "555");
+ map.put("stringProperty", "New String Property");
+
+ try {
+ BeanUtils.copyProperties(bean, map);
+ } catch (Throwable t) {
+ fail("Threw " + t.toString());
+ }
+
+ // Scalar properties
+ assertEquals("booleanProperty", false,
+ ((Boolean) bean.get("booleanProperty")).booleanValue());
+ assertEquals("byteProperty", (byte) 111,
+ ((Byte) bean.get("byteProperty")).byteValue());
+ assertEquals("doubleProperty", 333.0,
+ ((Double) bean.get("doubleProperty")).doubleValue(),
+ 0.005);
+ assertEquals("floatProperty", (float) 222.0,
+ ((Float) bean.get("floatProperty")).floatValue(),
+ (float) 0.005);
+ assertEquals("intProperty", 111,
+ ((Integer) bean.get("intProperty")).intValue());
+ assertEquals("longProperty", (long) 444,
+ ((Long) bean.get("longProperty")).longValue());
+ assertEquals("shortProperty", (short) 555,
+ ((Short) bean.get("shortProperty")).shortValue());
+ assertEquals("stringProperty", "New String Property",
+ (String) bean.get("stringProperty"));
+
+ // Indexed Properties
+ String dupProperty[] = (String[]) bean.get("dupProperty");
+ assertNotNull("dupProperty present", dupProperty);
+ assertEquals("dupProperty length", 3, dupProperty.length);
+ assertEquals("dupProperty[0]", "New 0", dupProperty[0]);
+ assertEquals("dupProperty[1]", "New 1", dupProperty[1]);
+ assertEquals("dupProperty[2]", "New 2", dupProperty[2]);
+ int intArray[] = (int[]) bean.get("intArray");
+ assertNotNull("intArray present", intArray);
+ assertEquals("intArray length", 3, intArray.length);
+ assertEquals("intArray[0]", 0, intArray[0]);
+ assertEquals("intArray[1]", 100, intArray[1]);
+ assertEquals("intArray[2]", 200, intArray[2]);
+
+ }
+
+
+ /**
+ * Test the copyProperties() method from a standard JavaBean.
+ */
+ public void testCopyPropertiesStandard() {
+
+ // Set up an origin bean with customized properties
+ TestBean orig = new TestBean();
+ orig.setBooleanProperty(false);
+ orig.setByteProperty((byte) 111);
+ orig.setDoubleProperty(333.33);
+ orig.setDupProperty(new String[] { "New 0", "New 1", "New 2" });
+ orig.setIntArray(new int[] { 100, 200, 300 });
+ orig.setIntProperty(333);
+ orig.setLongProperty(3333);
+ orig.setShortProperty((short) 33);
+ orig.setStringArray(new String[] { "New 0", "New 1" });
+ orig.setStringProperty("Custom string");
+
+ // Copy the origin bean to our destination test bean
+ try {
+ BeanUtils.copyProperties(bean, orig);
+ } catch (Exception e) {
+ fail("Threw exception: " + e);
+ }
+
+ // Validate the results for scalar properties
+ assertEquals("Copied boolean property",
+ false,
+ ((Boolean) bean.get("booleanProperty")).booleanValue());
+ assertEquals("Copied byte property",
+ (byte) 111,
+ ((Byte) bean.get("byteProperty")).byteValue());
+ assertEquals("Copied double property",
+ 333.33,
+ ((Double) bean.get("doubleProperty")).doubleValue(),
+ 0.005);
+ assertEquals("Copied int property",
+ 333,
+ ((Integer) bean.get("intProperty")).intValue());
+ assertEquals("Copied long property",
+ (long) 3333,
+ ((Long) bean.get("longProperty")).longValue());
+ assertEquals("Copied short property",
+ (short) 33,
+ ((Short) bean.get("shortProperty")).shortValue());
+ assertEquals("Copied string property",
+ "Custom string",
+ (String) bean.get("stringProperty"));
+
+ // Validate the results for array properties
+ String dupProperty[] = (String[]) bean.get("dupProperty");
+ assertNotNull("dupProperty present", dupProperty);
+ assertEquals("dupProperty length", 3, dupProperty.length);
+ assertEquals("dupProperty[0]", "New 0", dupProperty[0]);
+ assertEquals("dupProperty[1]", "New 1", dupProperty[1]);
+ assertEquals("dupProperty[2]", "New 2", dupProperty[2]);
+ int intArray[] = (int[]) bean.get("intArray");
+ assertNotNull("intArray present", intArray);
+ assertEquals("intArray length", 3, intArray.length);
+ assertEquals("intArray[0]", 100, intArray[0]);
+ assertEquals("intArray[1]", 200, intArray[1]);
+ assertEquals("intArray[2]", 300, intArray[2]);
+ String stringArray[] = (String[]) bean.get("stringArray");
+ assertNotNull("stringArray present", stringArray);
+ assertEquals("stringArray length", 2, stringArray.length);
+ assertEquals("stringArray[0]", "New 0", stringArray[0]);
+ assertEquals("stringArray[1]", "New 1", stringArray[1]);
+
+ }
+
+
+ /**
+ * Test the describe() method.
+ */
+ public void testDescribe() {
+
+ Map map = null;
+ try {
+ map = PropertyUtils.describe(bean);
+ } catch (Exception e) {
+ fail("Threw exception " + e);
+ }
+
+ // Verify existence of all the properties that should be present
+ for (int i = 0; i < describes.length; i++) {
+ assertTrue("Property '" + describes[i] + "' is present",
+ map.containsKey(describes[i]));
+ }
+ assertTrue("Property 'writeOnlyProperty' is not present",
+ !map.containsKey("writeOnlyProperty"));
+
+ // Verify the values of scalar properties
+ assertEquals("Value of 'booleanProperty'",
+ Boolean.TRUE,
+ (Boolean) map.get("booleanProperty"));
+ assertEquals("Value of 'byteProperty'",
+ new Byte((byte) 121),
+ (Byte) map.get("byteProperty"));
+ assertEquals("Value of 'doubleProperty'",
+ new Double(321.0),
+ (Double) map.get("doubleProperty"));
+ assertEquals("Value of 'floatProperty'",
+ new Float((float) 123.0),
+ (Float) map.get("floatProperty"));
+ assertEquals("Value of 'intProperty'",
+ new Integer(123),
+ (Integer) map.get("intProperty"));
+ assertEquals("Value of 'longProperty'",
+ new Long(321),
+ (Long) map.get("longProperty"));
+ assertEquals("Value of 'shortProperty'",
+ new Short((short) 987),
+ (Short) map.get("shortProperty"));
+ assertEquals("Value of 'stringProperty'",
+ "This is a string",
+ (String) map.get("stringProperty"));
+
+ }
+
+
+ /**
+ * Test populate() method on array properties as a whole.
+ */
+ public void testPopulateArrayProperties() {
+
+ try {
+
+ HashMap map = new HashMap();
+ // int intArray[] = new int[] { 123, 456, 789 };
+ String intArrayIn[] = new String[] { "123", "456", "789" };
+ map.put("intArray", intArrayIn);
+ String stringArray[] = new String[]
+ { "New String 0", "New String 1" };
+ map.put("stringArray", stringArray);
+
+ BeanUtils.populate(bean, map);
+
+ int intArray[] = (int[]) bean.get("intArray");
+ assertNotNull("intArray is present", intArray);
+ assertEquals("intArray length",
+ 3, intArray.length);
+ assertEquals("intArray[0]", 123, intArray[0]);
+ assertEquals("intArray[1]", 456, intArray[1]);
+ assertEquals("intArray[2]", 789, intArray[2]);
+ stringArray = (String[]) bean.get("stringArray");
+ assertNotNull("stringArray is present", stringArray);
+ assertEquals("stringArray length", 2, stringArray.length);
+ assertEquals("stringArray[0]", "New String 0", stringArray[0]);
+ assertEquals("stringArray[1]", "New String 1", stringArray[1]);
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * tests the string and int arrays of TestBean
+ */
+ public void testGetArrayProperty() {
+ try {
+ String arr[] = BeanUtils.getArrayProperty(bean, "stringArray");
+ String comp[] = (String[]) bean.get("stringArray");
+
+ assertTrue("String array length = " + comp.length,
+ (comp.length == arr.length));
+
+ arr = BeanUtils.getArrayProperty(bean, "intArray");
+ int iarr[] = (int[]) bean.get("intArray");
+
+ assertTrue("String array length = " + iarr.length,
+ (iarr.length == arr.length));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * tests getting an indexed property
+ */
+ public void testGetIndexedProperty1() {
+ try {
+ String val = BeanUtils.getIndexedProperty(bean, "intIndexed[3]");
+ String comp = String.valueOf(bean.get("intIndexed", 3));
+ assertTrue("intIndexed[3] == " + comp, val.equals(comp));
+
+ val = BeanUtils.getIndexedProperty(bean, "stringIndexed[3]");
+ comp = (String) bean.get("stringIndexed", 3);
+ assertTrue("stringIndexed[3] == " + comp, val.equals(comp));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * tests getting an indexed property
+ */
+ public void testGetIndexedProperty2() {
+ try {
+ String val = BeanUtils.getIndexedProperty(bean, "intIndexed", 3);
+ String comp = String.valueOf(bean.get("intIndexed", 3));
+
+ assertTrue("intIndexed,3 == " + comp, val.equals(comp));
+
+ val = BeanUtils.getIndexedProperty(bean, "stringIndexed", 3);
+ comp = (String) bean.get("stringIndexed", 3);
+
+ assertTrue("stringIndexed,3 == " + comp, val.equals(comp));
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * tests getting a nested property
+ */
+ public void testGetNestedProperty() {
+ try {
+ String val = BeanUtils.getNestedProperty(bean, "nested.stringProperty");
+ String comp = nested.getStringProperty();
+ assertTrue("nested.StringProperty == " + comp,
+ val.equals(comp));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * tests getting a 'whatever' property
+ */
+ public void testGetGeneralProperty() {
+ try {
+ String val = BeanUtils.getProperty(bean, "nested.intIndexed[2]");
+ String comp = String.valueOf(bean.get("intIndexed", 2));
+
+ assertTrue("nested.intIndexed[2] == " + comp,
+ val.equals(comp));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * tests getting a 'whatever' property
+ */
+ public void testGetSimpleProperty() {
+ try {
+ String val = BeanUtils.getSimpleProperty(bean, "shortProperty");
+ String comp = String.valueOf(bean.get("shortProperty"));
+
+ assertTrue("shortProperty == " + comp,
+ val.equals(comp));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+ }
+
+
+ /**
+ * Test populate() method on individual array elements.
+ */
+ public void testPopulateArrayElements() {
+
+ try {
+
+ HashMap map = new HashMap();
+ map.put("intIndexed[0]", "100");
+ map.put("intIndexed[2]", "120");
+ map.put("intIndexed[4]", "140");
+
+ BeanUtils.populate(bean, map);
+ Integer intIndexed0 = (Integer) bean.get("intIndexed", 0);
+ assertEquals("intIndexed[0] is 100",
+ 100, intIndexed0.intValue());
+ Integer intIndexed1 = (Integer) bean.get("intIndexed", 1);
+ assertEquals("intIndexed[1] is 10",
+ 10, intIndexed1.intValue());
+ Integer intIndexed2 = (Integer) bean.get("intIndexed", 2);
+ assertEquals("intIndexed[2] is 120",
+ 120, intIndexed2.intValue());
+ Integer intIndexed3 = (Integer) bean.get("intIndexed", 3);
+ assertEquals("intIndexed[3] is 30",
+ 30, intIndexed3.intValue());
+ Integer intIndexed4 = (Integer) bean.get("intIndexed", 4);
+ assertEquals("intIndexed[4] is 140",
+ 140, intIndexed4.intValue());
+
+ map.clear();
+ map.put("stringIndexed[1]", "New String 1");
+ map.put("stringIndexed[3]", "New String 3");
+
+ BeanUtils.populate(bean, map);
+
+ assertEquals("stringIndexed[0] is \"String 0\"",
+ "String 0",
+ (String) bean.get("stringIndexed", 0));
+ assertEquals("stringIndexed[1] is \"New String 1\"",
+ "New String 1",
+ (String) bean.get("stringIndexed", 1));
+ assertEquals("stringIndexed[2] is \"String 2\"",
+ "String 2",
+ (String) bean.get("stringIndexed", 2));
+ assertEquals("stringIndexed[3] is \"New String 3\"",
+ "New String 3",
+ (String) bean.get("stringIndexed", 3));
+ assertEquals("stringIndexed[4] is \"String 4\"",
+ "String 4",
+ (String) bean.get("stringIndexed", 4));
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * Test populate() on mapped properties.
+ */
+ public void testPopulateMapped() {
+
+ try {
+
+ HashMap map = new HashMap();
+ map.put("mappedProperty(First Key)", "New First Value");
+ map.put("mappedProperty(Third Key)", "New Third Value");
+
+ BeanUtils.populate(bean, map);
+
+ assertEquals("mappedProperty(First Key)",
+ "New First Value",
+ (String) bean.get("mappedProperty", "First Key"));
+ assertEquals("mappedProperty(Second Key)",
+ "Second Value",
+ (String) bean.get("mappedProperty", "Second Key"));
+ assertEquals("mappedProperty(Third Key)",
+ "New Third Value",
+ (String) bean.get("mappedProperty", "Third Key"));
+ assertNull("mappedProperty(Fourth Key",
+ (String) bean.get("mappedProperty", "Fourth Key"));
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * Test populate() method on nested properties.
+ */
+ public void testPopulateNested() {
+
+ try {
+
+ HashMap map = new HashMap();
+ map.put("nested.booleanProperty", "false");
+ // booleanSecond is left at true
+ map.put("nested.doubleProperty", "432.0");
+ // floatProperty is left at 123.0
+ map.put("nested.intProperty", "543");
+ // longProperty is left at 321
+ map.put("nested.shortProperty", "654");
+ // stringProperty is left at "This is a string"
+
+ BeanUtils.populate(bean, map);
+
+ TestBean nested = (TestBean) bean.get("nested");
+ assertTrue("booleanProperty is false",
+ !nested.getBooleanProperty());
+ assertTrue("booleanSecond is true",
+ nested.isBooleanSecond());
+ assertEquals("doubleProperty is 432.0",
+ (double) 432.0,
+ nested.getDoubleProperty(),
+ (double) 0.005);
+ assertEquals("floatProperty is 123.0",
+ (float) 123.0,
+ nested.getFloatProperty(),
+ (float) 0.005);
+ assertEquals("intProperty is 543",
+ 543, nested.getIntProperty());
+ assertEquals("longProperty is 321",
+ (long) 321, nested.getLongProperty());
+ assertEquals("shortProperty is 654",
+ (short) 654, nested.getShortProperty());
+ assertEquals("stringProperty is \"This is a string\"",
+ "This is a string",
+ nested.getStringProperty());
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * Test populate() method on scalar properties.
+ */
+ public void testPopulateScalar() {
+
+ try {
+
+ bean.set("nullProperty", "non-null value");
+
+ HashMap map = new HashMap();
+ map.put("booleanProperty", "false");
+ // booleanSecond is left at true
+ map.put("doubleProperty", "432.0");
+ // floatProperty is left at 123.0
+ map.put("intProperty", "543");
+ // longProperty is left at 321
+ map.put("nullProperty", null);
+ map.put("shortProperty", "654");
+ // stringProperty is left at "This is a string"
+
+ BeanUtils.populate(bean, map);
+
+ Boolean booleanProperty = (Boolean) bean.get("booleanProperty");
+ assertTrue("booleanProperty is false", !booleanProperty.booleanValue());
+ Boolean booleanSecond = (Boolean) bean.get("booleanSecond");
+ assertTrue("booleanSecond is true", booleanSecond.booleanValue());
+ Double doubleProperty = (Double) bean.get("doubleProperty");
+ assertEquals("doubleProperty is 432.0",
+ (double) 432.0, doubleProperty.doubleValue(),
+ (double) 0.005);
+ Float floatProperty = (Float) bean.get("floatProperty");
+ assertEquals("floatProperty is 123.0",
+ (float) 123.0, floatProperty.floatValue(),
+ (float) 0.005);
+ Integer intProperty = (Integer) bean.get("intProperty");
+ assertEquals("intProperty is 543",
+ 543, intProperty.intValue());
+ Long longProperty = (Long) bean.get("longProperty");
+ assertEquals("longProperty is 321",
+ (long) 321, longProperty.longValue());
+ assertNull("nullProperty is null", bean.get("nullProperty"));
+ Short shortProperty = (Short) bean.get("shortProperty");
+ assertEquals("shortProperty is 654",
+ (short) 654, shortProperty.shortValue());
+ assertEquals("stringProperty is \"This is a string\"",
+ "This is a string",
+ (String) bean.get("stringProperty"));
+
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ }
+
+ }
+
+
+ /**
+ * Test calling setProperty() with null property values.
+ */
+ public void testSetPropertyNullValues() throws Exception {
+
+ Object oldValue = null;
+ Object newValue = null;
+
+ // Scalar value into array
+ oldValue = PropertyUtils.getSimpleProperty(bean, "stringArray");
+ BeanUtils.setProperty(bean, "stringArray", (String) null);
+ newValue = PropertyUtils.getSimpleProperty(bean, "stringArray");
+ assertNotNull("stringArray is not null", newValue);
+ assertTrue("stringArray of correct type",
+ newValue instanceof String[]);
+ assertEquals("stringArray length",
+ 1, ((String[]) newValue).length);
+ assertTrue("stringArray[0] is null",
+ ((String[]) newValue)[0] == null);
+ PropertyUtils.setProperty(bean, "stringArray", oldValue);
+
+ // Indexed value into array
+ oldValue = PropertyUtils.getSimpleProperty(bean, "stringArray");
+ BeanUtils.setProperty(bean, "stringArray[2]", (String) null);
+ newValue = PropertyUtils.getSimpleProperty(bean, "stringArray");
+ assertNotNull("stringArray is not null", newValue);
+ assertTrue("stringArray of correct type",
+ newValue instanceof String[]);
+ assertEquals("stringArray length",
+ 5, ((String[]) newValue).length);
+ assertTrue("stringArray[2] is null",
+ ((String[]) newValue)[2] == null);
+ PropertyUtils.setProperty(bean, "stringArray", oldValue);
+
+ // Value into scalar
+ BeanUtils.setProperty(bean, "stringProperty", null);
+ assertTrue("stringProperty is now null",
+ BeanUtils.getProperty(bean, "stringProperty") == null);
+
+ }
+
+
+ /**
+ * Test converting to and from primitive wrapper types.
+ */
+ public void testSetPropertyOnPrimitiveWrappers() throws Exception {
+
+ BeanUtils.setProperty(bean,"intProperty", new Integer(1));
+ assertEquals(1,((Integer) bean.get("intProperty")).intValue());
+ BeanUtils.setProperty(bean,"stringProperty", new Integer(1));
+ assertEquals(1, Integer.parseInt((String) bean.get("stringProperty")));
+
+ }
+
+
+ /**
+ * Test setting a null property value.
+ */
+ public void testSetPropertyNull() throws Exception {
+
+ bean.set("nullProperty", "non-null value");
+ BeanUtils.setProperty(bean, "nullProperty", null);
+ assertNull("nullProperty is null", bean.get("nullProperty"));
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on byte.
+ */
+ public void testCopyPropertyByte() throws Exception {
+
+ BeanUtils.setProperty(bean, "byteProperty", new Byte((byte) 123));
+ assertEquals((byte) 123, ((Byte) bean.get("byteProperty")).byteValue());
+/*
+ BeanUtils.setProperty(bean, "byteProperty", new Double((double) 123));
+ assertEquals((byte) 123, ((Byte) bean.get("byteProperty")).byteValue());
+ BeanUtils.setProperty(bean, "byteProperty", new Float((float) 123));
+ assertEquals((byte) 123, ((Byte) bean.get("byteProperty")).byteValue());
+*/
+ BeanUtils.setProperty(bean, "byteProperty", new Integer((int) 123));
+ assertEquals((byte) 123, ((Byte) bean.get("byteProperty")).byteValue());
+ BeanUtils.setProperty(bean, "byteProperty", new Long((long) 123));
+ assertEquals((byte) 123, ((Byte) bean.get("byteProperty")).byteValue());
+ BeanUtils.setProperty(bean, "byteProperty", new Short((short) 123));
+ assertEquals((byte) 123, ((Byte) bean.get("byteProperty")).byteValue());
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on double.
+ */
+ public void testCopyPropertyDouble() throws Exception {
+
+ BeanUtils.setProperty(bean, "doubleProperty", new Byte((byte) 123));
+ assertEquals((double) 123, ((Double) bean.get("doubleProperty")).doubleValue(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Double((double) 123));
+ assertEquals((double) 123, ((Double) bean.get("doubleProperty")).doubleValue(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Float((float) 123));
+ assertEquals((double) 123, ((Double) bean.get("doubleProperty")).doubleValue(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Integer((int) 123));
+ assertEquals((double) 123, ((Double) bean.get("doubleProperty")).doubleValue(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Long((long) 123));
+ assertEquals((double) 123, ((Double) bean.get("doubleProperty")).doubleValue(), 0.005);
+ BeanUtils.setProperty(bean, "doubleProperty", new Short((short) 123));
+ assertEquals((double) 123, ((Double) bean.get("doubleProperty")).doubleValue(), 0.005);
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on float.
+ */
+ public void testCopyPropertyFloat() throws Exception {
+
+ BeanUtils.setProperty(bean, "floatProperty", new Byte((byte) 123));
+ assertEquals((float) 123, ((Float) bean.get("floatProperty")).floatValue(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Double((double) 123));
+ assertEquals((float) 123, ((Float) bean.get("floatProperty")).floatValue(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Float((float) 123));
+ assertEquals((float) 123, ((Float) bean.get("floatProperty")).floatValue(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Integer((int) 123));
+ assertEquals((float) 123, ((Float) bean.get("floatProperty")).floatValue(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Long((long) 123));
+ assertEquals((float) 123, ((Float) bean.get("floatProperty")).floatValue(), 0.005);
+ BeanUtils.setProperty(bean, "floatProperty", new Short((short) 123));
+ assertEquals((float) 123, ((Float) bean.get("floatProperty")).floatValue(), 0.005);
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on int.
+ */
+ public void testCopyPropertyInteger() throws Exception {
+
+ BeanUtils.setProperty(bean, "longProperty", new Byte((byte) 123));
+ assertEquals((int) 123, ((Integer) bean.get("intProperty")).intValue());
+/*
+ BeanUtils.setProperty(bean, "longProperty", new Double((double) 123));
+ assertEquals((int) 123, ((Integer) bean.get("intProperty")).intValue());
+ BeanUtils.setProperty(bean, "longProperty", new Float((float) 123));
+ assertEquals((int) 123, ((Integer) bean.get("intProperty")).intValue());
+*/
+ BeanUtils.setProperty(bean, "longProperty", new Integer((int) 123));
+ assertEquals((int) 123, ((Integer) bean.get("intProperty")).intValue());
+ BeanUtils.setProperty(bean, "longProperty", new Long((long) 123));
+ assertEquals((int) 123, ((Integer) bean.get("intProperty")).intValue());
+ BeanUtils.setProperty(bean, "longProperty", new Short((short) 123));
+ assertEquals((int) 123, ((Integer) bean.get("intProperty")).intValue());
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on long.
+ */
+ public void testCopyPropertyLong() throws Exception {
+
+ BeanUtils.setProperty(bean, "longProperty", new Byte((byte) 123));
+ assertEquals((long) 123, ((Long) bean.get("longProperty")).longValue());
+/*
+ BeanUtils.setProperty(bean, "longProperty", new Double((double) 123));
+ assertEquals((long) 123, ((Long) bean.get("longProperty")).longValue());
+ BeanUtils.setProperty(bean, "longProperty", new Float((float) 123));
+ assertEquals((long) 123, ((Long) bean.get("longProperty")).longValue());
+*/
+ BeanUtils.setProperty(bean, "longProperty", new Integer((int) 123));
+ assertEquals((long) 123, ((Long) bean.get("longProperty")).longValue());
+ BeanUtils.setProperty(bean, "longProperty", new Long((long) 123));
+ assertEquals((long) 123, ((Long) bean.get("longProperty")).longValue());
+ BeanUtils.setProperty(bean, "longProperty", new Short((short) 123));
+ assertEquals((long) 123, ((Long) bean.get("longProperty")).longValue());
+
+ }
+
+
+ /**
+ * Test copying a null property value.
+ */
+ public void testCopyPropertyNull() throws Exception {
+
+ bean.set("nullProperty", "non-null value");
+ BeanUtils.copyProperty(bean, "nullProperty", null);
+ assertNull("nullProperty is null", bean.get("nullProperty"));
+
+ }
+
+
+ /**
+ * Test narrowing and widening conversions on short.
+ */
+ public void testCopyPropertyShort() throws Exception {
+
+ BeanUtils.setProperty(bean, "shortProperty", new Byte((byte) 123));
+ assertEquals((short) 123, ((Short) bean.get("shortProperty")).shortValue());
+/*
+ BeanUtils.setProperty(bean, "shortProperty", new Double((double) 123));
+ assertEquals((short) 123, ((Short) bean.get("shortProperty")).shortValue());
+ BeanUtils.setProperty(bean, "shortProperty", new Float((float) 123));
+ assertEquals((short) 123, ((Short) bean.get("shortProperty")).shortValue());
+*/
+ BeanUtils.setProperty(bean, "shortProperty", new Integer((int) 123));
+ assertEquals((short) 123, ((Short) bean.get("shortProperty")).shortValue());
+ BeanUtils.setProperty(bean, "shortProperty", new Long((long) 123));
+ assertEquals((short) 123, ((Short) bean.get("shortProperty")).shortValue());
+ BeanUtils.setProperty(bean, "shortProperty", new Short((short) 123));
+ assertEquals((short) 123, ((Short) bean.get("shortProperty")).shortValue());
+
+ }
+
+
+ /**
+ * Test copying a property using a nested indexed array expression,
+ * with and without conversions.
+ */
+ public void testCopyPropertyNestedIndexedArray() throws Exception {
+
+ int origArray[] = { 0, 10, 20, 30, 40};
+ int intArray[] = { 0, 0, 0 };
+ ((TestBean) bean.get("nested")).setIntArray(intArray);
+ int intChanged[] = { 0, 0, 0 };
+
+ // No conversion required
+ BeanUtils.copyProperty(bean, "nested.intArray[1]", new Integer(1));
+ checkIntArray((int[]) bean.get("intArray"), origArray);
+ intChanged[1] = 1;
+ checkIntArray(((TestBean) bean.get("nested")).getIntArray(),
+ intChanged);
+
+ // Widening conversion required
+ BeanUtils.copyProperty(bean, "nested.intArray[1]", new Byte((byte) 2));
+ checkIntArray((int[]) bean.get("intArray"), origArray);
+ intChanged[1] = 2;
+ checkIntArray(((TestBean) bean.get("nested")).getIntArray(),
+ intChanged);
+
+ // Narrowing conversion required
+ BeanUtils.copyProperty(bean, "nested.intArray[1]", new Long((long) 3));
+ checkIntArray((int[]) bean.get("intArray"), origArray);
+ intChanged[1] = 3;
+ checkIntArray(((TestBean) bean.get("nested")).getIntArray(),
+ intChanged);
+
+ // String conversion required
+ BeanUtils.copyProperty(bean, "nested.intArray[1]", "4");
+ checkIntArray((int[]) bean.get("intArray"), origArray);
+ intChanged[1] = 4;
+ checkIntArray(((TestBean) bean.get("nested")).getIntArray(),
+ intChanged);
+
+ }
+
+
+ /**
+ * Test copying a property using a nested mapped map property.
+ */
+ public void testCopyPropertyNestedMappedMap() throws Exception {
+
+ Map origMap = new HashMap();
+ origMap.put("First Key", "First Value");
+ origMap.put("Second Key", "Second Value");
+ Map changedMap = new HashMap();
+ changedMap.put("First Key", "First Value");
+ changedMap.put("Second Key", "Second Value");
+
+ // No conversion required
+ BeanUtils.copyProperty(bean, "nested.mapProperty(Second Key)",
+ "New Second Value");
+ checkMap((Map) bean.get("mapProperty"), origMap);
+ changedMap.put("Second Key", "New Second Value");
+ checkMap(((TestBean) bean.get("nested")).getMapProperty(), changedMap);
+
+ }
+
+
+ /**
+ * Test copying a property using a nested simple expression, with and
+ * without conversions.
+ */
+ public void testCopyPropertyNestedSimple() throws Exception {
+
+ bean.set("intProperty", new Integer(0));
+ nested.setIntProperty(0);
+
+ // No conversion required
+ BeanUtils.copyProperty(bean, "nested.intProperty", new Integer(1));
+ assertEquals(0, ((Integer) bean.get("intProperty")).intValue());
+ assertEquals(1, nested.getIntProperty());
+
+ // Widening conversion required
+ BeanUtils.copyProperty(bean, "nested.intProperty", new Byte((byte) 2));
+ assertEquals(0, ((Integer) bean.get("intProperty")).intValue());
+ assertEquals(2, nested.getIntProperty());
+
+ // Narrowing conversion required
+ BeanUtils.copyProperty(bean, "nested.intProperty", new Long((long) 3));
+ assertEquals(0, ((Integer) bean.get("intProperty")).intValue());
+ assertEquals(3, nested.getIntProperty());
+
+ // String conversion required
+ BeanUtils.copyProperty(bean, "nested.intProperty", "4");
+ assertEquals(0, ((Integer) bean.get("intProperty")).intValue());
+ assertEquals(4, nested.getIntProperty());
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ // Ensure that the nested intArray matches the specified values
+ protected void checkIntArray(int actual[], int expected[]) {
+ assertNotNull("actual array not null", actual);
+ assertEquals("actual array length", expected.length, actual.length);
+ for (int i = 0; i < actual.length; i++) {
+ assertEquals("actual array value[" + i + "]",
+ expected[i], actual[i]);
+ }
+ }
+
+
+ // Ensure that the actual Map matches the expected Map
+ protected void checkMap(Map actual, Map expected) {
+ assertNotNull("actual map not null", actual);
+ assertEquals("actual map size", expected.size(), actual.size());
+ Iterator keys = expected.keySet().iterator();
+ while (keys.hasNext()) {
+ Object key = keys.next();
+ assertEquals("actual map value(" + key + ")",
+ expected.get(key), actual.get(key));
+ }
+ }
+
+
+ /**
+ * Create and return a <code>DynaClass</code> instance for our test
+ * <code>DynaBean</code>.
+ */
+ protected static DynaClass createDynaClass() {
+
+ int intArray[] = new int[0];
+ String stringArray[] = new String[0];
+
+ DynaClass dynaClass = new BasicDynaClass
+ ("TestDynaClass", null,
+ new DynaProperty[]{
+ new DynaProperty("booleanProperty", Boolean.TYPE),
+ new DynaProperty("booleanSecond", Boolean.TYPE),
+ new DynaProperty("byteProperty", Byte.TYPE),
+ new DynaProperty("doubleProperty", Double.TYPE),
+ new DynaProperty("dupProperty", stringArray.getClass()),
+ new DynaProperty("floatProperty", Float.TYPE),
+ new DynaProperty("intArray", intArray.getClass()),
+ new DynaProperty("intIndexed", intArray.getClass()),
+ new DynaProperty("intProperty", Integer.TYPE),
+ new DynaProperty("listIndexed", List.class),
+ new DynaProperty("longProperty", Long.TYPE),
+ new DynaProperty("mapProperty", Map.class),
+ new DynaProperty("mappedProperty", Map.class),
+ new DynaProperty("mappedIntProperty", Map.class),
+ new DynaProperty("nested", TestBean.class),
+ new DynaProperty("nullProperty", String.class),
+ new DynaProperty("shortProperty", Short.TYPE),
+ new DynaProperty("stringArray", stringArray.getClass()),
+ new DynaProperty("stringIndexed", stringArray.getClass()),
+ new DynaProperty("stringProperty", String.class),
+ });
+ return (dynaClass);
+
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/DynaPropertyUtilsTestCase.java b/trunk/src/test/org/apache/commons/beanutils/DynaPropertyUtilsTestCase.java
new file mode 100644
index 0000000..f9c999f
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/DynaPropertyUtilsTestCase.java
@@ -0,0 +1,2648 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * Test accessing DynaBeans transparently via PropertyUtils.
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.12 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class DynaPropertyUtilsTestCase extends TestCase {
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The basic test bean for each test.
+ */
+ protected DynaBean bean = null;
+
+
+ /**
+ * The set of properties that should be described.
+ */
+ protected String describes[] =
+ { "booleanProperty",
+ "booleanSecond",
+ "doubleProperty",
+ "floatProperty",
+ "intArray",
+ "intIndexed",
+ "intProperty",
+ "listIndexed",
+ "longProperty",
+ "mappedObjects",
+ "mappedProperty",
+ "mappedIntProperty",
+ "nested",
+ "nullProperty",
+ // "readOnlyProperty",
+ "shortProperty",
+ "stringArray",
+ "stringIndexed",
+ "stringProperty"
+ };
+
+
+ /**
+ * The nested bean pointed at by the "nested" property.
+ */
+ protected TestBean nested = null;
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public DynaPropertyUtilsTestCase(String name) {
+
+ super(name);
+
+ }
+
+
+ // --------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+
+ // Instantiate a new DynaBean instance
+ DynaClass dynaClass = createDynaClass();
+ bean = dynaClass.newInstance();
+
+ // Initialize the DynaBean's property values (like TestBean)
+ bean.set("booleanProperty", new Boolean(true));
+ bean.set("booleanSecond", new Boolean(true));
+ bean.set("doubleProperty", new Double(321.0));
+ bean.set("floatProperty", new Float((float) 123.0));
+ int intArray[] = { 0, 10, 20, 30, 40 };
+ bean.set("intArray", intArray);
+ int intIndexed[] = { 0, 10, 20, 30, 40 };
+ bean.set("intIndexed", intIndexed);
+ bean.set("intProperty", new Integer(123));
+ List listIndexed = new ArrayList();
+ listIndexed.add("String 0");
+ listIndexed.add("String 1");
+ listIndexed.add("String 2");
+ listIndexed.add("String 3");
+ listIndexed.add("String 4");
+ bean.set("listIndexed", listIndexed);
+ bean.set("longProperty", new Long((long) 321));
+ HashMap mapProperty = new HashMap();
+ mapProperty.put("First Key", "First Value");
+ mapProperty.put("Second Key", "Second Value");
+ bean.set("mapProperty", mapProperty);
+ HashMap mappedObjects = new HashMap();
+ mappedObjects.put("First Key", "First Value");
+ mappedObjects.put("Second Key", "Second Value");
+ bean.set("mappedObjects", mappedObjects);
+ HashMap mappedProperty = new HashMap();
+ mappedProperty.put("First Key", "First Value");
+ mappedProperty.put("Second Key", "Second Value");
+ bean.set("mappedProperty", mappedProperty);
+ HashMap mappedIntProperty = new HashMap();
+ mappedIntProperty.put("One", new Integer(1));
+ mappedIntProperty.put("Two", new Integer(2));
+ bean.set("mappedIntProperty", mappedIntProperty);
+ nested = new TestBean();
+ bean.set("nested", nested);
+ // Property "nullProperty" is not initialized, so it should return null
+ bean.set("shortProperty", new Short((short) 987));
+ String stringArray[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+ bean.set("stringArray", stringArray);
+ String stringIndexed[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+ bean.set("stringIndexed", stringIndexed);
+ bean.set("stringProperty", "This is a string");
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+
+ return (new TestSuite(DynaPropertyUtilsTestCase.class));
+
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+
+ bean = null;
+ nested = null;
+
+ }
+
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ /**
+ * Test copyProperties() when the origin is a a <code>Map</code>.
+ */
+ public void testCopyPropertiesMap() {
+
+ Map map = new HashMap();
+ map.put("booleanProperty", Boolean.FALSE);
+ map.put("doubleProperty", new Double(333.0));
+ map.put("dupProperty", new String[] { "New 0", "New 1", "New 2" });
+ map.put("floatProperty", new Float((float) 222.0));
+ map.put("intArray", new int[] { 0, 100, 200 });
+ map.put("intProperty", new Integer(111));
+ map.put("longProperty", new Long(444));
+ map.put("shortProperty", new Short((short) 555));
+ map.put("stringProperty", "New String Property");
+
+ try {
+ PropertyUtils.copyProperties(bean, map);
+ } catch (Throwable t) {
+ fail("Threw " + t.toString());
+ }
+
+ // Scalar properties
+ assertEquals("booleanProperty", false,
+ ((Boolean) bean.get("booleanProperty")).booleanValue());
+ assertEquals("doubleProperty", 333.0,
+ ((Double) bean.get("doubleProperty")).doubleValue(),
+ 0.005);
+ assertEquals("floatProperty", (float) 222.0,
+ ((Float) bean.get("floatProperty")).floatValue(),
+ (float) 0.005);
+ assertEquals("intProperty", 111,
+ ((Integer) bean.get("intProperty")).intValue());
+ assertEquals("longProperty", (long) 444,
+ ((Long) bean.get("longProperty")).longValue());
+ assertEquals("shortProperty", (short) 555,
+ ((Short) bean.get("shortProperty")).shortValue());
+ assertEquals("stringProperty", "New String Property",
+ (String) bean.get("stringProperty"));
+
+ // Indexed Properties
+ String dupProperty[] = (String[]) bean.get("dupProperty");
+ assertNotNull("dupProperty present", dupProperty);
+ assertEquals("dupProperty length", 3, dupProperty.length);
+ assertEquals("dupProperty[0]", "New 0", dupProperty[0]);
+ assertEquals("dupProperty[1]", "New 1", dupProperty[1]);
+ assertEquals("dupProperty[2]", "New 2", dupProperty[2]);
+ int intArray[] = (int[]) bean.get("intArray");
+ assertNotNull("intArray present", intArray);
+ assertEquals("intArray length", 3, intArray.length);
+ assertEquals("intArray[0]", 0, intArray[0]);
+ assertEquals("intArray[1]", 100, intArray[1]);
+ assertEquals("intArray[2]", 200, intArray[2]);
+
+ }
+
+
+ /**
+ * Test the describe() method.
+ */
+ public void testDescribe() {
+
+ Map map = null;
+ try {
+ map = PropertyUtils.describe(bean);
+ } catch (Exception e) {
+ fail("Threw exception " + e);
+ }
+
+ // Verify existence of all the properties that should be present
+ for (int i = 0; i < describes.length; i++) {
+ assertTrue("Property '" + describes[i] + "' is present",
+ map.containsKey(describes[i]));
+ }
+ assertTrue("Property 'writeOnlyProperty' is not present",
+ !map.containsKey("writeOnlyProperty"));
+
+ // Verify the values of scalar properties
+ assertEquals("Value of 'booleanProperty'",
+ Boolean.TRUE,
+ (Boolean) map.get("booleanProperty"));
+ assertEquals("Value of 'doubleProperty'",
+ new Double(321.0),
+ (Double) map.get("doubleProperty"));
+ assertEquals("Value of 'floatProperty'",
+ new Float((float) 123.0),
+ (Float) map.get("floatProperty"));
+ assertEquals("Value of 'intProperty'",
+ new Integer(123),
+ (Integer) map.get("intProperty"));
+ assertEquals("Value of 'longProperty'",
+ new Long(321),
+ (Long) map.get("longProperty"));
+ assertEquals("Value of 'shortProperty'",
+ new Short((short) 987),
+ (Short) map.get("shortProperty"));
+ assertEquals("Value of 'stringProperty'",
+ "This is a string",
+ (String) map.get("stringProperty"));
+
+ }
+
+
+ /**
+ * Corner cases on getIndexedProperty invalid arguments.
+ */
+ public void testGetIndexedArguments() {
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.getIndexedProperty(null, "intArray", 0);
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, null, 0);
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.getIndexedProperty(null,
+ "intArray[0]");
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, "[0]");
+ fail("Should throw NoSuchMethodException 4");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 4");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, "intArray");
+ fail("Should throw IllegalArgumentException 5");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 5");
+ }
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.getIndexedProperty(null, "intIndexed", 0);
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, null, 0);
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.getIndexedProperty(null,
+ "intIndexed[0]");
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, "[0]");
+ fail("Should throw NoSuchMethodException 4");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 4");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, "intIndexed");
+ fail("Should throw IllegalArgumentException 5");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 5");
+ }
+
+ }
+
+
+ /**
+ * Positive and negative tests on getIndexedProperty valid arguments.
+ */
+ public void testGetIndexedValues() {
+
+ Object value = null;
+
+ // Use explicit key argument
+
+ for (int i = 0; i < 5; i++) {
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "intArray", i);
+ assertNotNull("intArray returned value " + i, value);
+ assertTrue("intArray returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intArray returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intArray " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "intIndexed", i);
+ assertNotNull("intIndexed returned value " + i, value);
+ assertTrue("intIndexed returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intIndexed returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "listIndexed", i);
+ assertNotNull("listIndexed returned value " + i, value);
+ assertTrue("list returned String " + i,
+ value instanceof String);
+ assertEquals("listIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("listIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "stringArray", i);
+ assertNotNull("stringArray returned value " + i, value);
+ assertTrue("stringArray returned String " + i,
+ value instanceof String);
+ assertEquals("stringArray returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringArray " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "stringIndexed", i);
+ assertNotNull("stringIndexed returned value " + i, value);
+ assertTrue("stringIndexed returned String " + i,
+ value instanceof String);
+ assertEquals("stringIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringIndexed " + i + " threw " + t);
+ }
+
+ }
+
+ // Use key expression
+
+ for (int i = 0; i < 5; i++) {
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray[" + i + "]");
+ assertNotNull("intArray returned value " + i, value);
+ assertTrue("intArray returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intArray returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intArray " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed[" + i + "]");
+ assertNotNull("intIndexed returned value " + i, value);
+ assertTrue("intIndexed returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intIndexed returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed[" + i + "]");
+ assertNotNull("listIndexed returned value " + i, value);
+ assertTrue("listIndexed returned String " + i,
+ value instanceof String);
+ assertEquals("listIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("listIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray[" + i + "]");
+ assertNotNull("stringArray returned value " + i, value);
+ assertTrue("stringArray returned String " + i,
+ value instanceof String);
+ assertEquals("stringArray returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringArray " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringIndexed[" + i + "]");
+ assertNotNull("stringIndexed returned value " + i, value);
+ assertTrue("stringIndexed returned String " + i,
+ value instanceof String);
+ assertEquals("stringIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringIndexed " + i + " threw " + t);
+ }
+
+ }
+
+ // Index out of bounds tests
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray", -1);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray", 5);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed", -1);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed", 5);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed", -1);
+ fail("Should have thrown IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed", 5);
+ fail("Should have thrown IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray", -1);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray", 5);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringIndexed", -1);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringIndexed", 5);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getMappedProperty invalid arguments.
+ */
+ public void testGetMappedArguments() {
+
+ // Use explicit key argument
+
+ try {
+ PropertyUtils.getMappedProperty(null, "mappedProperty",
+ "First Key");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getMappedProperty(bean, null, "First Key");
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ try {
+ PropertyUtils.getMappedProperty(bean, "mappedProperty", null);
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ // Use key expression
+
+ try {
+ PropertyUtils.getMappedProperty(null,
+ "mappedProperty(First Key)");
+ fail("Should throw IllegalArgumentException 4");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 4");
+ }
+
+ try {
+ PropertyUtils.getMappedProperty(bean, "(Second Key)");
+ fail("Should throw IllegalArgumentException 5");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 5");
+ }
+
+ try {
+ PropertyUtils.getMappedProperty(bean, "mappedProperty");
+ fail("Should throw IllegalArgumentException 6");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 6");
+ }
+
+ }
+
+
+ /**
+ * Test getting mapped values with periods in the key.
+ */
+ public void testGetMappedPeriods() {
+
+ bean.set("mappedProperty", "key.with.a.dot", "Special Value");
+ assertEquals("Can retrieve directly",
+ "Special Value",
+ (String) bean.get("mappedProperty", "key.with.a.dot"));
+ try {
+ assertEquals("Can retrieve via getMappedProperty",
+ "Special Value",
+ PropertyUtils.getMappedProperty
+ (bean, "mappedProperty", "key.with.a.dot"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+ try {
+ assertEquals("Can retrieve via getNestedProperty",
+ "Special Value",
+ PropertyUtils.getNestedProperty
+ (bean, "mappedProperty(key.with.a.dot)"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+
+ bean.set("mappedObjects", "nested.property", new TestBean());
+ assertNotNull("Can retrieve directly",
+ bean.get("mappedObjects", "nested.property"));
+ try {
+ assertEquals("Can retrieve nested",
+ "This is a string",
+ PropertyUtils.getNestedProperty
+ (bean,
+ "mappedObjects(nested.property).stringProperty"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Test getting mapped values with slashes in the key. This is different
+ * from periods because slashes are not syntactically significant.
+ */
+ public void testGetMappedSlashes() {
+
+ bean.set("mappedProperty", "key/with/a/slash", "Special Value");
+ assertEquals("Can retrieve directly",
+ "Special Value",
+ bean.get("mappedProperty", "key/with/a/slash"));
+ try {
+ assertEquals("Can retrieve via getMappedProperty",
+ "Special Value",
+ PropertyUtils.getMappedProperty
+ (bean, "mappedProperty", "key/with/a/slash"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+ try {
+ assertEquals("Can retrieve via getNestedProperty",
+ "Special Value",
+ PropertyUtils.getNestedProperty
+ (bean, "mappedProperty(key/with/a/slash)"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+
+ bean.set("mappedObjects", "nested/property", new TestBean());
+ assertNotNull("Can retrieve directly",
+ bean.get("mappedObjects", "nested/property"));
+ try {
+ assertEquals("Can retrieve nested",
+ "This is a string",
+ PropertyUtils.getNestedProperty
+ (bean,
+ "mappedObjects(nested/property).stringProperty"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Positive and negative tests on getMappedProperty valid arguments.
+ */
+ public void testGetMappedValues() {
+
+ Object value = null;
+
+ // Use explicit key argument
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "First Key");
+ assertEquals("Can find first value", "First Value", value);
+ } catch (Throwable t) {
+ fail("Finding first value threw " + t);
+ }
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "Second Key");
+ assertEquals("Can find second value", "Second Value", value);
+ } catch (Throwable t) {
+ fail("Finding second value threw " + t);
+ }
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "Third Key");
+ assertNull("Can not find third value", value);
+ } catch (Throwable t) {
+ fail("Finding third value threw " + t);
+ }
+
+ // Use key expression with parentheses
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(First Key)");
+ assertEquals("Can find first value", "First Value", value);
+ } catch (Throwable t) {
+ fail("Finding first value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(Second Key)");
+ assertEquals("Can find second value", "Second Value", value);
+ } catch (Throwable t) {
+ fail("Finding second value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(Third Key)");
+ assertNull("Can not find third value", value);
+ } catch (Throwable t) {
+ fail("Finding third value threw " + t);
+ }
+
+ // Use key expression with dotted syntax
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.First Key");
+ assertEquals("Can find first value", "First Value", value);
+ } catch (Throwable t) {
+ fail("Finding first value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.Second Key");
+ assertEquals("Can find second value", "Second Value", value);
+ } catch (Throwable t) {
+ fail("Finding second value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.Third Key");
+ assertNull("Can not find third value", value);
+ } catch (Throwable t) {
+ fail("Finding third value threw " + t);
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getNestedProperty invalid arguments.
+ */
+ public void testGetNestedArguments() {
+
+ try {
+ PropertyUtils.getNestedProperty(null, "stringProperty");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getNestedProperty(bean, null);
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a boolean property.
+ */
+ public void testGetNestedBoolean() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.booleanProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Boolean));
+ TestBean nested = (TestBean) bean.get("nested");
+ assertTrue("Got correct value",
+ ((Boolean) value).booleanValue() ==
+ nested.getBooleanProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a double property.
+ */
+ public void testGetNestedDouble() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.doubleProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Double));
+ TestBean nested = (TestBean) bean.get("nested");
+ assertEquals("Got correct value",
+ ((Double) value).doubleValue(),
+ nested.getDoubleProperty(),
+ 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a float property.
+ */
+ public void testGetNestedFloat() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.floatProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Float));
+ TestBean nested = (TestBean) bean.get("nested");
+ assertEquals("Got correct value",
+ ((Float) value).floatValue(),
+ nested.getFloatProperty(),
+ (float) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on an int property.
+ */
+ public void testGetNestedInt() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.intProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Integer));
+ TestBean nested = (TestBean) bean.get("nested");
+ assertEquals("Got correct value",
+ ((Integer) value).intValue(),
+ nested.getIntProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a long property.
+ */
+ public void testGetNestedLong() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.longProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Long));
+ TestBean nested = (TestBean) bean.get("nested");
+ assertEquals("Got correct value",
+ ((Long) value).longValue(),
+ nested.getLongProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a read-only String property.
+ */
+ public void testGetNestedReadOnly() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.readOnlyProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof String));
+ TestBean nested = (TestBean) bean.get("nested");
+ assertEquals("Got correct value",
+ (String) value,
+ nested.getReadOnlyProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a short property.
+ */
+ public void testGetNestedShort() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.shortProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Short));
+ TestBean nested = (TestBean) bean.get("nested");
+ assertEquals("Got correct value",
+ ((Short) value).shortValue(),
+ nested.getShortProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a String property.
+ */
+ public void testGetNestedString() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.stringProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof String));
+ TestBean nested = (TestBean) bean.get("nested");
+ assertEquals("Got correct value",
+ ((String) value),
+ nested.getStringProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test getNestedProperty on an unknown property.
+ */
+ public void testGetNestedUnknown() {
+
+ try {
+ PropertyUtils.getNestedProperty(bean, "nested.unknown");
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getSimpleProperty invalid arguments.
+ */
+ public void testGetSimpleArguments() {
+
+ try {
+ PropertyUtils.getSimpleProperty(null, "stringProperty");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getSimpleProperty(bean, null);
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a boolean property.
+ */
+ public void testGetSimpleBoolean() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "booleanProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Boolean));
+ assertTrue("Got correct value",
+ ((Boolean) value).booleanValue() == true);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a double property.
+ */
+ public void testGetSimpleDouble() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "doubleProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Double));
+ assertEquals("Got correct value",
+ ((Double) value).doubleValue(),
+ (double) 321.0,
+ (double) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a float property.
+ */
+ public void testGetSimpleFloat() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "floatProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Float));
+ assertEquals("Got correct value",
+ ((Float) value).floatValue(),
+ (float) 123.0,
+ (float) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test getSimpleProperty on an indexed property.
+ */
+ public void testGetSimpleIndexed() {
+
+ Object value = null;
+ try {
+ value = PropertyUtils.getSimpleProperty(bean,
+ "intIndexed[0]");
+ fail("Should have thrown IllegalArgumentException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ ; // Correct result for this test
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on an int property.
+ */
+ public void testGetSimpleInt() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "intProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Integer));
+ assertEquals("Got correct value",
+ ((Integer) value).intValue(),
+ 123);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a long property.
+ */
+ public void testGetSimpleLong() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "longProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Long));
+ assertEquals("Got correct value",
+ ((Long) value).longValue(),
+ (long) 321);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test getSimpleProperty on a nested property.
+ */
+ public void testGetSimpleNested() {
+
+ Object value = null;
+ try {
+ value = PropertyUtils.getSimpleProperty(bean,
+ "nested.stringProperty");
+ fail("Should have thrown IllegaArgumentException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ ; // Correct result for this test
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a short property.
+ */
+ public void testGetSimpleShort() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "shortProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Short));
+ assertEquals("Got correct value",
+ ((Short) value).shortValue(),
+ (short) 987);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a String property.
+ */
+ public void testGetSimpleString() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "stringProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof String));
+ assertEquals("Got correct value",
+ (String) value,
+ "This is a string");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test getSimpleProperty on an unknown property.
+ */
+ public void testGetSimpleUnknown() {
+
+ try {
+ PropertyUtils.getSimpleProperty(bean, "unknown");
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Corner cases on setIndexedProperty invalid arguments.
+ */
+ public void testSetIndexedArguments() {
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.setIndexedProperty(null, "intArray", 0,
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, null, 0,
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.setIndexedProperty(null,
+ "intArray[0]",
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, "[0]",
+ new Integer(1));
+ fail("Should throw NoSuchMethodException 4");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 4");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, "intArray",
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 5");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 5");
+ }
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.setIndexedProperty(null, "intIndexed", 0,
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, null, 0,
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.setIndexedProperty(null,
+ "intIndexed[0]",
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, "[0]",
+ new Integer(1));
+ fail("Should throw NoSuchMethodException 4");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 4");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, "intIndexed",
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 5");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 5");
+ }
+
+ }
+
+
+ /**
+ * Positive and negative tests on setIndexedProperty valid arguments.
+ */
+ public void testSetIndexedValues() {
+
+ Object value = null;
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intArray", 0,
+ new Integer(1));
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray", 0);
+ assertNotNull("Returned new value 0", value);
+ assertTrue("Returned Integer new value 0",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 0", 1,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intIndexed", 1,
+ new Integer(11));
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed", 1);
+ assertNotNull("Returned new value 1", value);
+ assertTrue("Returned Integer new value 1",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 1", 11,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "listIndexed", 2,
+ "New Value 2");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed", 2);
+ assertNotNull("Returned new value 2", value);
+ assertTrue("Returned String new value 2",
+ value instanceof String);
+ assertEquals("Returned correct new value 2", "New Value 2",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray", 2,
+ "New Value 2");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray", 2);
+ assertNotNull("Returned new value 2", value);
+ assertTrue("Returned String new value 2",
+ value instanceof String);
+ assertEquals("Returned correct new value 2", "New Value 2",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray", 3,
+ "New Value 3");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray", 3);
+ assertNotNull("Returned new value 3", value);
+ assertTrue("Returned String new value 3",
+ value instanceof String);
+ assertEquals("Returned correct new value 3", "New Value 3",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intArray[4]",
+ new Integer(1));
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray[4]");
+ assertNotNull("Returned new value 4", value);
+ assertTrue("Returned Integer new value 4",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 4", 1,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intIndexed[3]",
+ new Integer(11));
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed[3]");
+ assertNotNull("Returned new value 5", value);
+ assertTrue("Returned Integer new value 5",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 5", 11,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "listIndexed[1]",
+ "New Value 2");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed[1]");
+ assertNotNull("Returned new value 6", value);
+ assertTrue("Returned String new value 6",
+ value instanceof String);
+ assertEquals("Returned correct new value 6", "New Value 2",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray[1]",
+ "New Value 2");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray[2]");
+ assertNotNull("Returned new value 6", value);
+ assertTrue("Returned String new value 6",
+ value instanceof String);
+ assertEquals("Returned correct new value 6", "New Value 2",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray[0]",
+ "New Value 3");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray[0]");
+ assertNotNull("Returned new value 7", value);
+ assertTrue("Returned String new value 7",
+ value instanceof String);
+ assertEquals("Returned correct new value 7", "New Value 3",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ // Index out of bounds tests
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intArray", -1,
+ new Integer(0));
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intArray", 5,
+ new Integer(0));
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intIndexed", -1,
+ new Integer(0));
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intIndexed", 5,
+ new Integer(0));
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "listIndexed", 5,
+ "New String");
+ fail("Should have thrown IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "listIndexed", -1,
+ "New String");
+ fail("Should have thrown IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray", -1,
+ "New String");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray", 5,
+ "New String");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringIndexed", -1,
+ "New String");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringIndexed", 5,
+ "New String");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getMappedProperty invalid arguments.
+ */
+ public void testSetMappedArguments() {
+
+ // Use explicit key argument
+
+ try {
+ PropertyUtils.setMappedProperty(null, "mappedProperty",
+ "First Key", "First Value");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, null, "First Key",
+ "First Value");
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, "mappedProperty", null,
+ "First Value");
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ // Use key expression
+
+ try {
+ PropertyUtils.setMappedProperty(null,
+ "mappedProperty(First Key)",
+ "First Value");
+ fail("Should throw IllegalArgumentException 4");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 4");
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, "(Second Key)",
+ "Second Value");
+ fail("Should throw IllegalArgumentException 5");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 5");
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, "mappedProperty",
+ "Third Value");
+ fail("Should throw IllegalArgumentException 6");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 6");
+ }
+
+ }
+
+
+ /**
+ * Positive and negative tests on setMappedProperty valid arguments.
+ */
+ public void testSetMappedValues() {
+
+ Object value = null;
+
+ // Use explicit key argument
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "Fourth Key");
+ assertNull("Can not find fourth value", value);
+ } catch (Throwable t) {
+ fail("Finding fourth value threw " + t);
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, "mappedProperty",
+ "Fourth Key", "Fourth Value");
+ } catch (Throwable t) {
+ fail("Setting fourth value threw " + t);
+ }
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "Fourth Key");
+ assertEquals("Can find fourth value", "Fourth Value", value);
+ } catch (Throwable t) {
+ fail("Finding fourth value threw " + t);
+ }
+
+ // Use key expression with parentheses
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(Fifth Key)");
+ assertNull("Can not find fifth value", value);
+ } catch (Throwable t) {
+ fail("Finding fifth value threw " + t);
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean,
+ "mappedProperty(Fifth Key)",
+ "Fifth Value");
+ } catch (Throwable t) {
+ fail("Setting fifth value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(Fifth Key)");
+ assertEquals("Can find fifth value", "Fifth Value", value);
+ } catch (Throwable t) {
+ fail("Finding fifth value threw " + t);
+ }
+
+ // Use key expression with dotted expression
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.Sixth Key");
+ assertNull("Can not find sixth value", value);
+ } catch (Throwable t) {
+ fail("Finding fifth value threw " + t);
+ }
+
+ try {
+ PropertyUtils.setNestedProperty(bean,
+ "mapProperty.Sixth Key",
+ "Sixth Value");
+ } catch (Throwable t) {
+ fail("Setting sixth value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.Sixth Key");
+ assertEquals("Can find sixth value", "Sixth Value", value);
+ } catch (Throwable t) {
+ fail("Finding sixth value threw " + t);
+ }
+
+ }
+
+
+ /**
+ * Corner cases on setNestedProperty invalid arguments.
+ */
+ public void testSetNestedArguments() {
+
+ try {
+ PropertyUtils.setNestedProperty(null, "stringProperty", "");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setNestedProperty(bean, null, "");
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ }
+
+
+ /**
+ * Test setNextedProperty on a boolean property.
+ */
+ public void testSetNestedBoolean() {
+
+ try {
+ boolean oldValue = nested.getBooleanProperty();
+ boolean newValue = !oldValue;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.booleanProperty",
+ new Boolean(newValue));
+ assertTrue("Matched new value",
+ newValue ==
+ nested.getBooleanProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a double property.
+ */
+ public void testSetNestedDouble() {
+
+ try {
+ double oldValue = nested.getDoubleProperty();
+ double newValue = oldValue + 1.0;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.doubleProperty",
+ new Double(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ nested.getDoubleProperty(),
+ 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a float property.
+ */
+ public void testSetNestedFloat() {
+
+ try {
+ float oldValue = nested.getFloatProperty();
+ float newValue = oldValue + (float) 1.0;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.floatProperty",
+ new Float(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ nested.getFloatProperty(),
+ (float) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a int property.
+ */
+ public void testSetNestedInt() {
+
+ try {
+ int oldValue = nested.getIntProperty();
+ int newValue = oldValue + 1;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.intProperty",
+ new Integer(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ nested.getIntProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a long property.
+ */
+ public void testSetNestedLong() {
+
+ try {
+ long oldValue = nested.getLongProperty();
+ long newValue = oldValue + 1;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.longProperty",
+ new Long(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ nested.getLongProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a read-only String property.
+ */
+ public void testSetNestedReadOnly() {
+
+ try {
+ String oldValue = nested.getWriteOnlyPropertyValue();
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setNestedProperty(bean,
+ "nested.readOnlyProperty",
+ newValue);
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a short property.
+ */
+ public void testSetNestedShort() {
+
+ try {
+ short oldValue = nested.getShortProperty();
+ short newValue = oldValue;
+ newValue++;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.shortProperty",
+ new Short(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ nested.getShortProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a String property.
+ */
+ public void testSetNestedString() {
+
+ try {
+ String oldValue = nested.getStringProperty();
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setNestedProperty(bean,
+ "nested.stringProperty",
+ newValue);
+ assertEquals("Matched new value",
+ newValue,
+ nested.getStringProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on an unknown property name.
+ */
+ public void testSetNestedUnknown() {
+
+ try {
+ String newValue = "New String Value";
+ PropertyUtils.setNestedProperty(bean,
+ "nested.unknown",
+ newValue);
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a write-only String property.
+ */
+ public void testSetNestedWriteOnly() {
+
+ try {
+ String oldValue = nested.getWriteOnlyPropertyValue();
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setNestedProperty(bean,
+ "nested.writeOnlyProperty",
+ newValue);
+ assertEquals("Matched new value",
+ newValue,
+ nested.getWriteOnlyPropertyValue());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Corner cases on setSimpleProperty invalid arguments.
+ */
+ public void testSetSimpleArguments() {
+
+ try {
+ PropertyUtils.setSimpleProperty(null, "stringProperty", "");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setSimpleProperty(bean, null, "");
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a boolean property.
+ */
+ public void testSetSimpleBoolean() {
+
+ try {
+ boolean oldValue = ((Boolean) bean.get("booleanProperty")).booleanValue();
+ boolean newValue = !oldValue;
+ PropertyUtils.setSimpleProperty(bean,
+ "booleanProperty",
+ new Boolean(newValue));
+ assertTrue("Matched new value",
+ newValue ==
+ ((Boolean) bean.get("booleanProperty")).booleanValue());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a double property.
+ */
+ public void testSetSimpleDouble() {
+
+ try {
+ double oldValue = ((Double) bean.get("doubleProperty")).doubleValue();
+ double newValue = oldValue + 1.0;
+ PropertyUtils.setSimpleProperty(bean,
+ "doubleProperty",
+ new Double(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Double) bean.get("doubleProperty")).doubleValue(),
+ 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a float property.
+ */
+ public void testSetSimpleFloat() {
+
+ try {
+ float oldValue = ((Float) bean.get("floatProperty")).floatValue();
+ float newValue = oldValue + (float) 1.0;
+ PropertyUtils.setSimpleProperty(bean,
+ "floatProperty",
+ new Float(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Float) bean.get("floatProperty")).floatValue(),
+ (float) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test setSimpleProperty on an indexed property.
+ */
+ public void testSetSimpleIndexed() {
+
+ try {
+ PropertyUtils.setSimpleProperty(bean,
+ "stringIndexed[0]",
+ "New String Value");
+ fail("Should have thrown IllegalArgumentException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ ; // Correct result for this test
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a int property.
+ */
+ public void testSetSimpleInt() {
+
+ try {
+ int oldValue = ((Integer) bean.get("intProperty")).intValue();
+ int newValue = oldValue + 1;
+ PropertyUtils.setSimpleProperty(bean,
+ "intProperty",
+ new Integer(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Integer) bean.get("intProperty")).intValue());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a long property.
+ */
+ public void testSetSimpleLong() {
+
+ try {
+ long oldValue = ((Long) bean.get("longProperty")).longValue();
+ long newValue = oldValue + 1;
+ PropertyUtils.setSimpleProperty(bean,
+ "longProperty",
+ new Long(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Long) bean.get("longProperty")).longValue());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test setSimpleProperty on a nested property.
+ */
+ public void testSetSimpleNested() {
+
+ try {
+ PropertyUtils.setSimpleProperty(bean,
+ "nested.stringProperty",
+ "New String Value");
+ fail("Should have thrown IllegalArgumentException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ ; // Correct result for this test
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a short property.
+ */
+ public void testSetSimpleShort() {
+
+ try {
+ short oldValue = ((Short) bean.get("shortProperty")).shortValue();
+ short newValue = oldValue;
+ newValue++;
+ PropertyUtils.setSimpleProperty(bean,
+ "shortProperty",
+ new Short(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ ((Short) bean.get("shortProperty")).shortValue());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a String property.
+ */
+ public void testSetSimpleString() {
+
+ try {
+ String oldValue = (String) bean.get("stringProperty");
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setSimpleProperty(bean,
+ "stringProperty",
+ newValue);
+ assertEquals("Matched new value",
+ newValue,
+ (String) bean.get("stringProperty"));
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on an unknown property name.
+ */
+ public void testSetSimpleUnknown() {
+
+ try {
+ String newValue = "New String Value";
+ PropertyUtils.setSimpleProperty(bean,
+ "unknown",
+ newValue);
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * Create and return a <code>DynaClass</code> instance for our test
+ * <code>DynaBean</code>.
+ */
+ protected DynaClass createDynaClass() {
+
+ int intArray[] = new int[0];
+ String stringArray[] = new String[0];
+
+ DynaClass dynaClass = new BasicDynaClass
+ ("TestDynaClass", null,
+ new DynaProperty[]{
+ new DynaProperty("booleanProperty", Boolean.TYPE),
+ new DynaProperty("booleanSecond", Boolean.TYPE),
+ new DynaProperty("doubleProperty", Double.TYPE),
+ new DynaProperty("dupProperty", stringArray.getClass()),
+ new DynaProperty("floatProperty", Float.TYPE),
+ new DynaProperty("intArray", intArray.getClass()),
+ new DynaProperty("intIndexed", intArray.getClass()),
+ new DynaProperty("intProperty", Integer.TYPE),
+ new DynaProperty("listIndexed", List.class),
+ new DynaProperty("longProperty", Long.TYPE),
+ new DynaProperty("mapProperty", Map.class),
+ new DynaProperty("mappedObjects", Map.class),
+ new DynaProperty("mappedProperty", Map.class),
+ new DynaProperty("mappedIntProperty", Map.class),
+ new DynaProperty("nested", TestBean.class),
+ new DynaProperty("nullProperty", String.class),
+ new DynaProperty("shortProperty", Short.TYPE),
+ new DynaProperty("stringArray", stringArray.getClass()),
+ new DynaProperty("stringIndexed", stringArray.getClass()),
+ new DynaProperty("stringProperty", String.class),
+ });
+ return (dynaClass);
+
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/DynaResultSetTestCase.java b/trunk/src/test/org/apache/commons/beanutils/DynaResultSetTestCase.java
new file mode 100644
index 0000000..96dd95e
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/DynaResultSetTestCase.java
@@ -0,0 +1,249 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.math.BigDecimal;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Map;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * Test accessing ResultSets via DynaBeans.
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.5 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class DynaResultSetTestCase extends TestCase {
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The mock result set DynaClass to be tested.
+ */
+ protected ResultSetDynaClass dynaClass = null;
+
+
+ /**
+ * Names of the columns for this test. Must match the order they are
+ * defined in {@link TestResultSetMetaData}, and must be all lower case.
+ */
+ protected String columns[] =
+ { "bigdecimalproperty", "booleanproperty",
+ "byteproperty", "dateproperty",
+ "doubleproperty", "floatproperty",
+ "intproperty", "longproperty",
+ "nullproperty", "shortproperty",
+ "stringproperty", "timeproperty",
+ "timestampproperty" };
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public DynaResultSetTestCase(String name) {
+
+ super(name);
+
+ }
+
+
+ // --------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+
+ dynaClass = new ResultSetDynaClass(new TestResultSet());
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+
+ return (new TestSuite(DynaResultSetTestCase.class));
+
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+
+ dynaClass = null;
+
+ }
+
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ public void testGetName() {
+
+ assertEquals("DynaClass name",
+ "org.apache.commons.beanutils.ResultSetDynaClass",
+ dynaClass.getName());
+
+
+ }
+
+
+ public void testGetDynaProperty() {
+
+ // Invalid argument test
+ try {
+ dynaClass.getDynaProperty(null);
+ fail("Did not throw IllegaArgumentException");
+ } catch (IllegalArgumentException e) {
+ ; // Expected result
+ }
+
+ // Negative test
+ DynaProperty dynaProp = dynaClass.getDynaProperty("unknownProperty");
+ assertTrue("unknown property returns null",
+ (dynaProp == null));
+
+ // Positive test
+ dynaProp = dynaClass.getDynaProperty("stringproperty");
+ assertNotNull("string property exists", dynaProp);
+ assertEquals("string property name", "stringproperty",
+ dynaProp.getName());
+ assertEquals("string property class", String.class,
+ dynaProp.getType());
+
+ }
+
+
+ public void testGetDynaProperties() {
+
+ DynaProperty dynaProps[] = dynaClass.getDynaProperties();
+ assertNotNull("dynaProps exists", dynaProps);
+ assertEquals("dynaProps length", columns.length, dynaProps.length);
+ for (int i = 0; i < columns.length; i++) {
+ assertEquals("Property " + columns[i],
+ columns[i], dynaProps[i].getName());
+ }
+
+ }
+
+
+ public void testNewInstance() {
+
+ try {
+ dynaClass.newInstance();
+ fail("Did not throw UnsupportedOperationException()");
+ } catch (UnsupportedOperationException e) {
+ ; // Expected result
+ } catch (Exception e) {
+ fail("Threw exception " + e);
+ }
+
+ }
+
+
+ public void testIteratorCount() {
+
+ Iterator rows = dynaClass.iterator();
+ assertNotNull("iterator exists", rows);
+ int n = 0;
+ while (rows.hasNext()) {
+ rows.next();
+ n++;
+ if (n > 10) {
+ fail("Returned too many rows");
+ }
+ }
+ assertEquals("iterator rows", 5, n);
+
+ }
+
+
+ public void testIteratorResults() {
+
+ // Grab the third row
+ Iterator rows = dynaClass.iterator();
+ rows.next();
+ rows.next();
+ DynaBean row = (DynaBean) rows.next();
+
+ // Invalid argument test
+ try {
+ row.get("unknownProperty");
+ fail("Did not throw IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ ; // Expected result
+ }
+
+ // Verify property values
+
+ Object bigDecimalProperty = row.get("bigdecimalproperty");
+ assertNotNull("bigDecimalProperty exists", bigDecimalProperty);
+ assertTrue("bigDecimalProperty type",
+ bigDecimalProperty instanceof BigDecimal);
+ assertEquals("bigDecimalProperty value",
+ 123.45,
+ ((BigDecimal) bigDecimalProperty).doubleValue(),
+ 0.005);
+
+ Object intProperty = row.get("intproperty");
+ assertNotNull("intProperty exists", intProperty);
+ assertTrue("intProperty type",
+ intProperty instanceof Integer);
+ assertEquals("intProperty value",
+ 103,
+ ((Integer) intProperty).intValue());
+
+ Object nullProperty = row.get("nullproperty");
+ assertNull("nullProperty null", nullProperty);
+
+ Object stringProperty = row.get("stringproperty");
+ assertNotNull("stringProperty exists", stringProperty);
+ assertTrue("stringProperty type",
+ stringProperty instanceof String);
+ assertEquals("stringProperty value",
+ "This is a string",
+ (String) stringProperty);
+
+
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/DynaRowSetTestCase.java b/trunk/src/test/org/apache/commons/beanutils/DynaRowSetTestCase.java
new file mode 100644
index 0000000..cada1b6
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/DynaRowSetTestCase.java
@@ -0,0 +1,247 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.math.BigDecimal;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Map;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * Test accessing RowSets via DynaBeans.
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.5 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class DynaRowSetTestCase extends TestCase {
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The mock result set DynaClass to be tested.
+ */
+ protected RowSetDynaClass dynaClass = null;
+
+
+ /**
+ * Names of the columns for this test. Must match the order they are
+ * defined in {@link TestResultSetMetaData}, and must be all lower case.
+ */
+ protected String columns[] =
+ { "bigdecimalproperty", "booleanproperty",
+ "byteproperty", "dateproperty",
+ "doubleproperty", "floatproperty",
+ "intproperty", "longproperty",
+ "nullproperty", "shortproperty",
+ "stringproperty", "timeproperty",
+ "timestampproperty" };
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public DynaRowSetTestCase(String name) {
+
+ super(name);
+
+ }
+
+
+ // --------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+
+ dynaClass = new RowSetDynaClass(new TestResultSet());
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+
+ return (new TestSuite(DynaRowSetTestCase.class));
+
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+
+ dynaClass = null;
+
+ }
+
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ public void testGetName() {
+
+ assertEquals("DynaClass name",
+ "org.apache.commons.beanutils.RowSetDynaClass",
+ dynaClass.getName());
+
+
+ }
+
+
+ public void testGetDynaProperty() {
+
+ // Invalid argument test
+ try {
+ dynaClass.getDynaProperty(null);
+ fail("Did not throw IllegaArgumentException");
+ } catch (IllegalArgumentException e) {
+ ; // Expected result
+ }
+
+ // Negative test
+ DynaProperty dynaProp = dynaClass.getDynaProperty("unknownProperty");
+ assertTrue("unknown property returns null",
+ (dynaProp == null));
+
+ // Positive test
+ dynaProp = dynaClass.getDynaProperty("stringproperty");
+ assertNotNull("string property exists", dynaProp);
+ assertEquals("string property name", "stringproperty",
+ dynaProp.getName());
+ assertEquals("string property class", String.class,
+ dynaProp.getType());
+
+ }
+
+
+ public void testGetDynaProperties() {
+
+ DynaProperty dynaProps[] = dynaClass.getDynaProperties();
+ assertNotNull("dynaProps exists", dynaProps);
+ assertEquals("dynaProps length", columns.length, dynaProps.length);
+ for (int i = 0; i < columns.length; i++) {
+ assertEquals("Property " + columns[i],
+ columns[i], dynaProps[i].getName());
+ }
+
+ }
+
+
+ public void testNewInstance() {
+
+ try {
+ dynaClass.newInstance();
+ fail("Did not throw UnsupportedOperationException()");
+ } catch (UnsupportedOperationException e) {
+ ; // Expected result
+ } catch (Exception e) {
+ fail("Threw exception " + e);
+ }
+
+ }
+
+
+ public void testListCount() {
+
+ List rows = dynaClass.getRows();
+ assertNotNull("list exists", rows);
+ assertEquals("list row count", 5, rows.size());
+
+ }
+
+
+ public void testListResults() {
+
+ // Grab the third row
+ List rows = dynaClass.getRows();
+ DynaBean row = (DynaBean) rows.get(2);
+
+ // Invalid argument test
+ try {
+ row.get("unknownProperty");
+ fail("Did not throw IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ ; // Expected result
+ }
+
+ // Verify property values
+
+ Object bigDecimalProperty = row.get("bigdecimalproperty");
+ assertNotNull("bigDecimalProperty exists", bigDecimalProperty);
+ assertTrue("bigDecimalProperty type",
+ bigDecimalProperty instanceof BigDecimal);
+ assertEquals("bigDecimalProperty value",
+ 123.45,
+ ((BigDecimal) bigDecimalProperty).doubleValue(),
+ 0.005);
+
+ Object intProperty = row.get("intproperty");
+ assertNotNull("intProperty exists", intProperty);
+ assertTrue("intProperty type",
+ intProperty instanceof Integer);
+ assertEquals("intProperty value",
+ 103,
+ ((Integer) intProperty).intValue());
+
+ Object nullProperty = row.get("nullproperty");
+ assertNull("nullProperty null", nullProperty);
+
+ Object stringProperty = row.get("stringproperty");
+ assertNotNull("stringProperty exists", stringProperty);
+ assertTrue("stringProperty type",
+ stringProperty instanceof String);
+ assertEquals("stringProperty value",
+ "This is a string",
+ (String) stringProperty);
+
+
+ }
+
+ public void testLimitedRows() throws Exception {
+
+ // created one with low limit
+ RowSetDynaClass limitedDynaClass = new RowSetDynaClass(new TestResultSet(), 3);
+ List rows = limitedDynaClass.getRows();
+ assertNotNull("list exists", rows);
+ assertEquals("limited row count", 3, rows.size());
+
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/ExtendMapBean.java b/trunk/src/test/org/apache/commons/beanutils/ExtendMapBean.java
new file mode 100644
index 0000000..d03310b
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/ExtendMapBean.java
@@ -0,0 +1,44 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+import java.util.Hashtable;
+
+
+/**
+ * Used to test
+ *
+ * @author Robert Burrell
+ */
+
+public class ExtendMapBean extends Hashtable {
+
+ private String dbName = "[UNSET]";
+
+ public ExtendMapBean() {}
+
+ public String getUnusuallyNamedProperty()
+ {
+ return dbName;
+ }
+
+ public void setUnusuallyNamedProperty(String dbName)
+ {
+ this.dbName = dbName;
+ }
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/LazyDynaBeanTestCase.java b/trunk/src/test/org/apache/commons/beanutils/LazyDynaBeanTestCase.java
new file mode 100644
index 0000000..80165d9
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/LazyDynaBeanTestCase.java
@@ -0,0 +1,483 @@
+/*
+ * Copyright 2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.beanutils;
+
+import java.util.HashMap;
+import java.util.TreeMap;
+import java.util.ArrayList;
+import java.util.LinkedList;
+import java.lang.reflect.InvocationTargetException;
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+/**
+ * <p>Test Case for the <code>LazyDynaBean</code> implementation class.</p>
+ *
+ * @author Niall Pemberton
+ */
+public class LazyDynaBeanTestCase extends TestCase {
+
+ protected LazyDynaBean bean = null;
+ protected LazyDynaClass dynaClass = null;
+ protected String testProperty = "myProperty";
+ protected String testPropertyA = "myProperty-A";
+ protected String testPropertyB = "myProperty-B";
+ protected String testString1 = "myStringValue-1";
+ protected String testString2 = "myStringValue-2";
+ protected Integer testInteger1 = new Integer(30);
+ protected Integer testInteger2 = new Integer(40);
+ protected String testKey = "myKey";
+
+ // ---------------------------------------------------------- Constructors
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public LazyDynaBeanTestCase(String name) {
+ super(name);
+ }
+
+ // -------------------------------------------------- Overall Test Methods
+
+ /**
+ * Run thus Test
+ */
+ public static void main(String[] args) {
+ junit.textui.TestRunner.run(suite());
+ }
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(LazyDynaBeanTestCase.class));
+ }
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+ bean = new LazyDynaBean();
+ dynaClass = (LazyDynaClass)bean.getDynaClass();
+ dynaClass.setReturnNull(true);
+ }
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ bean = null;
+ }
+
+ // ------------------------------------------------ Individual Test Methods
+
+ /**
+ * Test Getting/Setting a Simple Property
+ */
+ public void testSimpleProperty() {
+
+ // Check the property & value doesn't exist
+ assertNull("Check Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Value is null", bean.get(testProperty));
+
+ // Set a new property - should add new property and set value
+ bean.set(testProperty, testInteger1);
+ assertEquals("Check First Value is correct", testInteger1, bean.get(testProperty));
+ assertEquals("Check Property type is correct", Integer.class, dynaClass.getDynaProperty(testProperty).getType());
+
+ // Set the property again - should set the new value
+ bean.set(testProperty, testInteger2);
+ assertEquals("Check Second Value is correct", testInteger2, bean.get(testProperty));
+
+ // Set the property again - with a different type, should fail
+ try {
+ bean.set(testProperty, testString1);
+ fail("expected ConversionException trying to set an Integer property to a String");
+ } catch (ConversionException expected) {
+ // expected result
+ }
+
+ }
+
+ /**
+ * Test Setting a Simple Property when MutableDynaClass is set to restricted
+ */
+ public void testSimplePropertyRestricted() {
+
+ // Set the MutableDyanClass to 'restricted' (i.e. no new properties cab be added
+ dynaClass.setRestricted(true);
+ assertTrue("Check MutableDynaClass is restricted", dynaClass.isRestricted());
+
+ // Check the property & value doesn't exist
+ assertNull("Check Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Value is null", bean.get(testProperty));
+
+ // Set the property - should fail because property doesn't exist and MutableDynaClass is restricted
+ try {
+ bean.set(testProperty, testString1);
+ fail("expected IllegalArgumentException trying to add new property to restricted DynaClass");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+
+ }
+
+ /**
+ * Test Getting/Setting a 'Mapped' Property - default HashMap property
+ */
+ public void testMappedPropertyDefault() {
+
+ // Check the property & value doesn't exist
+ assertNull("Check Mapped Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Map is null", bean.get(testProperty));
+ assertNull("Check Mapped Value is null", bean.get(testProperty, testKey));
+
+ // Set a new mapped property - should add new HashMap property and set the mapped value
+ bean.set(testProperty, testKey, testInteger1);
+ assertEquals("Check Mapped Property exists", HashMap.class, bean.get(testProperty).getClass());
+ assertEquals("Check First Mapped Value is correct(a)", testInteger1, bean.get(testProperty, testKey));
+ assertEquals("Check First Mapped Value is correct(b)", testInteger1, ((HashMap)bean.get(testProperty)).get(testKey));
+
+ // Set the property again - should set the new value
+ bean.set(testProperty, testKey, testInteger2);
+ assertEquals("Check Second Mapped Value is correct(a)", testInteger2, bean.get(testProperty, testKey));
+ assertEquals("Check Second Mapped Value is correct(b)", testInteger2, ((HashMap)bean.get(testProperty)).get(testKey));
+ }
+
+ /**
+ * Test Getting/Setting a 'Mapped' Property - use TreeMap property
+ */
+ public void testMappedPropertyTreeMap() {
+
+ // Check the property & value doesn't exist
+ assertNull("Check Mapped Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+
+ // Add a 'TreeMap' property to the DynaClass
+ dynaClass.add(testProperty, TreeMap.class);
+ assertTrue("Check Property is mapped", dynaClass.getDynaProperty(testProperty).isMapped());
+ assertEquals("Check Property is correct type", TreeMap.class, dynaClass.getDynaProperty(testProperty).getType());
+ assertEquals("Check Mapped Property exists", TreeMap.class, bean.get(testProperty).getClass());
+// assertNull("Check mapped property is null", bean.get(testProperty));
+
+ // Set a new mapped property - should instatiate a new TreeMap property and set the mapped value
+ bean.set(testProperty, testKey, testInteger1);
+ assertEquals("Check Mapped Property exists", TreeMap.class, bean.get(testProperty).getClass());
+ assertEquals("Check First Mapped Value is correct(a)", testInteger1, bean.get(testProperty, testKey));
+ assertEquals("Check First Mapped Value is correct(b)", testInteger1, ((TreeMap)bean.get(testProperty)).get(testKey));
+
+ // Set the property again - should set the new value
+ bean.set(testProperty, testKey, testInteger2);
+ assertEquals("Check Second Mapped Value is correct(a)", testInteger2, bean.get(testProperty, testKey));
+ assertEquals("Check Second Mapped Value is correct(b)", testInteger2, ((TreeMap)bean.get(testProperty)).get(testKey));
+ }
+
+ /**
+ * Test Setting a 'Mapped' Property using PropertyUtils
+ */
+ public void testMappedPropertyUtils() {
+
+ dynaClass.setReturnNull(false);
+
+ // Check the property & value doesn't exist
+ assertFalse("Check Mapped Property doesn't exist", dynaClass.isDynaProperty(testProperty));
+ assertNull("Check Map is null", bean.get(testProperty));
+ assertNull("Check Mapped Value is null", bean.get(testProperty, testKey));
+
+ // Set the mapped property using PropertyUtils
+ try {
+ PropertyUtils.setProperty(bean, testProperty+"("+testKey+")", testString1);
+ }
+ catch (NoSuchMethodException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+ catch (InvocationTargetException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+ catch (IllegalAccessException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+
+ // Check property value correctly set
+ assertEquals("Check Mapped Bean Value is correct", testString1, bean.get(testProperty, testKey));
+
+ }
+
+ /**
+ * Test Setting a Mapped Property when MutableDynaClass is set to restricted
+ */
+ public void testMappedPropertyRestricted() {
+
+ // Set the MutableDyanClass to 'restricted' (i.e. no new properties cab be added
+ dynaClass.setRestricted(true);
+ assertTrue("Check MutableDynaClass is restricted", dynaClass.isRestricted());
+
+ // Check the property & value doesn't exist
+ assertNull("Check Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Value is null", bean.get(testProperty));
+
+ // Set the property - should fail because property doesn't exist and MutableDynaClass is restricted
+ try {
+ bean.set(testProperty, testKey, testInteger1);
+ fail("expected IllegalArgumentException trying to add new property to restricted MutableDynaClass");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+
+ }
+
+ /**
+ * Test setting mapped property for type which is not Map
+ */
+ public void testMappedInvalidType() {
+ dynaClass.add(testProperty, String.class);
+ assertFalse("Check Property is not mapped", dynaClass.getDynaProperty(testProperty).isMapped());
+ try {
+ bean.set(testProperty, testKey, testInteger1);
+ fail("set(property, key, value) should have thrown IllegalArgumentException");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test Getting/Setting an 'Indexed' Property - default ArrayList property
+ */
+ public void testIndexedPropertyDefault() {
+
+ int index = 3;
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", bean.get(testProperty));
+ assertNull("Check Indexed value is null", bean.get(testProperty, index));
+
+ // Set the property, should create new ArrayList and set appropriate indexed value
+ bean.set(testProperty, index, testInteger1);
+ assertNotNull("Check Indexed Property is not null", bean.get(testProperty));
+ assertEquals("Check Indexed Property is correct type", ArrayList.class, bean.get(testProperty).getClass());
+ assertEquals("Check First Indexed Value is correct", testInteger1, bean.get(testProperty, index));
+ assertEquals("Check First Array length is correct", new Integer(index+1), new Integer(((ArrayList)bean.get(testProperty)).size()));
+
+ // Set a second indexed value, should automatically grow the ArrayList and set appropriate indexed value
+ index = index + 2;
+ bean.set(testProperty, index, testString1);
+ assertEquals("Check Second Indexed Value is correct", testString1, bean.get(testProperty, index));
+ assertEquals("Check Second Array length is correct", new Integer(index+1), new Integer(((ArrayList)bean.get(testProperty)).size()));
+ }
+
+ /**
+ * Test Getting/Setting a List 'Indexed' Property - use alternative List (LinkedList)
+ */
+ public void testIndexedLinkedList() {
+
+ int index = 3;
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", bean.get(testProperty));
+
+ // Add a 'LinkedList' property to the DynaClass
+ dynaClass.add(testProperty, LinkedList.class);
+ assertTrue("Check Property is indexed", dynaClass.getDynaProperty(testProperty).isIndexed());
+ assertEquals("Check Property is correct type", LinkedList.class, dynaClass.getDynaProperty(testProperty).getType());
+ assertEquals("Check Property type is correct", LinkedList.class, bean.get(testProperty).getClass());
+
+ // Set the property, should instantiate a new LinkedList and set appropriate indexed value
+ bean.set(testProperty, index, testString1);
+ assertEquals("Check Property type is correct", LinkedList.class, bean.get(testProperty).getClass());
+ assertEquals("Check First Indexed Value is correct", testString1, bean.get(testProperty, index));
+ assertEquals("Check First Array length is correct", new Integer(index+1), new Integer(((LinkedList)bean.get(testProperty)).size()));
+
+ // Set a second indexed value, should automatically grow the LinkedList and set appropriate indexed value
+ index = index + 2;
+ bean.set(testProperty, index, testInteger1);
+ assertEquals("Check Second Indexed Value is correct", testInteger1, bean.get(testProperty, index));
+ assertEquals("Check Second Array length is correct", new Integer(index+1), new Integer(((LinkedList)bean.get(testProperty)).size()));
+ }
+
+ /**
+ * Test Getting/Setting a primitive array 'Indexed' Property - use int[]
+ */
+ public void testIndexedPrimitiveArray() {
+
+ int index = 3;
+ int[] primitiveArray = new int[0];
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", bean.get(testProperty));
+
+ // Add a DynaProperty of type int[]
+ dynaClass.add(testProperty, primitiveArray.getClass());
+ assertEquals("Check Indexed Property exists", primitiveArray.getClass(), dynaClass.getDynaProperty(testProperty).getType());
+ assertEquals("Check Indexed Property is correct type", primitiveArray.getClass(), bean.get(testProperty).getClass());
+
+ // Set an indexed value
+ bean.set(testProperty, index, testInteger1);
+ assertNotNull("Check Indexed Property is not null", bean.get(testProperty));
+ assertEquals("Check Indexed Property is correct type", primitiveArray.getClass(), bean.get(testProperty).getClass());
+ assertEquals("Check First Indexed Value is correct(a)", testInteger1, bean.get(testProperty, index));
+ assertEquals("Check First Indexed Value is correct(b)", testInteger1, new Integer(((int[])bean.get(testProperty))[index]));
+ assertEquals("Check Array length is correct", new Integer(index+1), new Integer(((int[])bean.get(testProperty)).length));
+
+ // Set a second indexed value, should automatically grow the int[] and set appropriate indexed value
+ index = index + 2;
+ bean.set(testProperty, index, testInteger2);
+ assertEquals("Check Second Indexed Value is correct(a)", testInteger2, bean.get(testProperty, index));
+ assertEquals("Check Second Indexed Value is correct(b)", testInteger2, new Integer(((int[])bean.get(testProperty))[index]));
+ assertEquals("Check Second Array length is correct", new Integer(index+1), new Integer(((int[])bean.get(testProperty)).length));
+
+ }
+
+ /**
+ * Test Getting/Setting an Object array 'Indexed' Property - use String[]
+ */
+ public void testIndexedObjectArray() {
+
+ int index = 3;
+ Object objectArray = new String[0];
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", bean.get(testProperty));
+
+ // Add a DynaProperty of type String[]
+ dynaClass.add(testProperty, objectArray.getClass());
+ assertEquals("Check Indexed Property exists", objectArray.getClass(), dynaClass.getDynaProperty(testProperty).getType());
+ assertEquals("Check Indexed Property is correct type", objectArray.getClass(), bean.get(testProperty).getClass());
+
+ // Set an indexed value
+ bean.set(testProperty, index, testString1);
+ assertNotNull("Check Indexed Property is not null", bean.get(testProperty));
+ assertEquals("Check Indexed Property is correct type", objectArray.getClass(), bean.get(testProperty).getClass());
+ assertEquals("Check First Indexed Value is correct(a)", testString1, bean.get(testProperty, index));
+ assertEquals("Check First Indexed Value is correct(b)", testString1, ((String[])bean.get(testProperty))[index]);
+ assertEquals("Check Array length is correct", new Integer(index+1), new Integer(((String[])bean.get(testProperty)).length));
+
+ // Set a second indexed value, should automatically grow the String[] and set appropriate indexed value
+ index = index + 2;
+ bean.set(testProperty, index, testString2);
+ assertEquals("Check Second Indexed Value is correct(a)", testString2, bean.get(testProperty, index));
+ assertEquals("Check Second Indexed Value is correct(b)", testString2, ((String[])bean.get(testProperty))[index]);
+ assertEquals("Check Second Array length is correct", new Integer(index+1), new Integer(((String[])bean.get(testProperty)).length));
+ }
+
+ /**
+ * Test Getting/Setting an DynaBean[] array
+ */
+ public void testIndexedDynaBeanArray() {
+
+ int index = 3;
+ Object objectArray = new LazyDynaMap[0];
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", bean.get(testProperty));
+
+ // Add a DynaProperty of type String[]
+ dynaClass.add(testProperty, objectArray.getClass());
+ assertEquals("Check Indexed Property exists", objectArray.getClass(), dynaClass.getDynaProperty(testProperty).getType());
+ assertEquals("Check Indexed Property is correct type", objectArray.getClass(), bean.get(testProperty).getClass());
+
+ // Retrieving from Array should initialize DynaBean
+ for (int i = index; i >= 0; i--) {
+ assertEquals("Check Array Components initialized", LazyDynaMap.class, bean.get(testProperty, index).getClass());
+ }
+
+ dynaClass.add(testPropertyB, objectArray.getClass());
+ LazyDynaMap newMap = new LazyDynaMap();
+ newMap.set(testPropertyB, testString2);
+ bean.set(testPropertyA, index, newMap);
+ assertEquals("Check Indexed Value is correct(a)", testString2, ((DynaBean)bean.get(testPropertyA, index)).get(testPropertyB));
+
+ }
+
+ /**
+ * Test Setting an 'Indexed' Property using PropertyUtils
+ */
+ public void testIndexedPropertyUtils() {
+
+ int index = 3;
+ dynaClass.setReturnNull(false);
+
+ // Check the property & value doesn't exist
+ assertFalse("Check Indexed Property doesn't exist", dynaClass.isDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", bean.get(testProperty));
+ assertNull("Check Indexed value is null", bean.get(testProperty, index));
+
+ // Use PropertyUtils to set the indexed value
+ try {
+ PropertyUtils.setProperty(bean, testProperty+"["+index+"]", testString1);
+ }
+ catch (NoSuchMethodException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+ catch (InvocationTargetException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+ catch (IllegalAccessException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+
+ // Check property value correctly set
+ assertEquals("Check Indexed Bean Value is correct", testString1, bean.get(testProperty, index));
+
+ }
+
+ /**
+ * Test Setting an Indexed Property when MutableDynaClass is set to restricted
+ */
+ public void testIndexedPropertyRestricted() {
+
+ int index = 3;
+
+ // Set the MutableDyanClass to 'restricted' (i.e. no new properties cab be added
+ dynaClass.setRestricted(true);
+ assertTrue("Check MutableDynaClass is restricted", dynaClass.isRestricted());
+
+ // Check the property & value doesn't exist
+ assertNull("Check Property doesn't exist", dynaClass.getDynaProperty(testProperty));
+ assertNull("Check Value is null", bean.get(testProperty));
+
+ // Set the property - should fail because property doesn't exist and MutableDynaClass is restricted
+ try {
+ bean.set(testProperty, index, testInteger1);
+ fail("expected IllegalArgumentException trying to add new property to restricted MutableDynaClass");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+
+ }
+
+ /**
+ * Test setting indexed property for type which is not List or Array
+ */
+ public void testIndexedInvalidType() {
+ int index = 3;
+ dynaClass.add(testProperty, String.class);
+ assertFalse("Check Property is not indexed", dynaClass.getDynaProperty(testProperty).isIndexed());
+ try {
+ bean.set(testProperty, index, testString1);
+ fail("set(property, index, value) should have thrown IllegalArgumentException");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+ }
+
+}
\ No newline at end of file
diff --git a/trunk/src/test/org/apache/commons/beanutils/LazyDynaClassTestCase.java b/trunk/src/test/org/apache/commons/beanutils/LazyDynaClassTestCase.java
new file mode 100644
index 0000000..6e1931f
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/LazyDynaClassTestCase.java
@@ -0,0 +1,256 @@
+/*
+ * Copyright 2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.beanutils;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+/**
+ * <p>Test Case for the <code>LazyDynaClass</code> implementation class.</p>
+ *
+ * @author Niall Pemberton
+ */
+public class LazyDynaClassTestCase extends TestCase {
+
+ protected LazyDynaClass dynaClass = null;
+ protected String testProperty = "myProperty";
+
+ // ---------------------------------------------------------- Constructors
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public LazyDynaClassTestCase(String name) {
+ super(name);
+ }
+
+ // -------------------------------------------------- Overall Test Methods
+
+ /**
+ * Run this Test
+ */
+ public static void main(String[] args) {
+ junit.textui.TestRunner.run(suite());
+ }
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+ dynaClass = new LazyDynaClass();
+ }
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(LazyDynaClassTestCase.class));
+ }
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ dynaClass = null;
+ }
+
+ // ------------------------------------------------ Individual Test Methods
+
+ /**
+ * Test add(name) method
+ */
+ public void testAddProperty1() {
+ dynaClass.add(testProperty);
+ DynaProperty dynaProperty = dynaClass.getDynaProperty(testProperty);
+ assertEquals("name is correct", testProperty, dynaProperty.getName());
+ assertEquals("type is correct", Object.class, dynaProperty.getType());
+ }
+
+ /**
+ * Test add(name, type) method
+ */
+ public void testAddProperty2() {
+ dynaClass.add(testProperty, String.class);
+ DynaProperty dynaProperty = dynaClass.getDynaProperty(testProperty);
+ assertEquals("name is correct", testProperty, dynaProperty.getName());
+ assertEquals("type is correct", String.class, dynaProperty.getType());
+ }
+
+ /**
+ * Test add(name, type, readable, writable) method
+ */
+ public void testAddProperty3() {
+ try {
+ dynaClass.add(testProperty, String.class, true, true);
+ fail("add(name, type, readable, writable) did not throw UnsupportedOperationException");
+ } catch (UnsupportedOperationException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test add(name) method with 'null' name
+ */
+ public void testAddPropertyNullName1() {
+ try {
+ dynaClass.add((String)null);
+ fail("null property name not prevented");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test add(name, type) method with 'null' name
+ */
+ public void testAddPropertyNullName2() {
+ try {
+ dynaClass.add(null, String.class);
+ fail("null property name not prevented");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test add(name, type, readable, writable) method with 'null' name
+ */
+ public void testAddPropertyNullName3() {
+ try {
+ dynaClass.add(null, String.class, true, true);
+ fail("add(name, type, readable, writable) did not throw UnsupportedOperationException");
+ } catch (UnsupportedOperationException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test add(name) method when restricted is set to 'true'
+ */
+ public void testAddPropertyRestricted1() {
+ dynaClass.setRestricted(true);
+ assertTrue("MutableDynaClass is restricted", dynaClass.isRestricted());
+ try {
+ dynaClass.add(testProperty);
+ fail("add(name) did not throw IllegalStateException");
+ } catch (IllegalStateException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test add(name, type) method when restricted is set to 'true'
+ */
+ public void testAddPropertyRestricted2() {
+ dynaClass.setRestricted(true);
+ assertTrue("MutableDynaClass is restricted", dynaClass.isRestricted());
+ try {
+ dynaClass.add(testProperty, String.class);
+ fail("add(name, type) did not throw IllegalStateException");
+ } catch (IllegalStateException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test add(name, type, readable, writable) method when restricted is set to 'true'
+ */
+ public void testAddPropertyRestricted3() {
+ dynaClass.setRestricted(true);
+ assertTrue("MutableDynaClass is restricted", dynaClass.isRestricted());
+ try {
+ dynaClass.add(testProperty, String.class, true, true);
+ fail("add(name, type, readable, writable) did not throw UnsupportedOperationException");
+ } catch (UnsupportedOperationException t) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test retrieving a property which doesn't exist (returnNull is 'false')
+ */
+ public void testGetPropertyDoesntExist1() {
+ dynaClass.setReturnNull(false);
+ assertFalse("returnNull is 'false'", dynaClass.isReturnNull());
+ DynaProperty dynaProperty = dynaClass.getDynaProperty(testProperty);
+ assertEquals("name is correct", testProperty, dynaProperty.getName());
+ assertEquals("type is correct", Object.class, dynaProperty.getType());
+ assertFalse("property doesnt exist", dynaClass.isDynaProperty(testProperty));
+ }
+
+
+ /**
+ * Test retrieving a property which doesn't exist (returnNull is 'true')
+ */
+ public void testGetPropertyDoesntExist2() {
+ dynaClass.setReturnNull(true);
+ assertTrue("returnNull is 'true'", dynaClass.isReturnNull());
+ assertNull("property is null", dynaClass.getDynaProperty(testProperty));
+ }
+
+ /**
+ * Test removing a property
+ */
+ public void testRemoveProperty() {
+ dynaClass.setReturnNull(true);
+ dynaClass.add(testProperty);
+ assertTrue("Property exists", dynaClass.isDynaProperty(testProperty));
+ assertNotNull("property is Not null", dynaClass.getDynaProperty(testProperty));
+ dynaClass.remove(testProperty);
+ assertFalse("Property doesn't exist", dynaClass.isDynaProperty(testProperty));
+ assertNull("property is null", dynaClass.getDynaProperty(testProperty));
+ }
+
+ /**
+ * Test removing a property, name is null
+ */
+ public void testRemovePropertyNullName() {
+ try {
+ dynaClass.remove(null);
+ fail("remove(null) did not throw IllegalArgumentException");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test removing a property, DynaClass is restricted
+ */
+ public void testRemovePropertyRestricted() {
+ dynaClass.add(testProperty);
+ assertTrue("Property exists", dynaClass.isDynaProperty(testProperty));
+ dynaClass.setRestricted(true);
+ assertTrue("MutableDynaClass is restricted", dynaClass.isRestricted());
+ try {
+ dynaClass.remove(testProperty);
+ fail("remove property when MutableDynaClassis restricted did not throw IllegalStateException");
+ } catch (IllegalStateException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test removing a property which doesn't exist
+ */
+ public void testRemovePropertyDoesntExist() {
+ assertFalse("property doesn't exist", dynaClass.isDynaProperty(testProperty));
+ dynaClass.remove(testProperty);
+ assertFalse("property still doesn't exist", dynaClass.isDynaProperty(testProperty));
+ }
+}
\ No newline at end of file
diff --git a/trunk/src/test/org/apache/commons/beanutils/LazyDynaMapTestCase.java b/trunk/src/test/org/apache/commons/beanutils/LazyDynaMapTestCase.java
new file mode 100644
index 0000000..a9df074
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/LazyDynaMapTestCase.java
@@ -0,0 +1,486 @@
+/*
+ * Copyright 2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.beanutils;
+
+import java.util.HashMap;
+import java.util.TreeMap;
+import java.util.ArrayList;
+import java.util.LinkedList;
+import java.lang.reflect.InvocationTargetException;
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+/**
+ * <p>Test Case for the <code>LazyDynaMap</code> implementation class.</p>
+ *
+ * @author Niall Pemberton
+ */
+public class LazyDynaMapTestCase extends TestCase {
+
+ protected LazyDynaMap dynaMap = null;
+ protected String testProperty = "myProperty";
+ protected String testPropertyA = "myProperty-A";
+ protected String testPropertyB = "myProperty-B";
+ protected String testString1 = "myStringValue-1";
+ protected String testString2 = "myStringValue-2";
+ protected Integer testInteger1 = new Integer(30);
+ protected Integer testInteger2 = new Integer(40);
+ protected String testKey = "myKey";
+
+ // ---------------------------------------------------------- Constructors
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public LazyDynaMapTestCase(String name) {
+ super(name);
+ }
+
+ // -------------------------------------------------- Overall Test Methods
+
+ /**
+ * Run thus Test
+ */
+ public static void main(String[] args) {
+ junit.textui.TestRunner.run(suite());
+ }
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(LazyDynaMapTestCase.class));
+ }
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+ dynaMap = new LazyDynaMap();
+ dynaMap.setReturnNull(true);
+ }
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ dynaMap = null;
+ }
+
+ // ------------------------------------------------ Individual Test Methods
+
+ /**
+ * General Tests
+ */
+ public void testGeneral() {
+// LazyDynaMap bean = new LazyDynaMap("TestBean");
+ assertEquals("Check DynaClass name", "TestBean", new LazyDynaMap("TestBean").getName());
+
+ }
+
+ /**
+ * Test Getting/Setting a Simple Property
+ */
+ public void testSimpleProperty() {
+
+ // Check the property & value doesn't exist
+ assertNull("Check Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Value is null", dynaMap.get(testProperty));
+
+ // Set a new property - should add new property and set value
+ dynaMap.set(testProperty, testInteger1);
+ assertEquals("Check First Value is correct", testInteger1, dynaMap.get(testProperty));
+ assertEquals("Check Property type is correct", Integer.class, dynaMap.getDynaProperty(testProperty).getType());
+
+ // Set the property again - should set the new value
+ dynaMap.set(testProperty, testInteger2);
+ assertEquals("Check Second Value is correct", testInteger2, dynaMap.get(testProperty));
+
+ // Set the property again - with a different type, should succeed
+ dynaMap.set(testProperty, testString1);
+ assertEquals("Check Third Value is correct", testString1, dynaMap.get(testProperty));
+
+ }
+
+ /**
+ * Test Setting a Simple Property when MutableDynaClass is set to restricted
+ */
+ public void testSimplePropertyRestricted() {
+
+ // Set the MutableDyanClass to 'restricted' (i.e. no new properties cab be added
+ dynaMap.setRestricted(true);
+ assertTrue("Check MutableDynaClass is restricted", dynaMap.isRestricted());
+
+ // Check the property & value doesn't exist
+ assertNull("Check Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Value is null", dynaMap.get(testProperty));
+
+ // Set the property - should fail because property doesn't exist and MutableDynaClass is restricted
+ try {
+ dynaMap.set(testProperty, testString1);
+ fail("expected IllegalArgumentException trying to add new property to restricted DynaClass");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+
+ }
+
+ /**
+ * Test Getting/Setting a 'Mapped' Property - default HashMap property
+ */
+ public void testMappedPropertyDefault() {
+
+ // Check the property & value doesn't exist
+ assertNull("Check Mapped Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Map is null", dynaMap.get(testProperty));
+ assertNull("Check Mapped Value is null", dynaMap.get(testProperty, testKey));
+
+ // Set a new mapped property - should add new HashMap property and set the mapped value
+ dynaMap.set(testProperty, testKey, testInteger1);
+ assertEquals("Check Mapped Property exists", HashMap.class, dynaMap.get(testProperty).getClass());
+ assertEquals("Check First Mapped Value is correct(a)", testInteger1, dynaMap.get(testProperty, testKey));
+ assertEquals("Check First Mapped Value is correct(b)", testInteger1, ((HashMap)dynaMap.get(testProperty)).get(testKey));
+
+ // Set the property again - should set the new value
+ dynaMap.set(testProperty, testKey, testInteger2);
+ assertEquals("Check Second Mapped Value is correct(a)", testInteger2, dynaMap.get(testProperty, testKey));
+ assertEquals("Check Second Mapped Value is correct(b)", testInteger2, ((HashMap)dynaMap.get(testProperty)).get(testKey));
+ }
+
+ /**
+ * Test Getting/Setting a 'Mapped' Property - use TreeMap property
+ */
+ public void testMappedPropertyTreeMap() {
+
+ // Check the property & value doesn't exist
+ assertNull("Check Mapped Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Map is null", dynaMap.get(testProperty));
+
+ // Add a 'TreeMap' property to the DynaClass
+ dynaMap.add(testProperty, TreeMap.class);
+ assertTrue("Check Property is mapped", dynaMap.getDynaProperty(testProperty).isMapped());
+ assertEquals("Check Property is correct type", TreeMap.class, dynaMap.getDynaProperty(testProperty).getType());
+ assertEquals("Check Mapped Property now exists", TreeMap.class, dynaMap.get(testProperty).getClass());
+
+ // Set a new mapped property - should instatiate a new TreeMap property and set the mapped value
+ dynaMap.set(testProperty, testKey, testInteger1);
+ assertEquals("Check Mapped Property exists", TreeMap.class, dynaMap.get(testProperty).getClass());
+ assertEquals("Check First Mapped Value is correct(a)", testInteger1, dynaMap.get(testProperty, testKey));
+ assertEquals("Check First Mapped Value is correct(b)", testInteger1, ((TreeMap)dynaMap.get(testProperty)).get(testKey));
+
+ // Set the property again - should set the new value
+ dynaMap.set(testProperty, testKey, testInteger2);
+ assertEquals("Check Second Mapped Value is correct(a)", testInteger2, dynaMap.get(testProperty, testKey));
+ assertEquals("Check Second Mapped Value is correct(b)", testInteger2, ((TreeMap)dynaMap.get(testProperty)).get(testKey));
+ }
+
+ /**
+ * Test Setting a 'Mapped' Property using PropertyUtils
+ */
+ public void testMappedPropertyUtils() {
+
+ dynaMap.setReturnNull(false);
+
+ // Check the property & value doesn't exist
+ assertFalse("Check Mapped Property doesn't exist", dynaMap.isDynaProperty(testProperty));
+ assertNull("Check Map is null", dynaMap.get(testProperty));
+ assertNull("Check Mapped Value is null", dynaMap.get(testProperty, testKey));
+
+ // Set the mapped property using PropertyUtils
+ try {
+ PropertyUtils.setProperty(dynaMap, testProperty+"("+testKey+")", testString1);
+ }
+ catch (NoSuchMethodException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+ catch (InvocationTargetException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+ catch (IllegalAccessException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+
+ // Check property value correctly set
+ assertEquals("Check Mapped Bean Value is correct", testString1, dynaMap.get(testProperty, testKey));
+
+ }
+
+ /**
+ * Test Setting a Mapped Property when MutableDynaClass is set to restricted
+ */
+ public void testMappedPropertyRestricted() {
+
+ // Set the MutableDyanClass to 'restricted' (i.e. no new properties cab be added
+ dynaMap.setRestricted(true);
+ assertTrue("Check MutableDynaClass is restricted", dynaMap.isRestricted());
+
+ // Check the property & value doesn't exist
+ assertNull("Check Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Value is null", dynaMap.get(testProperty));
+
+ // Set the property - should fail because property doesn't exist and MutableDynaClass is restricted
+ try {
+ dynaMap.set(testProperty, testKey, testInteger1);
+ fail("expected IllegalArgumentException trying to add new property to restricted MutableDynaClass");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+
+ }
+
+ /**
+ * Test setting mapped property for type which is not Map
+ */
+ public void testMappedInvalidType() {
+ dynaMap.set(testProperty, new Integer(1));
+ assertFalse("Check Property is not mapped", dynaMap.getDynaProperty(testProperty).isMapped());
+ try {
+ dynaMap.set(testProperty, testKey, testInteger1);
+ fail("set(property, key, value) should have thrown IllegalArgumentException");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+ }
+
+ /**
+ * Test Getting/Setting an 'Indexed' Property - default ArrayList property
+ */
+ public void testIndexedPropertyDefault() {
+
+ int index = 3;
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", dynaMap.get(testProperty));
+ assertNull("Check Indexed value is null", dynaMap.get(testProperty, index));
+
+ // Set the property, should create new ArrayList and set appropriate indexed value
+ dynaMap.set(testProperty, index, testInteger1);
+ assertNotNull("Check Indexed Property is not null", dynaMap.get(testProperty));
+ assertEquals("Check Indexed Property is correct type", ArrayList.class, dynaMap.get(testProperty).getClass());
+ assertEquals("Check First Indexed Value is correct", testInteger1, dynaMap.get(testProperty, index));
+ assertEquals("Check First Array length is correct", new Integer(index+1), new Integer(((ArrayList)dynaMap.get(testProperty)).size()));
+
+ // Set a second indexed value, should automatically grow the ArrayList and set appropriate indexed value
+ index = index + 2;
+ dynaMap.set(testProperty, index, testString1);
+ assertEquals("Check Second Indexed Value is correct", testString1, dynaMap.get(testProperty, index));
+ assertEquals("Check Second Array length is correct", new Integer(index+1), new Integer(((ArrayList)dynaMap.get(testProperty)).size()));
+ }
+
+ /**
+ * Test Getting/Setting a List 'Indexed' Property - use alternative List (LinkedList)
+ */
+ public void testIndexedLinkedList() {
+
+ int index = 3;
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", dynaMap.get(testProperty));
+
+ // Add a 'LinkedList' property to the DynaClass - should instantiate a new LinkedList
+ dynaMap.add(testProperty, LinkedList.class);
+ assertTrue("Check Property is indexed", dynaMap.getDynaProperty(testProperty).isIndexed());
+ assertEquals("Check Property is correct type", LinkedList.class, dynaMap.getDynaProperty(testProperty).getType());
+ assertEquals("Check Indexed Property now exists", LinkedList.class, dynaMap.get(testProperty).getClass());
+
+ // Set the Indexed property, should grow the list to the correct size
+ dynaMap.set(testProperty, index, testString1);
+ assertEquals("Check Property type is correct", LinkedList.class, dynaMap.get(testProperty).getClass());
+ assertEquals("Check First Indexed Value is correct", testString1, dynaMap.get(testProperty, index));
+ assertEquals("Check First Array length is correct", new Integer(index+1), new Integer(((LinkedList)dynaMap.get(testProperty)).size()));
+
+ // Set a second indexed value, should automatically grow the LinkedList and set appropriate indexed value
+ index = index + 2;
+ dynaMap.set(testProperty, index, testInteger1);
+ assertEquals("Check Second Indexed Value is correct", testInteger1, dynaMap.get(testProperty, index));
+ assertEquals("Check Second Array length is correct", new Integer(index+1), new Integer(((LinkedList)dynaMap.get(testProperty)).size()));
+ }
+
+ /**
+ * Test Getting/Setting a primitive array 'Indexed' Property - use int[]
+ */
+ public void testIndexedPrimitiveArray() {
+
+ int index = 3;
+ int[] primitiveArray = new int[0];
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", dynaMap.get(testProperty));
+
+ // Add a DynaProperty of type int[]
+ dynaMap.add(testProperty, primitiveArray.getClass());
+ assertEquals("Check Indexed Property exists", primitiveArray.getClass(), dynaMap.getDynaProperty(testProperty).getType());
+ assertTrue("Check Indexed Property exists", dynaMap.get(testProperty).getClass().isInstance(primitiveArray));
+
+ // Set an indexed value
+ dynaMap.set(testProperty, index, testInteger1);
+ assertNotNull("Check Indexed Property is not null", dynaMap.get(testProperty));
+ assertEquals("Check Indexed Property is correct type", primitiveArray.getClass(), dynaMap.get(testProperty).getClass());
+ assertEquals("Check First Indexed Value is correct(a)", testInteger1, dynaMap.get(testProperty, index));
+ assertEquals("Check First Indexed Value is correct(b)", testInteger1, new Integer(((int[])dynaMap.get(testProperty))[index]));
+ assertEquals("Check Array length is correct", new Integer(index+1), new Integer(((int[])dynaMap.get(testProperty)).length));
+
+ // Set a second indexed value, should automatically grow the int[] and set appropriate indexed value
+ index = index + 2;
+ dynaMap.set(testProperty, index, testInteger2);
+ assertEquals("Check Second Indexed Value is correct(a)", testInteger2, dynaMap.get(testProperty, index));
+ assertEquals("Check Second Indexed Value is correct(b)", testInteger2, new Integer(((int[])dynaMap.get(testProperty))[index]));
+ assertEquals("Check Second Array length is correct", new Integer(index+1), new Integer(((int[])dynaMap.get(testProperty)).length));
+
+ }
+
+ /**
+ * Test Getting/Setting an Object array 'Indexed' Property - use String[]
+ */
+ public void testIndexedObjectArray() {
+
+ int index = 3;
+ Object objectArray = new String[0];
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", dynaMap.get(testProperty));
+
+ // Add a DynaProperty of type String[]
+ dynaMap.add(testProperty, objectArray.getClass());
+ assertEquals("Check Indexed Property exists", objectArray.getClass(), dynaMap.getDynaProperty(testProperty).getType());
+ assertTrue("Check Indexed Property exists", dynaMap.get(testProperty).getClass().isInstance(objectArray));
+
+ // Set an indexed value
+ dynaMap.set(testProperty, index, testString1);
+ assertNotNull("Check Indexed Property is not null", dynaMap.get(testProperty));
+ assertEquals("Check Indexed Property is correct type", objectArray.getClass(), dynaMap.get(testProperty).getClass());
+ assertEquals("Check First Indexed Value is correct(a)", testString1, dynaMap.get(testProperty, index));
+ assertEquals("Check First Indexed Value is correct(b)", testString1, ((String[])dynaMap.get(testProperty))[index]);
+ assertEquals("Check Array length is correct", new Integer(index+1), new Integer(((String[])dynaMap.get(testProperty)).length));
+
+ // Set a second indexed value, should automatically grow the String[] and set appropriate indexed value
+ index = index + 2;
+ dynaMap.set(testProperty, index, testString2);
+ assertEquals("Check Second Indexed Value is correct(a)", testString2, dynaMap.get(testProperty, index));
+ assertEquals("Check Second Indexed Value is correct(b)", testString2, ((String[])dynaMap.get(testProperty))[index]);
+ assertEquals("Check Second Array length is correct", new Integer(index+1), new Integer(((String[])dynaMap.get(testProperty)).length));
+ }
+
+ /**
+ * Test Getting/Setting an DynaBean[] array
+ */
+ public void testIndexedDynaBeanArray() {
+
+ int index = 3;
+ Object objectArray = new LazyDynaBean[0];
+
+ // Check the property & value doesn't exist
+ assertNull("Check Indexed Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", dynaMap.get(testProperty));
+
+ // Add a DynaProperty of type String[]
+ dynaMap.add(testProperty, objectArray.getClass());
+ assertEquals("Check Indexed Property exists", objectArray.getClass(), dynaMap.getDynaProperty(testProperty).getType());
+ assertEquals("Check Indexed Property is correct type", objectArray.getClass(), dynaMap.get(testProperty).getClass());
+
+ // Retrieving from Array should initialize DynaBean
+ for (int i = index; i >= 0; i--) {
+ assertEquals("Check Array Components initialized", LazyDynaBean.class, dynaMap.get(testProperty, index).getClass());
+ }
+
+ dynaMap.add(testPropertyB, objectArray.getClass());
+ LazyDynaBean newBean = new LazyDynaBean();
+ newBean.set(testPropertyB, testString2);
+ dynaMap.set(testPropertyA, index, newBean);
+ assertEquals("Check Indexed Value is correct(a)", testString2, ((DynaBean)dynaMap.get(testPropertyA, index)).get(testPropertyB));
+
+ }
+
+ /**
+ * Test Setting an 'Indexed' Property using PropertyUtils
+ */
+ public void testIndexedPropertyUtils() {
+
+ int index = 3;
+ dynaMap.setReturnNull(false);
+
+ // Check the property & value doesn't exist
+ assertFalse("Check Indexed Property doesn't exist", dynaMap.isDynaProperty(testProperty));
+ assertNull("Check Indexed Property is null", dynaMap.get(testProperty));
+ assertNull("Check Indexed value is null", dynaMap.get(testProperty, index));
+
+ // Use PropertyUtils to set the indexed value
+ try {
+ PropertyUtils.setProperty(dynaMap, testProperty+"["+index+"]", testString1);
+ }
+ catch (NoSuchMethodException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+ catch (InvocationTargetException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+ catch (IllegalAccessException ex) {
+ fail("testIndexedPropertyUtils threw "+ex);
+ }
+
+ // Check property value correctly set
+ assertEquals("Check Indexed Bean Value is correct", testString1, dynaMap.get(testProperty, index));
+
+ }
+
+ /**
+ * Test Setting an Indexed Property when MutableDynaClass is set to restricted
+ */
+ public void testIndexedPropertyRestricted() {
+
+ int index = 3;
+
+ // Set the MutableDyanClass to 'restricted' (i.e. no new properties cab be added
+ dynaMap.setRestricted(true);
+ assertTrue("Check MutableDynaClass is restricted", dynaMap.isRestricted());
+
+ // Check the property & value doesn't exist
+ assertNull("Check Property doesn't exist", dynaMap.getDynaProperty(testProperty));
+ assertNull("Check Value is null", dynaMap.get(testProperty));
+
+ // Set the property - should fail because property doesn't exist and MutableDynaClass is restricted
+ try {
+ dynaMap.set(testProperty, index, testInteger1);
+ fail("expected IllegalArgumentException trying to add new property to restricted MutableDynaClass");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+
+ }
+
+ /**
+ * Test setting indexed property for type which is not List or Array
+ */
+ public void testIndexedInvalidType() {
+ int index = 3;
+ dynaMap.set(testProperty, "Test String");
+ assertFalse("Check Property is not indexed", dynaMap.getDynaProperty(testProperty).isIndexed());
+ try {
+ dynaMap.set(testProperty, index, testString1);
+ fail("set(property, index, value) should have thrown IllegalArgumentException");
+ } catch (IllegalArgumentException expected) {
+ // expected result
+ }
+ }
+
+}
\ No newline at end of file
diff --git a/trunk/src/test/org/apache/commons/beanutils/MappedPropertyTestBean.java b/trunk/src/test/org/apache/commons/beanutils/MappedPropertyTestBean.java
new file mode 100644
index 0000000..388b6b2
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/MappedPropertyTestBean.java
@@ -0,0 +1,43 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+import java.util.HashMap;
+import java.util.Map;
+
+/**
+ * Just a java bean (JAJB) to try to replicate a reported bug
+ *
+ * @author Robert Burrell Donkin
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class MappedPropertyTestBean {
+
+ private Map map = new HashMap();
+
+ // -------------------------------------------------------------- Properties
+
+ public String getMapproperty(String key) {
+ return (String) map.get(key);
+ }
+
+ public void setMapproperty(String key, String value) {
+ map.put(key, value);
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/MethodUtilsTestCase.java b/trunk/src/test/org/apache/commons/beanutils/MethodUtilsTestCase.java
new file mode 100644
index 0000000..1ac5b6c
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/MethodUtilsTestCase.java
@@ -0,0 +1,560 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+import java.io.OutputStream;
+import java.io.PrintStream;
+
+import java.lang.reflect.Method;
+import java.lang.reflect.Modifier;
+
+import org.apache.commons.beanutils.priv.PrivateBeanFactory;
+import org.apache.commons.beanutils.priv.PublicSubBean;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * <p> Test case for <code>MethodUtils</code> </p>
+ *
+ */
+public class MethodUtilsTestCase extends TestCase {
+
+ // ---------------------------------------------------- Instance Variables
+
+ protected PrivateBeanFactory privateBeanFactory;
+
+ // ---------------------------------------------------------- Constructors
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public MethodUtilsTestCase(String name) {
+ super(name);
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() {
+ privateBeanFactory = new PrivateBeanFactory();
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(MethodUtilsTestCase.class));
+ }
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ privateBeanFactory = null;
+ }
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+ /**
+ * <p> Test <code>getAccessibleMethod</code>.
+ */
+ public void testGetAccessibleMethod() {
+ // test MethodUtils.getAccessibleMethod
+ // we'll make things easier by using the convenience methods
+
+ // easy bit first - find a public method
+ // METHOD ONE
+ Method method = MethodUtils.getAccessibleMethod
+ (TestBean.class, "setStringProperty", String.class);
+
+ // check that we've found one that matches
+ assertNotNull(method);
+ assertEquals("method ONE is named correctly",
+ "setStringProperty", method.getName());
+ assertTrue("Method ONE is public",
+ Modifier.isPublic(method.getModifiers()));
+
+ // trickier this one - find a method in a direct interface
+ // METHOD TWO
+ method = MethodUtils.getAccessibleMethod
+ (privateBeanFactory.create().getClass(),
+ "methodBar",
+ String.class);
+
+ // check that we've found one that matches
+ assertNotNull(method);
+ assertEquals("Method TWO is named correctly",
+ "methodBar", method.getName());
+ assertTrue("Method TWO is public",
+ Modifier.isPublic(method.getModifiers()));
+
+ // trickier this one - find a method in a indirect interface
+ // METHOD THREE
+ method = MethodUtils.getAccessibleMethod
+ (privateBeanFactory.createSubclass().getClass(),
+ "methodBaz",
+ String.class);
+
+ // check that we've found one that matches
+ assertNotNull(method);
+ assertEquals("Method THREE is named correctly",
+ "methodBaz", method.getName());
+ assertTrue("Method THREE is public",
+ Modifier.isPublic(method.getModifiers()));
+
+ }
+
+
+ /**
+ * <p> Test <code>invokeExactMethod</code>.
+ */
+ public void testInvokeExactMethod() {
+ // test MethodUtils.invokeExactMethod
+ // easy bit first - invoke a public method
+ // METHOD ONE
+ try {
+
+ TestBean bean = new TestBean();
+ Object ret = MethodUtils.invokeExactMethod(bean, "setStringProperty", "TEST");
+ // check that the return's right and that the properties been set
+ assertNull(ret);
+ assertEquals("Method ONE was invoked", "TEST", bean.getStringProperty());
+
+ } catch (Throwable t) {
+ // ONE
+ fail("Exception in method ONE prevented invokation: " + t.toString());
+ }
+
+ // trickier this one - find a method in a direct interface
+ // METHOD TWO FAILURE
+ try {
+
+ Object ret = MethodUtils.invokeExactMethod(
+ privateBeanFactory.create(),
+ "methodBar",
+ "ANOTHER TEST");
+
+ // check that we've found one that matches
+ assertEquals("Method TWO was invoked correctly", "ANOTHER TEST", ret);
+
+ } catch (Throwable t) {
+ // METHOD TWO FAILURE
+ fail("Exception in method TWO prevented invokation: " + t.toString());
+ }
+
+
+ // trickier this one - find a method in a indirect interface
+ // METHOD THREE
+ try {
+
+ Object ret = MethodUtils.invokeExactMethod(
+ privateBeanFactory.createSubclass(),
+ "methodBaz",
+ "YET ANOTHER TEST");
+
+
+ // check that we've found one that matches
+ assertEquals("Method TWO was invoked correctly", "YET ANOTHER TEST", ret);
+
+
+ } catch (Throwable t) {
+ // METHOD THREE FAILURE
+ fail("Exception in method THREE prevented invokation: " + t.toString());
+
+ }
+ }
+
+ /**
+ * <p> Test <code>invokeMethod</code>.
+ */
+ public void testInvokeMethod() throws Exception {
+ // i'm going to test that the actual calls work first and then try them via reflection
+
+ AbstractParent parent = new AlphaBean("parent");
+
+ // try testAddChild through abstract superclass
+ BetaBean childOne = new BetaBean("ChildOne");
+
+ assertEquals("Oh no! Badly coded test case! (1)", "ChildOne", parent.testAddChild(childOne));
+
+ // let's try MethodUtils version
+ assertEquals(
+ "Cannot invoke through abstract class (1)",
+ "ChildOne",
+ MethodUtils.invokeMethod(parent, "testAddChild", childOne));
+
+
+ // try adding through interface
+ AlphaBean childTwo = new AlphaBean("ChildTwo");
+
+ assertEquals("Oh no! Badly coded test case! (2)", "ChildTwo", parent.testAddChild(childTwo));
+
+ // let's try MethodUtils version
+ assertEquals(
+ "Cannot invoke through interface (1)",
+ "ChildTwo",
+ MethodUtils.invokeMethod(parent, "testAddChild", childTwo));
+
+
+ Object[] params = new Object[2];
+
+ assertEquals("Oh no! Badly coded test case! (3)", "ChildOne", parent.testAddChild2("parameter", childOne));
+
+
+ // let's try MethodUtils version
+ params[0] = "parameter";
+ params[1] = childOne;
+
+ assertEquals(
+ "Cannot invoke through abstract class (1)",
+ "ChildOne",
+ MethodUtils.invokeMethod(parent, "testAddChild2", params));
+
+ assertEquals("Oh no! Badly coded test case! (4)", "ChildTwo", parent.testAddChild2("parameter", childTwo));
+
+ // let's try MethodUtils version
+ params[0] = "parameter";
+ params[1] = childTwo;
+
+ assertEquals(
+ "Cannot invoke through abstract class (1)",
+ "ChildTwo",
+ MethodUtils.invokeMethod(parent, "testAddChild2", params));
+
+ // test that exception is correctly thrown when a method cannot be found with matching params
+ try {
+ // the next line
+ parent = new AlphaBean("parent");
+ childOne = new BetaBean("ChildOne");
+ MethodUtils.invokeMethod(parent, "bogus", childOne);
+ // should get here!
+ fail("No exception thrown when no appropriate method exists");
+
+ } catch (NoSuchMethodException e) {
+ // this is what we're expecting!
+ }
+
+ MethodUtils.invokeMethod(parent, "getName", null);
+ MethodUtils.invokeMethod(parent, "getName", null, null);
+ MethodUtils.invokeExactMethod(parent, "getName", null);
+ MethodUtils.invokeExactMethod(parent, "getName", null, null);
+ }
+
+
+ /**
+ * <p> Test <code>invokeMethod</code> with a primitive.
+ */
+ public void testInvokeMethodWithPrimitives() throws Exception {
+ // first test that the bean works
+ PrimitiveBean bean = new PrimitiveBean();
+ bean.setFloat(20.0f);
+ bean.setLong(10l);
+ bean.setBoolean(true);
+ bean.setInt(12);
+ bean.setDouble(25.5d);
+
+ assertEquals("Bug in PrimitiveBean (1)", 20.0f, bean.getFloat(), 0.01f);
+ assertEquals("Bug in PrimitiveBean (2)", 10, bean.getLong());
+ assertEquals("Bug in PrimitiveBean (3)", true, bean.getBoolean());
+ assertEquals("Bug in PrimitiveBean (4)", 12, bean.getInt());
+ assertEquals("Bug in PrimitiveBean (5)", 25.5d, bean.getDouble(), 0.01f);
+
+ bean = new PrimitiveBean();
+ MethodUtils.invokeMethod(bean, "setBoolean", new Boolean(true));
+ assertEquals("Call boolean property using invokeMethod", true, bean.getBoolean());
+
+ bean = new PrimitiveBean();
+ MethodUtils.invokeMethod(bean, "setFloat", new Float(20.0f));
+ assertEquals("Call float property using invokeMethod", 20.0f, bean.getFloat(), 0.01f);
+
+ bean = new PrimitiveBean();
+ MethodUtils.invokeMethod(bean, "setLong", new Long(10));
+ assertEquals("Call float property using invokeMethod", 10, bean.getLong());
+
+ bean = new PrimitiveBean();
+ MethodUtils.invokeMethod(bean, "setInt", new Integer(12));
+ assertEquals("Set float property using invokeMethod", 12, bean.getInt());
+
+ bean = new PrimitiveBean();
+ MethodUtils.invokeMethod(bean, "setDouble", new Double(25.5d));
+ assertEquals("Set float property using invokeMethod", 25.5d, bean.getDouble(), 0.01d);
+ }
+
+
+ /**
+ * Simple tests for accessing static methods via invokeMethod().
+ */
+ public void testSimpleStatic1() {
+
+ TestBean bean = new TestBean();
+ Object value = null;
+ int current = TestBean.currentCounter();
+
+ try {
+
+ // Return initial value of the counter
+ value = MethodUtils.invokeMethod
+ (bean, "currentCounter", new Object[0], new Class[0]);
+ assertNotNull("currentCounter exists", value);
+ assertTrue("currentCounter type",
+ value instanceof Integer);
+ assertEquals("currentCounter value",
+ current,
+ ((Integer) value).intValue());
+
+ // Increment via no-arguments version
+ MethodUtils.invokeMethod
+ (bean, "incrementCounter", new Object[0], new Class[0]);
+
+ // Validate updated value
+ current++;
+ value = MethodUtils.invokeMethod
+ (bean, "currentCounter", new Object[0], new Class[0]);
+ assertNotNull("currentCounter exists", value);
+ assertTrue("currentCounter type",
+ value instanceof Integer);
+ assertEquals("currentCounter value",
+ current,
+ ((Integer) value).intValue());
+
+ // Increment via specified-argument version
+ MethodUtils.invokeMethod
+ (bean, "incrementCounter",
+ new Object[] { new Integer(5) },
+ new Class[] { Integer.TYPE });
+
+ // Validate updated value
+ current += 5;
+ value = MethodUtils.invokeMethod
+ (bean, "currentCounter", new Object[0], new Class[0]);
+ assertNotNull("currentCounter exists", value);
+ assertTrue("currentCounter type",
+ value instanceof Integer);
+ assertEquals("currentCounter value",
+ current,
+ ((Integer) value).intValue());
+
+ } catch (Exception e) {
+ fail("Threw exception" + e);
+ }
+
+ }
+
+
+ /**
+ * Simple tests for accessing static methods via invokeExactMethod().
+ */
+ public void testSimpleStatic2() {
+
+ TestBean bean = new TestBean();
+ Object value = null;
+ int current = TestBean.currentCounter();
+
+ try {
+
+ // Return initial value of the counter
+ value = MethodUtils.invokeExactMethod
+ (bean, "currentCounter", new Object[0], new Class[0]);
+ assertNotNull("currentCounter exists", value);
+ assertTrue("currentCounter type",
+ value instanceof Integer);
+ assertEquals("currentCounter value",
+ current,
+ ((Integer) value).intValue());
+
+ // Increment via no-arguments version
+ MethodUtils.invokeExactMethod
+ (bean, "incrementCounter", new Object[0], new Class[0]);
+
+ // Validate updated value
+ current++;
+ value = MethodUtils.invokeExactMethod
+ (bean, "currentCounter", new Object[0], new Class[0]);
+ assertNotNull("currentCounter exists", value);
+ assertTrue("currentCounter type",
+ value instanceof Integer);
+ assertEquals("currentCounter value",
+ current,
+ ((Integer) value).intValue());
+
+ // Increment via specified-argument version
+ MethodUtils.invokeExactMethod
+ (bean, "incrementCounter",
+ new Object[] { new Integer(5) },
+ new Class[] { Integer.TYPE });
+
+ // Validate updated value
+ current += 5;
+ value = MethodUtils.invokeExactMethod
+ (bean, "currentCounter", new Object[0], new Class[0]);
+ assertNotNull("currentCounter exists", value);
+ assertTrue("currentCounter type",
+ value instanceof Integer);
+ assertEquals("currentCounter value",
+ current,
+ ((Integer) value).intValue());
+
+
+ } catch (Exception e) {
+ fail("Threw exception" + e);
+ }
+
+ }
+
+
+ /**
+ * Simple tests for accessing static methods via getAccessibleMethod()
+ */
+ public void testSimpleStatic3() {
+
+ Object value = null;
+ int current = TestBean.currentCounter();
+
+ try {
+
+ // Acquire the methods we need
+ Method currentCounterMethod = MethodUtils.getAccessibleMethod
+ (TestBean.class, "currentCounter",
+ new Class[0]);
+ assertNotNull("currentCounterMethod exists",
+ currentCounterMethod);
+ assertEquals("currentCounterMethod name",
+ "currentCounter",
+ currentCounterMethod.getName());
+ assertEquals("currentCounterMethod args",
+ 0,
+ currentCounterMethod.getParameterTypes().length);
+ assertTrue("currentCounterMethod public",
+ Modifier.isPublic(currentCounterMethod.getModifiers()));
+ assertTrue("currentCounterMethod static",
+ Modifier.isStatic(currentCounterMethod.getModifiers()));
+ Method incrementCounterMethod1 = MethodUtils.getAccessibleMethod
+ (TestBean.class, "incrementCounter",
+ new Class[0]);
+ assertNotNull("incrementCounterMethod1 exists",
+ incrementCounterMethod1);
+ assertEquals("incrementCounterMethod1 name",
+ "incrementCounter",
+ incrementCounterMethod1.getName());
+ assertEquals("incrementCounterMethod1 args",
+ 0,
+ incrementCounterMethod1.getParameterTypes().length);
+ assertTrue("incrementCounterMethod1 public",
+ Modifier.isPublic(incrementCounterMethod1.getModifiers()));
+ assertTrue("incrementCounterMethod1 static",
+ Modifier.isStatic(incrementCounterMethod1.getModifiers()));
+ Method incrementCounterMethod2 = MethodUtils.getAccessibleMethod
+ (TestBean.class, "incrementCounter",
+ new Class[] { Integer.TYPE });
+ assertNotNull("incrementCounterMethod2 exists",
+ incrementCounterMethod2);
+ assertEquals("incrementCounterMethod2 name",
+ "incrementCounter",
+ incrementCounterMethod2.getName());
+ assertEquals("incrementCounterMethod2 args",
+ 1,
+ incrementCounterMethod2.getParameterTypes().length);
+ assertTrue("incrementCounterMethod2 public",
+ Modifier.isPublic(incrementCounterMethod2.getModifiers()));
+ assertTrue("incrementCounterMethod2 static",
+ Modifier.isStatic(incrementCounterMethod2.getModifiers()));
+
+ // Return initial value of the counter
+ value = currentCounterMethod.invoke(null, new Object[0]);
+ assertNotNull("currentCounter exists", value);
+ assertTrue("currentCounter type",
+ value instanceof Integer);
+ assertEquals("currentCounter value",
+ current,
+ ((Integer) value).intValue());
+
+ // Increment via no-arguments version
+ incrementCounterMethod1.invoke(null, new Object[0]);
+
+ // Validate updated value
+ current++;
+ value = currentCounterMethod.invoke(null, new Object[0]);
+ assertNotNull("currentCounter exists", value);
+ assertTrue("currentCounter type",
+ value instanceof Integer);
+ assertEquals("currentCounter value",
+ current,
+ ((Integer) value).intValue());
+
+ // Increment via specified-argument version
+ incrementCounterMethod2.invoke(null,
+ new Object[] { new Integer(5) });
+
+ // Validate updated value
+ current += 5;
+ value = currentCounterMethod.invoke(null, new Object[0]);
+ assertNotNull("currentCounter exists", value);
+ assertTrue("currentCounter type",
+ value instanceof Integer);
+ assertEquals("currentCounter value",
+ current,
+ ((Integer) value).intValue());
+
+ } catch (Exception e) {
+ fail("Threw exception" + e);
+ }
+
+ }
+
+ public void testPublicSub() throws Exception {
+ // make sure that bean does what it should
+ PublicSubBean bean = new PublicSubBean();
+ assertEquals("Start value (foo)", bean.getFoo(), "This is foo");
+ assertEquals("Start value (bar)", bean.getBar(), "This is bar");
+ bean.setFoo("new foo");
+ bean.setBar("new bar");
+ assertEquals("Set value (foo)", bean.getFoo(), "new foo");
+ assertEquals("Set value (bar)", bean.getBar(), "new bar");
+
+ // see if we can access public methods in a default access superclass
+ // from a public access subclass instance
+ MethodUtils.invokeMethod(bean, "setFoo", "alpha");
+ assertEquals("Set value (foo:2)", bean.getFoo(), "alpha");
+ MethodUtils.invokeMethod(bean, "setBar", "beta");
+ assertEquals("Set value (bar:2)", bean.getFoo(), "alpha");
+ }
+
+ public void testParentMethod() throws Exception {
+ OutputStream os = new PrintStream(System.out);
+ PrintStream ps = new PrintStream(System.out);
+
+ A a = new A();
+ MethodUtils.invokeMethod(a, "foo", os);
+ assertTrue("Method Invoked(1)", a.called);
+
+ a = new A();
+ MethodUtils.invokeMethod(a, "foo", ps);
+ assertTrue("Method Invoked(2)", a.called);
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/NestedTestBean.java b/trunk/src/test/org/apache/commons/beanutils/NestedTestBean.java
new file mode 100644
index 0000000..ce5cd97
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/NestedTestBean.java
@@ -0,0 +1,107 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+
+/**
+ * Specialist test bean for complex nested properties.
+ *
+ * @author Robert Burrell Donkin
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class NestedTestBean {
+
+
+ // ------------------------------------------------------------- Constructors
+ public NestedTestBean(String name) {
+ setName(name);
+ }
+
+
+ // ------------------------------------------------------------- Properties
+
+ private String name;
+
+ public String getName() {
+ return name;
+ }
+
+ public void setName(String name) {
+ this.name = name;
+ }
+
+
+ private String testString = "NOT SET";
+
+ public String getTestString() {
+ return testString;
+ }
+
+ public void setTestString(String testString) {
+ this.testString = testString;
+ }
+
+
+ private boolean testBoolean = false;
+
+ public boolean getTestBoolean() {
+ return testBoolean;
+ }
+
+ public void setTestBoolean(boolean testBoolean) {
+ this.testBoolean = testBoolean;
+ }
+
+
+ private NestedTestBean indexedBeans[];
+
+ public void init() {
+ indexedBeans = new NestedTestBean[5];
+ indexedBeans[0] = new NestedTestBean("Bean at 0");
+ indexedBeans[1] = new NestedTestBean("Bean at 1");
+ indexedBeans[2] = new NestedTestBean("Bean at 2");
+ indexedBeans[3] = new NestedTestBean("Bean at 3");
+ indexedBeans[4] = new NestedTestBean("Bean at 4");
+
+ simpleBean = new NestedTestBean("Simple Property Bean");
+ };
+
+ public NestedTestBean getIndexedProperty(int index) {
+ return (this.indexedBeans[index]);
+ }
+
+ public void setIndexedProperty(int index, NestedTestBean value) {
+ this.indexedBeans[index] = value;
+ }
+
+ private NestedTestBean simpleBean;
+
+ public NestedTestBean getSimpleBeanProperty() {
+ return simpleBean;
+ }
+
+ // ------------------------------------------------------- Static Variables
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/PassTestException.java b/trunk/src/test/org/apache/commons/beanutils/PassTestException.java
new file mode 100644
index 0000000..6212743
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/PassTestException.java
@@ -0,0 +1,27 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+
+/**
+ * Just a runtime exception. Useful for check that a method is called.
+ *
+ * @author Robert Burrell Donkin
+ */
+
+public class PassTestException extends RuntimeException {
+}
\ No newline at end of file
diff --git a/trunk/src/test/org/apache/commons/beanutils/PrimitiveBean.java b/trunk/src/test/org/apache/commons/beanutils/PrimitiveBean.java
new file mode 100644
index 0000000..508f00d
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/PrimitiveBean.java
@@ -0,0 +1,69 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+/**
+ * Bean that has primitive properties
+ */
+public class PrimitiveBean {
+
+ private float _float;
+ private double _double;
+ private boolean _boolean;
+ private long _long;
+ private int _int;
+
+ public float getFloat() {
+ return _float;
+ }
+
+ public void setFloat(float _float) {
+ this._float = _float;
+ }
+
+ public double getDouble() {
+ return _double;
+ }
+
+ public void setDouble(double _double) {
+ this._double = _double;
+ }
+
+ public boolean getBoolean() {
+ return _boolean;
+ }
+
+ public void setBoolean(boolean _boolean) {
+ this._boolean = _boolean;
+ }
+
+ public long getLong() {
+ return _long;
+ }
+
+ public void setLong(long _long) {
+ this._long = _long;
+ }
+
+ public int getInt() {
+ return _int;
+ }
+
+ public void setInt(int _int) {
+ this._int = _int;
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/PropertyUtilsBenchCase.java b/trunk/src/test/org/apache/commons/beanutils/PropertyUtilsBenchCase.java
new file mode 100644
index 0000000..a4ea6a8
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/PropertyUtilsBenchCase.java
@@ -0,0 +1,266 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.Map;
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * JUnit Test Case containing microbenchmarks for PropertyUtils.
+ */
+
+public class PropertyUtilsBenchCase extends TestCase {
+
+
+ // ------------------------------------------------------------ Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public PropertyUtilsBenchCase(String name) {
+
+ super(name);
+
+ }
+
+
+ // ------------------------------------------------------ Instance Variables
+
+
+ // Basic loop counter
+ private long counter = 100000;
+
+ // DynaClass for inDyna and outDyna
+ private DynaClass dynaClass = null;
+
+ // Input objects that have identical sets of properties and values.
+ private BenchBean inBean = null;
+ private DynaBean inDyna = null;
+ private Map inMap = null;
+
+ // Output objects that have identical sets of properties.
+ private BenchBean outBean = null;
+ private DynaBean outDyna = null;
+
+ // PropertyUtilsBean instance to be used
+ private PropertyUtilsBean pu = null;
+
+
+ // ---------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+
+ // Set up loop counter (if property specified)
+ String prop = System.getProperty("counter");
+ if (prop != null) {
+ counter = Long.parseLong(prop);
+ }
+
+ // Set up DynaClass for our DynaBean instances
+ dynaClass = new BasicDynaClass
+ ("BenchDynaClass", null,
+ new DynaProperty[]{
+ new DynaProperty("booleanProperty", Boolean.TYPE),
+ new DynaProperty("byteProperty", Byte.TYPE),
+ new DynaProperty("doubleProperty", Double.TYPE),
+ new DynaProperty("floatProperty", Float.TYPE),
+ new DynaProperty("intProperty", Integer.TYPE),
+ new DynaProperty("longProperty", Long.TYPE),
+ new DynaProperty("shortProperty", Short.TYPE),
+ new DynaProperty("stringProperty", String.class),
+ });
+
+ // Create input instances
+ inBean = new BenchBean();
+ inMap = new HashMap();
+ inMap.put("booleanProperty", new Boolean(inBean.getBooleanProperty()));
+ inMap.put("byteProperty", new Byte(inBean.getByteProperty()));
+ inMap.put("doubleProperty", new Double(inBean.getDoubleProperty()));
+ inMap.put("floatProperty", new Float(inBean.getFloatProperty()));
+ inMap.put("intProperty", new Integer(inBean.getIntProperty()));
+ inMap.put("longProperty", new Long(inBean.getLongProperty()));
+ inMap.put("shortProperty", new Short(inBean.getShortProperty()));
+ inMap.put("stringProperty", inBean.getStringProperty());
+ inDyna = dynaClass.newInstance();
+ Iterator inKeys = inMap.keySet().iterator();
+ while (inKeys.hasNext()) {
+ String inKey = (String) inKeys.next();
+ inDyna.set(inKey, inMap.get(inKey));
+ }
+
+ // Create output instances
+ outBean = new BenchBean();
+ outDyna = dynaClass.newInstance();
+ Iterator outKeys = inMap.keySet().iterator();
+ while (outKeys.hasNext()) {
+ String outKey = (String) outKeys.next();
+ outDyna.set(outKey, inMap.get(outKey));
+ }
+
+ // Set up PropertyUtilsBean instance we will use
+ pu = PropertyUtilsBean.getInstance();
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+
+ return (new TestSuite(PropertyUtilsBenchCase.class));
+
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+
+ dynaClass = null;
+ inBean = null;
+ inDyna = null;
+ inMap = null;
+ outBean = null;
+ outDyna = null;
+ pu = null;
+
+ }
+
+
+
+ // ------------------------------------------------- Individual Test Methods
+
+
+ // Time copyProperties() from a bean
+ public void testCopyPropertiesBean() throws Exception {
+
+ long start;
+ long stop;
+
+ // Bean->Bean
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outBean, inBean);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outBean, inBean);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("PU.copyProperties(bean,bean), count=" + counter +
+ ", time=" + (stop - start));
+
+ // Bean->Dyna
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outDyna, inBean);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outDyna, inBean);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("PU.copyProperties(dyna,bean), count=" + counter +
+ ", time=" + (stop - start));
+
+ }
+
+
+ // Time copyProperties() from a DynaBean
+ public void testCopyPropertiesDyna() throws Exception {
+
+ long start;
+ long stop;
+
+ // Dyna->Bean
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outBean, inDyna);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outBean, inDyna);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("PU.copyProperties(bean,dyna), count=" + counter +
+ ", time=" + (stop - start));
+
+ // Dyna->Dyna
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outDyna, inDyna);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outDyna, inDyna);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("PU.copyProperties(dyna,dyna), count=" + counter +
+ ", time=" + (stop - start));
+
+ }
+
+
+ // Time copyProperties() from a Map
+ public void testCopyPropertiesMap() throws Exception {
+
+ long start;
+ long stop;
+
+ // Dyna->Bean
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outBean, inMap);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outBean, inMap);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("PU.copyProperties(bean, map), count=" + counter +
+ ", time=" + (stop - start));
+
+ // Dyna->Dyna
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outDyna, inMap);
+ }
+ start = System.currentTimeMillis();
+ for (long i = 0; i < counter; i++) {
+ pu.copyProperties(outDyna, inMap);
+ }
+ stop = System.currentTimeMillis();
+ System.err.println("PU.copyProperties(dyna, map), count=" + counter +
+ ", time=" + (stop - start));
+
+ }
+
+
+ // --------------------------------------------------------- Support Methods
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/PropertyUtilsTestCase.java b/trunk/src/test/org/apache/commons/beanutils/PropertyUtilsTestCase.java
new file mode 100755
index 0000000..131dea7
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/PropertyUtilsTestCase.java
@@ -0,0 +1,3665 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.beans.PropertyDescriptor;
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Method;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import org.apache.commons.beanutils.priv.PrivateBeanFactory;
+import org.apache.commons.beanutils.priv.PrivateDirect;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * <p>Test Case for the PropertyUtils class. The majority of these tests use
+ * instances of the TestBean class, so be sure to update the tests if you
+ * change the characteristics of that class.</p>
+ *
+ * <p>So far, this test case has tests for the following methods of the
+ * <code>PropertyUtils</code> class:</p>
+ * <ul>
+ * <li>getIndexedProperty(Object,String)</li>
+ * <li>getIndexedProperty(Object,String,int)</li>
+ * <li>getMappedProperty(Object,String)</li>
+ * <li>getMappedProperty(Object,String,String</li>
+ * <li>getNestedProperty(Object,String)</li>
+ * <li>getPropertyDescriptor(Object,String)</li>
+ * <li>getPropertyDescriptors(Object)</li>
+ * <li>getPropertyType(Object,String)</li>
+ * <li>getSimpleProperty(Object,String)</li>
+ * <li>setIndexedProperty(Object,String,Object)</li>
+ * <li>setIndexedProperty(Object,String,String,Object)</li>
+ * <li>setMappedProperty(Object,String,Object)</li>
+ * <li>setMappedProperty(Object,String,String,Object)</li>
+ * <li>setNestedProperty(Object,String,Object)</li>
+ * <li>setSimpleProperty(Object,String,Object)</li>
+ * </ul>
+ *
+ * @author Craig R. McClanahan
+ * @author Jan Sorensen
+ * @version $Revision: 1.34 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class PropertyUtilsTestCase extends TestCase {
+
+
+ // ---------------------------------------------------- Instance Variables
+
+
+ /**
+ * The fully qualified class name of our private directly
+ * implemented interface.
+ */
+ private static final String PRIVATE_DIRECT_CLASS =
+ "org.apache.commons.beanutils.priv.PrivateDirect";
+
+
+ /**
+ * The fully qualified class name of our private indirectly
+ * implemented interface.
+ */
+ private static final String PRIVATE_INDIRECT_CLASS =
+ "org.apache.commons.beanutils.priv.PrivateIndirect";
+
+
+ /**
+ * The fully qualified class name of our test bean class.
+ */
+ private static final String TEST_BEAN_CLASS =
+ "org.apache.commons.beanutils.TestBean";
+
+
+ /**
+ * The basic test bean for each test.
+ */
+ protected TestBean bean = null;
+
+
+ /**
+ * The "package private subclass" test bean for each test.
+ */
+ protected TestBeanPackageSubclass beanPackageSubclass = null;
+
+
+ /**
+ * The test bean for private access tests.
+ */
+ protected PrivateDirect beanPrivate = null;
+
+
+ /**
+ * The test bean for private access tests of subclasses.
+ */
+ protected PrivateDirect beanPrivateSubclass = null;
+
+
+ /**
+ * The "public subclass" test bean for each test.
+ */
+ protected TestBeanPublicSubclass beanPublicSubclass = null;
+
+
+ /**
+ * The set of properties that should be described.
+ */
+ protected String describes[] =
+ { "booleanProperty",
+ "booleanSecond",
+ "doubleProperty",
+ "floatProperty",
+ "intArray",
+ // "intIndexed",
+ "intProperty",
+ "listIndexed",
+ "longProperty",
+ // "mappedObjects",
+ // "mappedProperty",
+ // "mappedIntProperty",
+ "nested",
+ "nullProperty",
+ // "readOnlyProperty",
+ "shortProperty",
+ "stringArray",
+ // "stringIndexed",
+ "stringProperty"
+ };
+
+
+ /**
+ * The set of property names we expect to have returned when calling
+ * <code>getPropertyDescriptors()</code>. You should update this list
+ * when new properties are added to TestBean.
+ */
+ protected final static String[] properties = {
+ "booleanProperty",
+ "booleanSecond",
+ "doubleProperty",
+ "dupProperty",
+ "floatProperty",
+ "intArray",
+ "intIndexed",
+ "intProperty",
+ "listIndexed",
+ "longProperty",
+ "nested",
+ "nullProperty",
+ "readOnlyProperty",
+ "shortProperty",
+ "stringArray",
+ "stringIndexed",
+ "stringProperty",
+ "writeOnlyProperty",
+ };
+
+
+ // ---------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public PropertyUtilsTestCase(String name) {
+
+ super(name);
+
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() {
+
+ bean = new TestBean();
+ beanPackageSubclass = new TestBeanPackageSubclass();
+ beanPrivate = PrivateBeanFactory.create();
+ beanPrivateSubclass = PrivateBeanFactory.createSubclass();
+ beanPublicSubclass = new TestBeanPublicSubclass();
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+
+ return (new TestSuite(PropertyUtilsTestCase.class));
+
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+
+ bean = null;
+ beanPackageSubclass = null;
+ beanPrivate = null;
+ beanPrivateSubclass = null;
+ beanPublicSubclass = null;
+
+ }
+
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ /**
+ * Test copyProperties() when the origin is a a <code>Map</code>.
+ */
+ public void testCopyPropertiesMap() {
+
+ Map map = new HashMap();
+ map.put("booleanProperty", Boolean.FALSE);
+ map.put("doubleProperty", new Double(333.0));
+ map.put("dupProperty", new String[] { "New 0", "New 1", "New 2" });
+ map.put("floatProperty", new Float((float) 222.0));
+ map.put("intArray", new int[] { 0, 100, 200 });
+ map.put("intProperty", new Integer(111));
+ map.put("longProperty", new Long(444));
+ map.put("shortProperty", new Short((short) 555));
+ map.put("stringProperty", "New String Property");
+
+ try {
+ PropertyUtils.copyProperties(bean, map);
+ } catch (Throwable t) {
+ fail("Threw " + t.toString());
+ }
+
+ // Scalar properties
+ assertEquals("booleanProperty", false,
+ bean.getBooleanProperty());
+ assertEquals("doubleProperty", 333.0,
+ bean.getDoubleProperty(), 0.005);
+ assertEquals("floatProperty", (float) 222.0,
+ bean.getFloatProperty(), (float) 0.005);
+ assertEquals("intProperty", 111,
+ bean.getIntProperty());
+ assertEquals("longProperty", (long) 444,
+ bean.getLongProperty());
+ assertEquals("shortProperty", (short) 555,
+ bean.getShortProperty());
+ assertEquals("stringProperty", "New String Property",
+ bean.getStringProperty());
+
+ // Indexed Properties
+ String dupProperty[] = bean.getDupProperty();
+ assertNotNull("dupProperty present", dupProperty);
+ assertEquals("dupProperty length", 3, dupProperty.length);
+ assertEquals("dupProperty[0]", "New 0", dupProperty[0]);
+ assertEquals("dupProperty[1]", "New 1", dupProperty[1]);
+ assertEquals("dupProperty[2]", "New 2", dupProperty[2]);
+ int intArray[] = bean.getIntArray();
+ assertNotNull("intArray present", intArray);
+ assertEquals("intArray length", 3, intArray.length);
+ assertEquals("intArray[0]", 0, intArray[0]);
+ assertEquals("intArray[1]", 100, intArray[1]);
+ assertEquals("intArray[2]", 200, intArray[2]);
+
+ }
+
+
+ /**
+ * Test the describe() method.
+ */
+ public void testDescribe() {
+
+ Map map = null;
+ try {
+ map = PropertyUtils.describe(bean);
+ } catch (Exception e) {
+ fail("Threw exception " + e);
+ }
+
+ // Verify existence of all the properties that should be present
+ for (int i = 0; i < describes.length; i++) {
+ assertTrue("Property '" + describes[i] + "' is present",
+ map.containsKey(describes[i]));
+ }
+ assertTrue("Property 'writeOnlyProperty' is not present",
+ !map.containsKey("writeOnlyProperty"));
+
+ // Verify the values of scalar properties
+ assertEquals("Value of 'booleanProperty'",
+ Boolean.TRUE,
+ (Boolean) map.get("booleanProperty"));
+ assertEquals("Value of 'doubleProperty'",
+ new Double(321.0),
+ (Double) map.get("doubleProperty"));
+ assertEquals("Value of 'floatProperty'",
+ new Float((float) 123.0),
+ (Float) map.get("floatProperty"));
+ assertEquals("Value of 'intProperty'",
+ new Integer(123),
+ (Integer) map.get("intProperty"));
+ assertEquals("Value of 'longProperty'",
+ new Long(321),
+ (Long) map.get("longProperty"));
+ assertEquals("Value of 'shortProperty'",
+ new Short((short) 987),
+ (Short) map.get("shortProperty"));
+ assertEquals("Value of 'stringProperty'",
+ "This is a string",
+ (String) map.get("stringProperty"));
+
+ }
+
+
+ /**
+ * Corner cases on getPropertyDescriptor invalid arguments.
+ */
+ public void testGetDescriptorArguments() {
+
+ try {
+ PropertyUtils.getPropertyDescriptor(null, "stringProperty");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getPropertyDescriptor(bean, null);
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property <code>booleanProperty</code>.
+ */
+ public void testGetDescriptorBoolean() {
+
+ testGetDescriptorBase("booleanProperty", "getBooleanProperty",
+ "setBooleanProperty");
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property <code>doubleProperty</code>.
+ */
+ public void testGetDescriptorDouble() {
+
+ testGetDescriptorBase("doubleProperty", "getDoubleProperty",
+ "setDoubleProperty");
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property <code>floatProperty</code>.
+ */
+ public void testGetDescriptorFloat() {
+
+ testGetDescriptorBase("floatProperty", "getFloatProperty",
+ "setFloatProperty");
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property <code>intProperty</code>.
+ */
+ public void testGetDescriptorInt() {
+
+ testGetDescriptorBase("intProperty", "getIntProperty",
+ "setIntProperty");
+
+ }
+
+
+ /**
+ * <p>Negative tests on an invalid property with two different boolean
+ * getters (which is fine, according to the JavaBeans spec) but a
+ * String setter instead of a boolean setter.</p>
+ *
+ * <p>Although one could logically argue that this combination of method
+ * signatures should not identify a property at all, there is a sentence
+ * in Section 8.3.1 making it clear that the behavior tested for here
+ * is correct: "If we find only one of these methods, then we regard
+ * it as defining either a read-only or write-only property called
+ * <em><property-name></em>.</p>
+ */
+ public void testGetDescriptorInvalidBoolean() throws Exception {
+
+ PropertyDescriptor pd =
+ PropertyUtils.getPropertyDescriptor(bean, "invalidBoolean");
+ assertNotNull("invalidBoolean is a property", pd);
+ assertNotNull("invalidBoolean has a getter method",
+ pd.getReadMethod());
+ assertNull("invalidBoolean has no write method",
+ pd.getWriteMethod());
+ assertTrue("invalidBoolean getter method is isInvalidBoolean",
+ "isInvalidBoolean".equals(pd.getReadMethod().getName()));
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property <code>longProperty</code>.
+ */
+ public void testGetDescriptorLong() {
+
+ testGetDescriptorBase("longProperty", "getLongProperty",
+ "setLongProperty");
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property
+ * <code>readOnlyProperty</code>.
+ */
+ public void testGetDescriptorReadOnly() {
+
+ testGetDescriptorBase("readOnlyProperty", "getReadOnlyProperty",
+ null);
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property <code>booleanSecond</code>
+ * that uses an "is" method as the getter.
+ */
+ public void testGetDescriptorSecond() {
+
+ testGetDescriptorBase("booleanSecond", "isBooleanSecond",
+ "setBooleanSecond");
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property <code>shortProperty</code>.
+ */
+ public void testGetDescriptorShort() {
+
+ testGetDescriptorBase("shortProperty", "getShortProperty",
+ "setShortProperty");
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property <code>stringProperty</code>.
+ */
+ public void testGetDescriptorString() {
+
+ testGetDescriptorBase("stringProperty", "getStringProperty",
+ "setStringProperty");
+
+ }
+
+
+ /**
+ * Negative getPropertyDescriptor on property <code>unknown</code>.
+ */
+ public void testGetDescriptorUnknown() {
+
+ testGetDescriptorBase("unknown", null, null);
+
+ }
+
+
+ /**
+ * Positive getPropertyDescriptor on property
+ * <code>writeOnlyProperty</code>.
+ */
+ public void testGetDescriptorWriteOnly() {
+
+ testGetDescriptorBase("writeOnlyProperty", null,
+ "setWriteOnlyProperty");
+
+ }
+
+
+ /**
+ * Positive test for getPropertyDescriptors(). Each property name
+ * listed in <code>properties</code> should be returned exactly once.
+ */
+ public void testGetDescriptors() {
+
+ PropertyDescriptor pd[] =
+ PropertyUtils.getPropertyDescriptors(bean);
+ assertNotNull("Got descriptors", pd);
+ int count[] = new int[properties.length];
+ for (int i = 0; i < pd.length; i++) {
+ String name = pd[i].getName();
+ for (int j = 0; j < properties.length; j++) {
+ if (name.equals(properties[j]))
+ count[j]++;
+ }
+ }
+ for (int j = 0; j < properties.length; j++) {
+ if (count[j] < 0)
+ fail("Missing property " + properties[j]);
+ else if (count[j] > 1)
+ fail("Duplicate property " + properties[j]);
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getPropertyDescriptors invalid arguments.
+ */
+ public void testGetDescriptorsArguments() {
+
+ try {
+ PropertyUtils.getPropertyDescriptors(null);
+ fail("Should throw IllegalArgumentException");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException");
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getIndexedProperty invalid arguments.
+ */
+ public void testGetIndexedArguments() {
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.getIndexedProperty(null, "intArray", 0);
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, null, 0);
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.getIndexedProperty(null,
+ "intArray[0]");
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, "[0]");
+ fail("Should throw NoSuchMethodException 4");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 4");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, "intArray");
+ fail("Should throw IllegalArgumentException 5");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 5");
+ }
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.getIndexedProperty(null, "intIndexed", 0);
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, null, 0);
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.getIndexedProperty(null,
+ "intIndexed[0]");
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, "[0]");
+ fail("Should throw NoSuchMethodException 4");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 4");
+ }
+
+ try {
+ PropertyUtils.getIndexedProperty(bean, "intIndexed");
+ fail("Should throw IllegalArgumentException 5");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 5");
+ }
+
+ }
+
+
+ /**
+ * Positive and negative tests on getIndexedProperty valid arguments.
+ */
+ public void testGetIndexedValues() {
+
+ Object value = null;
+
+ // Use explicit key argument
+
+ for (int i = 0; i < 5; i++) {
+
+ try {
+ value = PropertyUtils.getIndexedProperty
+ (bean, "dupProperty", i);
+ assertNotNull("dupProperty returned value " + i, value);
+ assertTrue("dupProperty returned String " + i,
+ value instanceof String);
+ assertEquals("dupProperty returned correct " + i,
+ "Dup " + i,
+ (String) value);
+ } catch (Throwable t) {
+ fail("dupProperty " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "intArray", i);
+ assertNotNull("intArray returned value " + i, value);
+ assertTrue("intArray returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intArray returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intArray " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "intIndexed", i);
+ assertNotNull("intIndexed returned value " + i, value);
+ assertTrue("intIndexed returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intIndexed returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "listIndexed", i);
+ assertNotNull("listIndexed returned value " + i, value);
+ assertTrue("list returned String " + i,
+ value instanceof String);
+ assertEquals("listIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("listIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "stringArray", i);
+ assertNotNull("stringArray returned value " + i, value);
+ assertTrue("stringArray returned String " + i,
+ value instanceof String);
+ assertEquals("stringArray returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringArray " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean, "stringIndexed", i);
+ assertNotNull("stringIndexed returned value " + i, value);
+ assertTrue("stringIndexed returned String " + i,
+ value instanceof String);
+ assertEquals("stringIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringIndexed " + i + " threw " + t);
+ }
+
+ }
+
+ // Use key expression
+
+ for (int i = 0; i < 5; i++) {
+
+ try {
+ value = PropertyUtils.getIndexedProperty
+ (bean, "dupProperty[" + i + "]");
+ assertNotNull("dupProperty returned value " + i, value);
+ assertTrue("dupProperty returned String " + i,
+ value instanceof String);
+ assertEquals("dupProperty returned correct " + i,
+ "Dup " + i,
+ (String) value);
+ } catch (Throwable t) {
+ fail("dupProperty " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray[" + i + "]");
+ assertNotNull("intArray returned value " + i, value);
+ assertTrue("intArray returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intArray returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intArray " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed[" + i + "]");
+ assertNotNull("intIndexed returned value " + i, value);
+ assertTrue("intIndexed returned Integer " + i,
+ value instanceof Integer);
+ assertEquals("intIndexed returned correct " + i, i * 10,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("intIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed[" + i + "]");
+ assertNotNull("listIndexed returned value " + i, value);
+ assertTrue("listIndexed returned String " + i,
+ value instanceof String);
+ assertEquals("listIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("listIndexed " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray[" + i + "]");
+ assertNotNull("stringArray returned value " + i, value);
+ assertTrue("stringArray returned String " + i,
+ value instanceof String);
+ assertEquals("stringArray returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringArray " + i + " threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringIndexed[" + i + "]");
+ assertNotNull("stringIndexed returned value " + i, value);
+ assertTrue("stringIndexed returned String " + i,
+ value instanceof String);
+ assertEquals("stringIndexed returned correct " + i,
+ "String " + i, (String) value);
+ } catch (Throwable t) {
+ fail("stringIndexed " + i + " threw " + t);
+ }
+
+ }
+
+ // Index out of bounds tests
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "dupProperty", -1);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "dupProperty", 5);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray", -1);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray", 5);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed", -1);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed", 5);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed", -1);
+ fail("Should have thrown IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed", 5);
+ fail("Should have thrown IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray", -1);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray", 5);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringIndexed", -1);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringIndexed", 5);
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getMappedProperty invalid arguments.
+ */
+ public void testGetMappedArguments() {
+
+ // Use explicit key argument
+
+ try {
+ PropertyUtils.getMappedProperty(null, "mappedProperty",
+ "First Key");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getMappedProperty(bean, null, "First Key");
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ try {
+ PropertyUtils.getMappedProperty(bean, "mappedProperty", null);
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ // Use key expression
+
+ try {
+ PropertyUtils.getMappedProperty(null,
+ "mappedProperty(First Key)");
+ fail("Should throw IllegalArgumentException 4");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 4");
+ }
+
+ try {
+ PropertyUtils.getMappedProperty(bean, "(Second Key)");
+ fail("Should throw IllegalArgumentException 5");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 5");
+ }
+
+ try {
+ PropertyUtils.getMappedProperty(bean, "mappedProperty");
+ fail("Should throw IllegalArgumentException 6");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 6");
+ }
+
+ }
+
+
+ /**
+ * Test getting mapped values with periods in the key.
+ */
+ public void testGetMappedPeriods() {
+
+ bean.setMappedProperty("key.with.a.dot", "Special Value");
+ assertEquals("Can retrieve directly",
+ "Special Value",
+ bean.getMappedProperty("key.with.a.dot"));
+ try {
+ assertEquals("Can retrieve via getMappedProperty",
+ "Special Value",
+ PropertyUtils.getMappedProperty
+ (bean, "mappedProperty", "key.with.a.dot"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+ try {
+ assertEquals("Can retrieve via getNestedProperty",
+ "Special Value",
+ PropertyUtils.getNestedProperty
+ (bean, "mappedProperty(key.with.a.dot)"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+
+ bean.setMappedObjects("nested.property", new TestBean());
+ assertNotNull("Can retrieve directly",
+ bean.getMappedObjects("nested.property"));
+ try {
+ assertEquals("Can retrieve nested",
+ "This is a string",
+ PropertyUtils.getNestedProperty
+ (bean,
+ "mappedObjects(nested.property).stringProperty"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+
+ try
+ {
+ assertEquals("Can't retrieved nested with mapped property",
+ "Mapped Value",
+ PropertyUtils.getNestedProperty(
+ bean,"mappedNested.value(Mapped Key)"));
+ } catch (Exception e)
+ {
+ fail("Thew exception: " + e);
+ }
+ }
+
+
+ /**
+ * Test getting mapped values with slashes in the key. This is different
+ * from periods because slashes are not syntactically significant.
+ */
+ public void testGetMappedSlashes() {
+
+ bean.setMappedProperty("key/with/a/slash", "Special Value");
+ assertEquals("Can retrieve directly",
+ "Special Value",
+ bean.getMappedProperty("key/with/a/slash"));
+ try {
+ assertEquals("Can retrieve via getMappedProperty",
+ "Special Value",
+ PropertyUtils.getMappedProperty
+ (bean, "mappedProperty", "key/with/a/slash"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+ try {
+ assertEquals("Can retrieve via getNestedProperty",
+ "Special Value",
+ PropertyUtils.getNestedProperty
+ (bean, "mappedProperty(key/with/a/slash)"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+
+ bean.setMappedObjects("nested/property", new TestBean());
+ assertNotNull("Can retrieve directly",
+ bean.getMappedObjects("nested/property"));
+ try {
+ assertEquals("Can retrieve nested",
+ "This is a string",
+ PropertyUtils.getNestedProperty
+ (bean,
+ "mappedObjects(nested/property).stringProperty"));
+ } catch (Exception e) {
+ fail("Thew exception: " + e);
+ }
+
+ }
+
+
+ /**
+ * Positive and negative tests on getMappedProperty valid arguments.
+ */
+ public void testGetMappedValues() {
+
+ Object value = null;
+
+ // Use explicit key argument
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "First Key");
+ assertEquals("Can find first value", "First Value", value);
+ } catch (Throwable t) {
+ fail("Finding first value threw " + t);
+ }
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "Second Key");
+ assertEquals("Can find second value", "Second Value", value);
+ } catch (Throwable t) {
+ fail("Finding second value threw " + t);
+ }
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "Third Key");
+ assertNull("Can not find third value", value);
+ } catch (Throwable t) {
+ fail("Finding third value threw " + t);
+ }
+
+ // Use key expression with parentheses
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(First Key)");
+ assertEquals("Can find first value", "First Value", value);
+ } catch (Throwable t) {
+ fail("Finding first value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(Second Key)");
+ assertEquals("Can find second value", "Second Value", value);
+ } catch (Throwable t) {
+ fail("Finding second value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(Third Key)");
+ assertNull("Can not find third value", value);
+ } catch (Throwable t) {
+ fail("Finding third value threw " + t);
+ }
+
+ // Use key expression with dotted syntax
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.First Key");
+ assertEquals("Can find first value", "First Value", value);
+ } catch (Throwable t) {
+ fail("Finding first value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.Second Key");
+ assertEquals("Can find second value", "Second Value", value);
+ } catch (Throwable t) {
+ fail("Finding second value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.Third Key");
+ assertNull("Can not find third value", value);
+ } catch (Throwable t) {
+ fail("Finding third value threw " + t);
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getNestedProperty invalid arguments.
+ */
+ public void testGetNestedArguments() {
+
+ try {
+ PropertyUtils.getNestedProperty(null, "stringProperty");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getNestedProperty(bean, null);
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a boolean property.
+ */
+ public void testGetNestedBoolean() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.booleanProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Boolean));
+ assertTrue("Got correct value",
+ ((Boolean) value).booleanValue() ==
+ bean.getNested().getBooleanProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a double property.
+ */
+ public void testGetNestedDouble() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.doubleProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Double));
+ assertEquals("Got correct value",
+ ((Double) value).doubleValue(),
+ bean.getNested().getDoubleProperty(),
+ 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a float property.
+ */
+ public void testGetNestedFloat() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.floatProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Float));
+ assertEquals("Got correct value",
+ ((Float) value).floatValue(),
+ bean.getNested().getFloatProperty(),
+ (float) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on an int property.
+ */
+ public void testGetNestedInt() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.intProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Integer));
+ assertEquals("Got correct value",
+ ((Integer) value).intValue(),
+ bean.getNested().getIntProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a long property.
+ */
+ public void testGetNestedLong() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.longProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Long));
+ assertEquals("Got correct value",
+ ((Long) value).longValue(),
+ bean.getNested().getLongProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a read-only String property.
+ */
+ public void testGetNestedReadOnly() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.readOnlyProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof String));
+ assertEquals("Got correct value",
+ (String) value,
+ bean.getReadOnlyProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a short property.
+ */
+ public void testGetNestedShort() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.shortProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Short));
+ assertEquals("Got correct value",
+ ((Short) value).shortValue(),
+ bean.getNested().getShortProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getNestedProperty on a String property.
+ */
+ public void testGetNestedString() {
+
+ try {
+ Object value =
+ PropertyUtils.getNestedProperty
+ (bean, "nested.stringProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof String));
+ assertEquals("Got correct value",
+ ((String) value),
+ bean.getNested().getStringProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test getNestedProperty on an unknown property.
+ */
+ public void testGetNestedUnknown() {
+
+ try {
+ PropertyUtils.getNestedProperty(bean, "nested.unknown");
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+ /**
+ * When a bean has a null property which is reference by the standard access language,
+ * this should throw a NestedNullException.
+ */
+ public void testThrowNestedNull() throws Exception {
+ NestedTestBean nestedBean = new NestedTestBean("base");
+ // don't init!
+
+ try {
+ NestedTestBean value = (NestedTestBean) PropertyUtils.getProperty(
+ nestedBean,
+ "simpleBeanProperty.indexedProperty[0]");
+ fail("NestedNullException not thrown");
+ } catch (NestedNullException e) {
+ // that's what we wanted!
+ }
+ }
+
+ /**
+ * Test getNestedProperty on a write-only String property.
+ */
+ public void testGetNestedWriteOnly() {
+
+ try {
+ PropertyUtils.getNestedProperty(bean, "writeOnlyProperty");
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Test getPropertyType() on all kinds of properties.
+ */
+ public void testGetPropertyType() {
+
+ Class clazz = null;
+ int intArray[] = new int[0];
+ String stringArray[] = new String[0];
+
+ try {
+
+ // Scalar and Indexed Properties
+ clazz = PropertyUtils.getPropertyType(bean, "booleanProperty");
+ assertEquals("booleanProperty type", Boolean.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "booleanSecond");
+ assertEquals("booleanSecond type", Boolean.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "doubleProperty");
+ assertEquals("doubleProperty type", Double.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "dupProperty");
+ assertEquals("dupProperty type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "floatProperty");
+ assertEquals("floatProperty type", Float.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "intArray");
+ assertEquals("intArray type", intArray.getClass(), clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "intIndexed");
+ assertEquals("intIndexed type", Integer.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "intProperty");
+ assertEquals("intProperty type", Integer.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "listIndexed");
+ assertEquals("listIndexed type", List.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "longProperty");
+ assertEquals("longProperty type", Long.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "mappedProperty");
+ assertEquals("mappedProperty type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "mappedIntProperty");
+ assertEquals("mappedIntProperty type", Integer.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "readOnlyProperty");
+ assertEquals("readOnlyProperty type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "shortProperty");
+ assertEquals("shortProperty type", Short.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "stringArray");
+ assertEquals("stringArray type", stringArray.getClass(), clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "stringIndexed");
+ assertEquals("stringIndexed type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "stringProperty");
+ assertEquals("stringProperty type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "writeOnlyProperty");
+ assertEquals("writeOnlyProperty type", String.class, clazz);
+
+ // Nested Properties
+ clazz = PropertyUtils.getPropertyType(bean, "nested.booleanProperty");
+ assertEquals("booleanProperty type", Boolean.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.booleanSecond");
+ assertEquals("booleanSecond type", Boolean.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.doubleProperty");
+ assertEquals("doubleProperty type", Double.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.dupProperty");
+ assertEquals("dupProperty type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.floatProperty");
+ assertEquals("floatProperty type", Float.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.intArray");
+ assertEquals("intArray type", intArray.getClass(), clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.intIndexed");
+ assertEquals("intIndexed type", Integer.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.intProperty");
+ assertEquals("intProperty type", Integer.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.listIndexed");
+ assertEquals("listIndexed type", List.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.longProperty");
+ assertEquals("longProperty type", Long.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.mappedProperty");
+ assertEquals("mappedProperty type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.mappedIntProperty");
+ assertEquals("mappedIntProperty type", Integer.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.readOnlyProperty");
+ assertEquals("readOnlyProperty type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.shortProperty");
+ assertEquals("shortProperty type", Short.TYPE, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.stringArray");
+ assertEquals("stringArray type", stringArray.getClass(), clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.stringIndexed");
+ assertEquals("stringIndexed type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.stringProperty");
+ assertEquals("stringProperty type", String.class, clazz);
+ clazz = PropertyUtils.getPropertyType(bean, "nested.writeOnlyProperty");
+ assertEquals("writeOnlyProperty type", String.class, clazz);
+
+ } catch (Exception e) {
+ fail("Exception: " + e.getMessage());
+ }
+
+ }
+
+
+ /**
+ * Test getting accessible property reader methods for a specified
+ * list of properties of our standard test bean.
+ */
+ public void testGetReadMethodBasic() {
+
+ testGetReadMethod(bean, properties, TEST_BEAN_CLASS);
+
+ }
+
+
+ /**
+ * Test getting accessible property reader methods for a specified
+ * list of properties of a package private subclass of our standard
+ * test bean.
+ */
+ public void testGetReadMethodPackageSubclass() {
+
+ testGetReadMethod(beanPackageSubclass, properties, TEST_BEAN_CLASS);
+
+ }
+
+
+ /**
+ * Test getting accessible property reader methods for a specified
+ * list of properties that are declared either directly or via
+ * implemented interfaces.
+ */
+ public void testGetReadMethodPublicInterface() {
+
+ // Properties "bar" and "baz" are visible via implemented interfaces
+ // (one direct and one indirect)
+ testGetReadMethod(beanPrivate,
+ new String[]{ "bar" },
+ PRIVATE_DIRECT_CLASS);
+ testGetReadMethod(beanPrivate,
+ new String[]{ "baz" },
+ PRIVATE_INDIRECT_CLASS);
+
+ // Properties "bar" and "baz" are visible via implemented interfaces
+ // (one direct and one indirect). The interface is implemented in
+ // a superclass
+ testGetReadMethod(beanPrivateSubclass,
+ new String[]{ "bar" },
+ PRIVATE_DIRECT_CLASS);
+ testGetReadMethod(beanPrivateSubclass,
+ new String[]{ "baz" },
+ PRIVATE_INDIRECT_CLASS);
+
+ // Property "foo" is not accessible because the underlying
+ // class has package scope
+ PropertyDescriptor pd[] =
+ PropertyUtils.getPropertyDescriptors(beanPrivate);
+ int n = -1;
+ for (int i = 0; i < pd.length; i++) {
+ if ("foo".equals(pd[i].getName())) {
+ n = i;
+ break;
+ }
+ }
+ assertTrue("Found foo descriptor", n >= 0);
+ Method reader = pd[n].getReadMethod();
+ assertNotNull("Found foo read method", reader);
+ Object value = null;
+ try {
+ value = reader.invoke(beanPrivate, new Class[0]);
+ fail("Foo reader did throw IllegalAccessException");
+ } catch (IllegalAccessException e) {
+ ; // Expected result for this test
+ } catch (Throwable t) {
+ fail("Invoke foo reader: " + t);
+ }
+
+ }
+
+
+ /**
+ * Test getting accessible property reader methods for a specified
+ * list of properties of a public subclass of our standard test bean.
+ */
+ public void testGetReadMethodPublicSubclass() {
+
+ testGetReadMethod(beanPublicSubclass, properties, TEST_BEAN_CLASS);
+
+ }
+
+
+ /**
+ * Corner cases on getSimpleProperty invalid arguments.
+ */
+ public void testGetSimpleArguments() {
+
+ try {
+ PropertyUtils.getSimpleProperty(null, "stringProperty");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.getSimpleProperty(bean, null);
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a boolean property.
+ */
+ public void testGetSimpleBoolean() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "booleanProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Boolean));
+ assertTrue("Got correct value",
+ ((Boolean) value).booleanValue() ==
+ bean.getBooleanProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a double property.
+ */
+ public void testGetSimpleDouble() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "doubleProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Double));
+ assertEquals("Got correct value",
+ ((Double) value).doubleValue(),
+ bean.getDoubleProperty(),
+ (double) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a float property.
+ */
+ public void testGetSimpleFloat() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "floatProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Float));
+ assertEquals("Got correct value",
+ ((Float) value).floatValue(),
+ bean.getFloatProperty(),
+ (float) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test getSimpleProperty on an indexed property.
+ */
+ public void testGetSimpleIndexed() {
+
+ Object value = null;
+ try {
+ value = PropertyUtils.getSimpleProperty(bean,
+ "intIndexed[0]");
+ fail("Should have thrown IllegalArgumentException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ ; // Correct result for this test
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on an int property.
+ */
+ public void testGetSimpleInt() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "intProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Integer));
+ assertEquals("Got correct value",
+ ((Integer) value).intValue(),
+ bean.getIntProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a long property.
+ */
+ public void testGetSimpleLong() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "longProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Long));
+ assertEquals("Got correct value",
+ ((Long) value).longValue(),
+ bean.getLongProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test getSimpleProperty on a nested property.
+ */
+ public void testGetSimpleNested() {
+
+ Object value = null;
+ try {
+ value = PropertyUtils.getSimpleProperty(bean,
+ "nested.stringProperty");
+ fail("Should have thrown IllegaArgumentException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ ; // Correct result for this test
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a read-only String property.
+ */
+ public void testGetSimpleReadOnly() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "readOnlyProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof String));
+ assertEquals("Got correct value",
+ (String) value,
+ bean.getReadOnlyProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a short property.
+ */
+ public void testGetSimpleShort() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "shortProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof Short));
+ assertEquals("Got correct value",
+ ((Short) value).shortValue(),
+ bean.getShortProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a String property.
+ */
+ public void testGetSimpleString() {
+
+ try {
+ Object value =
+ PropertyUtils.getSimpleProperty(bean,
+ "stringProperty");
+ assertNotNull("Got a value", value);
+ assertTrue("Got correct type", (value instanceof String));
+ assertEquals("Got correct value",
+ (String) value,
+ bean.getStringProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test getSimpleProperty on an unknown property.
+ */
+ public void testGetSimpleUnknown() {
+
+ try {
+ PropertyUtils.getSimpleProperty(bean, "unknown");
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Test getSimpleProperty on a write-only String property.
+ */
+ public void testGetSimpleWriteOnly() {
+
+ try {
+ PropertyUtils.getSimpleProperty(bean, "writeOnlyProperty");
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Test getting accessible property writer methods for a specified
+ * list of properties of our standard test bean.
+ */
+ public void testGetWriteMethodBasic() {
+
+ testGetWriteMethod(bean, properties, TEST_BEAN_CLASS);
+
+ }
+
+
+ /**
+ * Test getting accessible property writer methods for a specified
+ * list of properties of a package private subclass of our standard
+ * test bean.
+ */
+ public void testGetWriteMethodPackageSubclass() {
+
+ testGetWriteMethod(beanPackageSubclass, properties, TEST_BEAN_CLASS);
+
+ }
+
+
+ /**
+ * Test getting accessible property writer methods for a specified
+ * list of properties of a public subclass of our standard test bean.
+ */
+ public void testGetWriteMethodPublicSubclass() {
+
+ testGetWriteMethod(beanPublicSubclass, properties, TEST_BEAN_CLASS);
+
+ }
+
+
+ /**
+ * Test the mappedPropertyType of MappedPropertyDescriptor.
+ */
+ public void testMappedPropertyType() throws Exception {
+
+ MappedPropertyDescriptor desc;
+
+ // Check a String property
+ desc = (MappedPropertyDescriptor)
+ PropertyUtils.getPropertyDescriptor(bean,
+ "mappedProperty");
+ assertEquals(String.class, desc.getMappedPropertyType());
+
+ // Check an int property
+ desc = (MappedPropertyDescriptor)
+ PropertyUtils.getPropertyDescriptor(bean,
+ "mappedIntProperty");
+ assertEquals(Integer.TYPE, desc.getMappedPropertyType());
+
+ }
+
+
+ /**
+ * Corner cases on setIndexedProperty invalid arguments.
+ */
+ public void testSetIndexedArguments() {
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.setIndexedProperty(null, "intArray", 0,
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, null, 0,
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.setIndexedProperty(null,
+ "intArray[0]",
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, "[0]",
+ new Integer(1));
+ fail("Should throw NoSuchMethodException 4");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 4");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, "intArray",
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 5");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 5");
+ }
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.setIndexedProperty(null, "intIndexed", 0,
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, null, 0,
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.setIndexedProperty(null,
+ "intIndexed[0]",
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, "[0]",
+ new Integer(1));
+ fail("Should throw NoSuchMethodException 4");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 4");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean, "intIndexed",
+ new Integer(1));
+ fail("Should throw IllegalArgumentException 5");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 5");
+ }
+
+ }
+
+
+ /**
+ * Positive and negative tests on setIndexedProperty valid arguments.
+ */
+ public void testSetIndexedValues() {
+
+ Object value = null;
+
+ // Use explicit index argument
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "dupProperty", 0,
+ "New 0");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "dupProperty", 0);
+ assertNotNull("Returned new value 0", value);
+ assertTrue("Returned String new value 0",
+ value instanceof String);
+ assertEquals("Returned correct new value 0", "New 0",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intArray", 0,
+ new Integer(1));
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray", 0);
+ assertNotNull("Returned new value 0", value);
+ assertTrue("Returned Integer new value 0",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 0", 1,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intIndexed", 1,
+ new Integer(11));
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed", 1);
+ assertNotNull("Returned new value 1", value);
+ assertTrue("Returned Integer new value 1",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 1", 11,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "listIndexed", 2,
+ "New Value 2");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed", 2);
+ assertNotNull("Returned new value 2", value);
+ assertTrue("Returned String new value 2",
+ value instanceof String);
+ assertEquals("Returned correct new value 2", "New Value 2",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray", 2,
+ "New Value 2");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray", 2);
+ assertNotNull("Returned new value 2", value);
+ assertTrue("Returned String new value 2",
+ value instanceof String);
+ assertEquals("Returned correct new value 2", "New Value 2",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray", 3,
+ "New Value 3");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray", 3);
+ assertNotNull("Returned new value 3", value);
+ assertTrue("Returned String new value 3",
+ value instanceof String);
+ assertEquals("Returned correct new value 3", "New Value 3",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ // Use index expression
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "dupProperty[4]",
+ "New 4");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "dupProperty[4]");
+ assertNotNull("Returned new value 4", value);
+ assertTrue("Returned String new value 4",
+ value instanceof String);
+ assertEquals("Returned correct new value 4", "New 4",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intArray[4]",
+ new Integer(1));
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intArray[4]");
+ assertNotNull("Returned new value 4", value);
+ assertTrue("Returned Integer new value 4",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 4", 1,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intIndexed[3]",
+ new Integer(11));
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "intIndexed[3]");
+ assertNotNull("Returned new value 5", value);
+ assertTrue("Returned Integer new value 5",
+ value instanceof Integer);
+ assertEquals("Returned correct new value 5", 11,
+ ((Integer) value).intValue());
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "listIndexed[1]",
+ "New Value 2");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "listIndexed[1]");
+ assertNotNull("Returned new value 6", value);
+ assertTrue("Returned String new value 6",
+ value instanceof String);
+ assertEquals("Returned correct new value 6", "New Value 2",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray[1]",
+ "New Value 2");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray[2]");
+ assertNotNull("Returned new value 6", value);
+ assertTrue("Returned String new value 6",
+ value instanceof String);
+ assertEquals("Returned correct new value 6", "New Value 2",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray[0]",
+ "New Value 3");
+ value =
+ PropertyUtils.getIndexedProperty(bean,
+ "stringArray[0]");
+ assertNotNull("Returned new value 7", value);
+ assertTrue("Returned String new value 7",
+ value instanceof String);
+ assertEquals("Returned correct new value 7", "New Value 3",
+ (String) value);
+ } catch (Throwable t) {
+ fail("Threw " + t);
+ }
+
+ // Index out of bounds tests
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "dupProperty", -1,
+ "New -1");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "dupProperty", 5,
+ "New 5");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intArray", -1,
+ new Integer(0));
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intArray", 5,
+ new Integer(0));
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intIndexed", -1,
+ new Integer(0));
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "intIndexed", 5,
+ new Integer(0));
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "listIndexed", 5,
+ "New String");
+ fail("Should have thrown IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "listIndexed", -1,
+ "New String");
+ fail("Should have thrown IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray", -1,
+ "New String");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringArray", 5,
+ "New String");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringIndexed", -1,
+ "New String");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ try {
+ PropertyUtils.setIndexedProperty(bean,
+ "stringIndexed", 5,
+ "New String");
+ fail("Should have thrown ArrayIndexOutOfBoundsException");
+ } catch (ArrayIndexOutOfBoundsException t) {
+ ; // Expected results
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of ArrayIndexOutOfBoundsException");
+ }
+
+ }
+
+
+ /**
+ * Corner cases on getMappedProperty invalid arguments.
+ */
+ public void testSetMappedArguments() {
+
+ // Use explicit key argument
+
+ try {
+ PropertyUtils.setMappedProperty(null, "mappedProperty",
+ "First Key", "First Value");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, null, "First Key",
+ "First Value");
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, "mappedProperty", null,
+ "First Value");
+ fail("Should throw IllegalArgumentException 3");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 3");
+ }
+
+ // Use key expression
+
+ try {
+ PropertyUtils.setMappedProperty(null,
+ "mappedProperty(First Key)",
+ "First Value");
+ fail("Should throw IllegalArgumentException 4");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 4");
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, "(Second Key)",
+ "Second Value");
+ fail("Should throw IllegalArgumentException 5");
+ } catch (NoSuchMethodException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of NoSuchMethodException 5");
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, "mappedProperty",
+ "Third Value");
+ fail("Should throw IllegalArgumentException 6");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 6");
+ }
+
+ }
+
+
+ /**
+ * Positive and negative tests on setMappedProperty valid arguments.
+ */
+ public void testSetMappedValues() {
+
+ Object value = null;
+
+ // Use explicit key argument
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "Fourth Key");
+ assertNull("Can not find fourth value", value);
+ } catch (Throwable t) {
+ fail("Finding fourth value threw " + t);
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean, "mappedProperty",
+ "Fourth Key", "Fourth Value");
+ } catch (Throwable t) {
+ fail("Setting fourth value threw " + t);
+ }
+
+ try {
+ value = PropertyUtils.getMappedProperty(bean, "mappedProperty",
+ "Fourth Key");
+ assertEquals("Can find fourth value", "Fourth Value", value);
+ } catch (Throwable t) {
+ fail("Finding fourth value threw " + t);
+ }
+
+ // Use key expression with parentheses
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(Fifth Key)");
+ assertNull("Can not find fifth value", value);
+ } catch (Throwable t) {
+ fail("Finding fifth value threw " + t);
+ }
+
+ try {
+ PropertyUtils.setMappedProperty(bean,
+ "mappedProperty(Fifth Key)",
+ "Fifth Value");
+ } catch (Throwable t) {
+ fail("Setting fifth value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getMappedProperty(bean,
+ "mappedProperty(Fifth Key)");
+ assertEquals("Can find fifth value", "Fifth Value", value);
+ } catch (Throwable t) {
+ fail("Finding fifth value threw " + t);
+ }
+
+ // Use key expression with dotted expression
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.Sixth Key");
+ assertNull("Can not find sixth value", value);
+ } catch (Throwable t) {
+ fail("Finding fifth value threw " + t);
+ }
+
+ try {
+ PropertyUtils.setNestedProperty(bean,
+ "mapProperty.Sixth Key",
+ "Sixth Value");
+ } catch (Throwable t) {
+ fail("Setting sixth value threw " + t);
+ }
+
+ try {
+ value =
+ PropertyUtils.getNestedProperty(bean,
+ "mapProperty.Sixth Key");
+ assertEquals("Can find sixth value", "Sixth Value", value);
+ } catch (Throwable t) {
+ fail("Finding sixth value threw " + t);
+ }
+
+ }
+
+
+ /**
+ * Corner cases on setNestedProperty invalid arguments.
+ */
+ public void testSetNestedArguments() {
+
+ try {
+ PropertyUtils.setNestedProperty(null, "stringProperty", "");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setNestedProperty(bean, null, "");
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ }
+
+
+ /**
+ * Test setNextedProperty on a boolean property.
+ */
+ public void testSetNestedBoolean() {
+
+ try {
+ boolean oldValue = bean.getNested().getBooleanProperty();
+ boolean newValue = !oldValue;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.booleanProperty",
+ new Boolean(newValue));
+ assertTrue("Matched new value",
+ newValue ==
+ bean.getNested().getBooleanProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a double property.
+ */
+ public void testSetNestedDouble() {
+
+ try {
+ double oldValue = bean.getNested().getDoubleProperty();
+ double newValue = oldValue + 1.0;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.doubleProperty",
+ new Double(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getNested().getDoubleProperty(),
+ 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a float property.
+ */
+ public void testSetNestedFloat() {
+
+ try {
+ float oldValue = bean.getNested().getFloatProperty();
+ float newValue = oldValue + (float) 1.0;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.floatProperty",
+ new Float(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getNested().getFloatProperty(),
+ (float) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a int property.
+ */
+ public void testSetNestedInt() {
+
+ try {
+ int oldValue = bean.getNested().getIntProperty();
+ int newValue = oldValue + 1;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.intProperty",
+ new Integer(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getNested().getIntProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a long property.
+ */
+ public void testSetNestedLong() {
+
+ try {
+ long oldValue = bean.getNested().getLongProperty();
+ long newValue = oldValue + 1;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.longProperty",
+ new Long(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getNested().getLongProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a read-only String property.
+ */
+ public void testSetNestedReadOnly() {
+
+ try {
+ String oldValue = bean.getNested().getWriteOnlyPropertyValue();
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setNestedProperty(bean,
+ "nested.readOnlyProperty",
+ newValue);
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a short property.
+ */
+ public void testSetNestedShort() {
+
+ try {
+ short oldValue = bean.getNested().getShortProperty();
+ short newValue = oldValue;
+ newValue++;
+ PropertyUtils.setNestedProperty(bean,
+ "nested.shortProperty",
+ new Short(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getNested().getShortProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a String property.
+ */
+ public void testSetNestedString() {
+
+ try {
+ String oldValue = bean.getNested().getStringProperty();
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setNestedProperty(bean,
+ "nested.stringProperty",
+ newValue);
+ assertEquals("Matched new value",
+ newValue,
+ bean.getNested().getStringProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on an unknown property name.
+ */
+ public void testSetNestedUnknown() {
+
+ try {
+ String newValue = "New String Value";
+ PropertyUtils.setNestedProperty(bean,
+ "nested.unknown",
+ newValue);
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Test setNestedProperty on a write-only String property.
+ */
+ public void testSetNestedWriteOnly() {
+
+ try {
+ String oldValue = bean.getNested().getWriteOnlyPropertyValue();
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setNestedProperty(bean,
+ "nested.writeOnlyProperty",
+ newValue);
+ assertEquals("Matched new value",
+ newValue,
+ bean.getNested().getWriteOnlyPropertyValue());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Corner cases on setSimpleProperty invalid arguments.
+ */
+ public void testSetSimpleArguments() {
+
+ try {
+ PropertyUtils.setSimpleProperty(null, "stringProperty", "");
+ fail("Should throw IllegalArgumentException 1");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 1");
+ }
+
+ try {
+ PropertyUtils.setSimpleProperty(bean, null, "");
+ fail("Should throw IllegalArgumentException 2");
+ } catch (IllegalArgumentException e) {
+ ; // Expected response
+ } catch (Throwable t) {
+ fail("Threw " + t + " instead of IllegalArgumentException 2");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a boolean property.
+ */
+ public void testSetSimpleBoolean() {
+
+ try {
+ boolean oldValue = bean.getBooleanProperty();
+ boolean newValue = !oldValue;
+ PropertyUtils.setSimpleProperty(bean,
+ "booleanProperty",
+ new Boolean(newValue));
+ assertTrue("Matched new value",
+ newValue ==
+ bean.getBooleanProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a double property.
+ */
+ public void testSetSimpleDouble() {
+
+ try {
+ double oldValue = bean.getDoubleProperty();
+ double newValue = oldValue + 1.0;
+ PropertyUtils.setSimpleProperty(bean,
+ "doubleProperty",
+ new Double(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getDoubleProperty(),
+ 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a float property.
+ */
+ public void testSetSimpleFloat() {
+
+ try {
+ float oldValue = bean.getFloatProperty();
+ float newValue = oldValue + (float) 1.0;
+ PropertyUtils.setSimpleProperty(bean,
+ "floatProperty",
+ new Float(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getFloatProperty(),
+ (float) 0.005);
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test setSimpleProperty on an indexed property.
+ */
+ public void testSetSimpleIndexed() {
+
+ try {
+ PropertyUtils.setSimpleProperty(bean,
+ "stringIndexed[0]",
+ "New String Value");
+ fail("Should have thrown IllegalArgumentException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ ; // Correct result for this test
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a int property.
+ */
+ public void testSetSimpleInt() {
+
+ try {
+ int oldValue = bean.getIntProperty();
+ int newValue = oldValue + 1;
+ PropertyUtils.setSimpleProperty(bean,
+ "intProperty",
+ new Integer(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getIntProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a long property.
+ */
+ public void testSetSimpleLong() {
+
+ try {
+ long oldValue = bean.getLongProperty();
+ long newValue = oldValue + 1;
+ PropertyUtils.setSimpleProperty(bean,
+ "longProperty",
+ new Long(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getLongProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Negative test setSimpleProperty on a nested property.
+ */
+ public void testSetSimpleNested() {
+
+ try {
+ PropertyUtils.setSimpleProperty(bean,
+ "nested.stringProperty",
+ "New String Value");
+ fail("Should have thrown IllegalArgumentException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ ; // Correct result for this test
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a read-only String property.
+ */
+ public void testSetSimpleReadOnly() {
+
+ try {
+ String oldValue = bean.getWriteOnlyPropertyValue();
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setSimpleProperty(bean,
+ "readOnlyProperty",
+ newValue);
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a short property.
+ */
+ public void testSetSimpleShort() {
+
+ try {
+ short oldValue = bean.getShortProperty();
+ short newValue = oldValue;
+ newValue++;
+ PropertyUtils.setSimpleProperty(bean,
+ "shortProperty",
+ new Short(newValue));
+ assertEquals("Matched new value",
+ newValue,
+ bean.getShortProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a String property.
+ */
+ public void testSetSimpleString() {
+
+ try {
+ String oldValue = bean.getStringProperty();
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setSimpleProperty(bean,
+ "stringProperty",
+ newValue);
+ assertEquals("Matched new value",
+ newValue,
+ bean.getStringProperty());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on an unknown property name.
+ */
+ public void testSetSimpleUnknown() {
+
+ try {
+ String newValue = "New String Value";
+ PropertyUtils.setSimpleProperty(bean,
+ "unknown",
+ newValue);
+ fail("Should have thrown NoSuchMethodException");
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ ; // Correct result for this test
+ }
+
+ }
+
+
+ /**
+ * Test setSimpleProperty on a write-only String property.
+ */
+ public void testSetSimpleWriteOnly() {
+
+ try {
+ String oldValue = bean.getWriteOnlyPropertyValue();
+ String newValue = oldValue + " Extra Value";
+ PropertyUtils.setSimpleProperty(bean,
+ "writeOnlyProperty",
+ newValue);
+ assertEquals("Matched new value",
+ newValue,
+ bean.getWriteOnlyPropertyValue());
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (IllegalArgumentException e) {
+ fail("IllegalArgumentException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ // ------------------------------------------------------ Protected Methods
+
+
+ /**
+ * Base for testGetDescriptorXxxxx() series of tests.
+ *
+ * @param name Name of the property to be retrieved
+ * @param read Expected name of the read method (or null)
+ * @param write Expected name of the write method (or null)
+ */
+ protected void testGetDescriptorBase(String name, String read,
+ String write) {
+
+ try {
+ PropertyDescriptor pd =
+ PropertyUtils.getPropertyDescriptor(bean, name);
+ if ((read != null) || (write != null)) {
+ assertNotNull("Got descriptor", pd);
+ } else {
+ assertNull("Got descriptor", pd);
+ return;
+ }
+ Method rm = pd.getReadMethod();
+ if (read != null) {
+ assertNotNull("Got read method", rm);
+ assertEquals("Got correct read method",
+ rm.getName(), read);
+ } else {
+ assertNull("Got read method", rm);
+ }
+ Method wm = pd.getWriteMethod();
+ if (write != null) {
+ assertNotNull("Got write method", wm);
+ assertEquals("Got correct write method",
+ wm.getName(), write);
+ } else {
+ assertNull("Got write method", wm);
+ }
+ } catch (IllegalAccessException e) {
+ fail("IllegalAccessException");
+ } catch (InvocationTargetException e) {
+ fail("InvocationTargetException");
+ } catch (NoSuchMethodException e) {
+ fail("NoSuchMethodException");
+ }
+
+ }
+
+
+ /**
+ * Base for testGetReadMethod() series of tests.
+ *
+ * @param bean Bean for which to retrieve read methods.
+ * @param properties Property names to search for
+ * @param className Class name where this method should be defined
+ */
+ protected void testGetReadMethod(Object bean, String properties[],
+ String className) {
+
+ PropertyDescriptor pd[] =
+ PropertyUtils.getPropertyDescriptors(bean);
+ for (int i = 0; i < properties.length; i++) {
+
+ // Identify the property descriptor for this property
+ if (properties[i].equals("intIndexed"))
+ continue;
+ if (properties[i].equals("stringIndexed"))
+ continue;
+ if (properties[i].equals("writeOnlyProperty"))
+ continue;
+ int n = -1;
+ for (int j = 0; j < pd.length; j++) {
+ if (properties[i].equals(pd[j].getName())) {
+ n = j;
+ break;
+ }
+ }
+ assertTrue("PropertyDescriptor for " + properties[i],
+ n >= 0);
+
+ // Locate an accessible property reader method for it
+ Method reader = PropertyUtils.getReadMethod(pd[n]);
+ assertNotNull("Reader for " + properties[i],
+ reader);
+ Class clazz = reader.getDeclaringClass();
+ assertNotNull("Declaring class for " + properties[i],
+ clazz);
+ assertEquals("Correct declaring class for " + properties[i],
+ clazz.getName(),
+ className);
+
+ // Actually call the reader method we received
+ try {
+ reader.invoke(bean, new Class[0]);
+ } catch (Throwable t) {
+ fail("Call for " + properties[i] + ": " + t);
+ }
+
+ }
+
+ }
+
+
+ /**
+ * Base for testGetWriteMethod() series of tests.
+ *
+ * @param bean Bean for which to retrieve write methods.
+ * @param properties Property names to search for
+ * @param className Class name where this method should be defined
+ */
+ protected void testGetWriteMethod(Object bean, String properties[],
+ String className) {
+
+
+ PropertyDescriptor pd[] =
+ PropertyUtils.getPropertyDescriptors(bean);
+ for (int i = 0; i < properties.length; i++) {
+
+ // Identify the property descriptor for this property
+ if (properties[i].equals("intIndexed"))
+ continue;
+ if (properties[i].equals("listIndexed"))
+ continue;
+ if (properties[i].equals("nested"))
+ continue; // This property is read only
+ if (properties[i].equals("readOnlyProperty"))
+ continue;
+ if (properties[i].equals("stringIndexed"))
+ continue;
+ int n = -1;
+ for (int j = 0; j < pd.length; j++) {
+ if (properties[i].equals(pd[j].getName())) {
+ n = j;
+ break;
+ }
+ }
+ assertTrue("PropertyDescriptor for " + properties[i],
+ n >= 0);
+
+ // Locate an accessible property reader method for it
+ Method writer = PropertyUtils.getWriteMethod(pd[n]);
+ assertNotNull("Writer for " + properties[i],
+ writer);
+ Class clazz = writer.getDeclaringClass();
+ assertNotNull("Declaring class for " + properties[i],
+ clazz);
+ assertEquals("Correct declaring class for " + properties[i],
+ clazz.getName(),
+ className);
+
+ }
+
+ }
+
+ public void testNestedWithIndex() throws Exception
+ {
+ NestedTestBean nestedBean = new NestedTestBean("base");
+ nestedBean.init();
+ nestedBean.getSimpleBeanProperty().init();
+
+ NestedTestBean
+
+ // test first calling properties on indexed beans
+
+ value = (NestedTestBean) PropertyUtils.getProperty(
+ nestedBean,
+ "indexedProperty[0]");
+ assertEquals("Cannot get simple index(1)", "Bean at 0", value.getName());
+ assertEquals("Bug in NestedTestBean", "NOT SET", value.getTestString());
+
+ value = (NestedTestBean) PropertyUtils.getProperty(
+ nestedBean,
+ "indexedProperty[1]");
+ assertEquals("Cannot get simple index(1)", "Bean at 1", value.getName());
+ assertEquals("Bug in NestedTestBean", "NOT SET", value.getTestString());
+
+ String
+ prop = (String) PropertyUtils.getProperty(
+ nestedBean,
+ "indexedProperty[0].testString");
+ assertEquals("Get property on indexes failed (1)", "NOT SET", prop);
+
+ prop = (String) PropertyUtils.getProperty(
+ nestedBean,
+ "indexedProperty[1].testString");
+ assertEquals("Get property on indexes failed (2)", "NOT SET", prop);
+
+ PropertyUtils.setProperty(
+ nestedBean,
+ "indexedProperty[0].testString",
+ "Test#1");
+ assertEquals(
+ "Cannot set property on indexed bean (1)",
+ "Test#1",
+ nestedBean.getIndexedProperty(0).getTestString());
+
+ PropertyUtils.setProperty(
+ nestedBean,
+ "indexedProperty[1].testString",
+ "Test#2");
+ assertEquals(
+ "Cannot set property on indexed bean (2)",
+ "Test#2",
+ nestedBean.getIndexedProperty(1).getTestString());
+
+
+ // test first calling indexed properties on a simple property
+
+ value = (NestedTestBean) PropertyUtils.getProperty(
+ nestedBean,
+ "simpleBeanProperty");
+ assertEquals("Cannot get simple bean", "Simple Property Bean", value.getName());
+ assertEquals("Bug in NestedTestBean", "NOT SET", value.getTestString());
+
+ value = (NestedTestBean) PropertyUtils.getProperty(
+ nestedBean,
+ "simpleBeanProperty.indexedProperty[3]");
+ assertEquals("Cannot get index property on property", "Bean at 3", value.getName());
+ assertEquals("Bug in NestedTestBean", "NOT SET", value.getTestString());
+
+ PropertyUtils.setProperty(
+ nestedBean,
+ "simpleBeanProperty.indexedProperty[3].testString",
+ "Test#3");
+ assertEquals(
+ "Cannot set property on indexed property on property",
+ "Test#3",
+ nestedBean.getSimpleBeanProperty().getIndexedProperty(3).getTestString());
+ }
+
+ /** Text case for setting properties on inner classes */
+ public void testGetSetInnerBean() throws Exception {
+ BeanWithInnerBean bean = new BeanWithInnerBean();
+
+ PropertyUtils.setProperty(bean, "innerBean.fish(loiterTimer)", "5");
+ String out = (String) PropertyUtils.getProperty(bean.getInnerBean(), "fish(loiterTimer)");
+ assertEquals(
+ "(1) Inner class property set/get property failed.",
+ "5",
+ out);
+
+ out = (String) PropertyUtils.getProperty(bean, "innerBean.fish(loiterTimer)");
+
+ assertEquals(
+ "(2) Inner class property set/get property failed.",
+ "5",
+ out);
+ }
+
+ /** Text case for setting properties on parent */
+ public void testGetSetParentBean() throws Exception {
+
+ SonOfAlphaBean bean = new SonOfAlphaBean("Roger");
+
+ String out = (String) PropertyUtils.getProperty(bean, "name");
+ assertEquals(
+ "(1) Get/Set On Parent.",
+ "Roger",
+ out);
+
+ PropertyUtils.setProperty(bean, "name", "abcd");
+ assertEquals(
+ "(2) Get/Set On Parent.",
+ "abcd",
+ bean.getName());
+ }
+
+ public void testSetNoGetter() throws Exception
+ {
+ BetaBean bean = new BetaBean("Cedric");
+
+ // test standard no getter
+ bean.setNoGetterProperty("Sigma");
+ assertEquals("BetaBean test failed", "Sigma", bean.getSecret());
+
+ assertNotNull("Descriptor is null", PropertyUtils.getPropertyDescriptor(bean, "noGetterProperty"));
+
+ BeanUtils.setProperty(bean, "noGetterProperty", "Omega");
+ assertEquals("Cannot set no-getter property", "Omega", bean.getSecret());
+
+ // test mapped no getter descriptor
+ MappedPropertyDescriptor descriptor
+ = new MappedPropertyDescriptor("noGetterMappedProperty", BetaBean.class);
+
+ assertNotNull("Map Descriptor is null", PropertyUtils.getPropertyDescriptor(bean, "noGetterMappedProperty"));
+
+ PropertyUtils.setMappedProperty(bean, "noGetterMappedProperty", "Epsilon", "Epsilon");
+ assertEquals("Cannot set mapped no-getter property", "MAP:Epsilon", bean.getSecret());
+ }
+
+ /**
+ * This tests to see that classes that implement Map can have
+ * their standard properties set.
+ */
+ public void testSetMapExtension() throws Exception {
+ ExtendMapBean bean = new ExtendMapBean();
+
+ bean.setUnusuallyNamedProperty("bean value");
+ assertEquals("Set property direct failed", "bean value", bean.getUnusuallyNamedProperty());
+
+ PropertyUtils.setSimpleProperty(bean, "unusuallyNamedProperty", "new value");
+ assertEquals("Set property on map failed (1)", "new value", bean.getUnusuallyNamedProperty());
+
+ PropertyUtils.setProperty(bean, "unusuallyNamedProperty", "next value");
+ assertEquals("Set property on map failed (2)", "next value", bean.getUnusuallyNamedProperty());
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/SonOfAlphaBean.java b/trunk/src/test/org/apache/commons/beanutils/SonOfAlphaBean.java
new file mode 100644
index 0000000..1acc625
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/SonOfAlphaBean.java
@@ -0,0 +1,25 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils;
+
+/** @author Robert Burrell Donkin */
+public class SonOfAlphaBean extends AlphaBean {
+
+ public SonOfAlphaBean(String name) {
+ super(name);
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/TestBean.java b/trunk/src/test/org/apache/commons/beanutils/TestBean.java
new file mode 100755
index 0000000..5ff6e04
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/TestBean.java
@@ -0,0 +1,572 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+
+/**
+ * General purpose test bean for JUnit tests for the "beanutils" component.
+ *
+ * @author Craig R. McClanahan
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.20 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class TestBean {
+
+ // ----------------------------------------------------------- Constructors
+
+ public TestBean() {
+ }
+
+ public TestBean(String stringProperty) {
+ setStringProperty(stringProperty);
+ }
+
+ public TestBean(float floatProperty) {
+ setFloatProperty(floatProperty);
+ }
+
+ public TestBean(boolean booleanProperty) {
+ setBooleanProperty(booleanProperty);
+ }
+
+ public TestBean(Boolean booleanSecond) {
+ setBooleanSecond(booleanSecond.booleanValue());
+ }
+
+ public TestBean(float floatProperty, String stringProperty) {
+ setFloatProperty(floatProperty);
+ setStringProperty(stringProperty);
+ }
+
+ public TestBean(boolean booleanProperty, String stringProperty) {
+ setBooleanProperty(booleanProperty);
+ setStringProperty(stringProperty);
+ }
+
+ public TestBean(Boolean booleanSecond, String stringProperty) {
+ setBooleanSecond(booleanSecond.booleanValue());
+ setStringProperty(stringProperty);
+ }
+
+ public TestBean(Integer intProperty) {
+ setIntProperty(intProperty.intValue());
+ }
+
+ public TestBean(double doubleProperty) {
+ setDoubleProperty(doubleProperty);
+ }
+
+ TestBean(int intProperty) {
+ setIntProperty(intProperty);
+ }
+
+ protected TestBean(boolean booleanProperty, boolean booleanSecond, String stringProperty) {
+ setBooleanProperty(booleanProperty);
+ setBooleanSecond(booleanSecond);
+ setStringProperty(stringProperty);
+ }
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ * A boolean property.
+ */
+ private boolean booleanProperty = true;
+
+ public boolean getBooleanProperty() {
+ return (booleanProperty);
+ }
+
+ public void setBooleanProperty(boolean booleanProperty) {
+ this.booleanProperty = booleanProperty;
+ }
+
+
+ /**
+ * A boolean property that uses an "is" method for the getter.
+ */
+ private boolean booleanSecond = true;
+
+ public boolean isBooleanSecond() {
+ return (booleanSecond);
+ }
+
+ public void setBooleanSecond(boolean booleanSecond) {
+ this.booleanSecond = booleanSecond;
+ }
+
+
+ /**
+ * A byte property.
+ */
+ private byte byteProperty = (byte) 121;
+
+ public byte getByteProperty() {
+ return (this.byteProperty);
+ }
+
+ public void setByteProperty(byte byteProperty) {
+ this.byteProperty = byteProperty;
+ }
+
+
+ /**
+ * A double property.
+ */
+ private double doubleProperty = 321.0;
+
+ public double getDoubleProperty() {
+ return (this.doubleProperty);
+ }
+
+ public void setDoubleProperty(double doubleProperty) {
+ this.doubleProperty = doubleProperty;
+ }
+
+
+ /**
+ * An "indexed property" accessible via both array and subscript
+ * based getters and setters.
+ */
+ private String dupProperty[] =
+ { "Dup 0", "Dup 1", "Dup 2", "Dup 3", "Dup 4" };
+
+ public String[] getDupProperty() {
+ return (this.dupProperty);
+ }
+
+ public String getDupProperty(int index) {
+ return (this.dupProperty[index]);
+ }
+
+ public void setDupProperty(int index, String value) {
+ this.dupProperty[index] = value;
+ }
+
+ public void setDupProperty(String dupProperty[]) {
+ this.dupProperty = dupProperty;
+ }
+
+
+ /**
+ * A float property.
+ */
+ private float floatProperty = (float) 123.0;
+
+ public float getFloatProperty() {
+ return (this.floatProperty);
+ }
+
+ public void setFloatProperty(float floatProperty) {
+ this.floatProperty = floatProperty;
+ }
+
+
+ /**
+ * An integer array property accessed as an array.
+ */
+ private int intArray[] = { 0, 10, 20, 30, 40 };
+
+ public int[] getIntArray() {
+ return (this.intArray);
+ }
+
+ public void setIntArray(int intArray[]) {
+ this.intArray = intArray;
+ }
+
+
+ /**
+ * An integer array property accessed as an indexed property.
+ */
+ private int intIndexed[] = { 0, 10, 20, 30, 40 };
+
+ public int getIntIndexed(int index) {
+ return (intIndexed[index]);
+ }
+
+ public void setIntIndexed(int index, int value) {
+ intIndexed[index] = value;
+ }
+
+
+ /**
+ * An integer property.
+ */
+ private int intProperty = 123;
+
+ public int getIntProperty() {
+ return (this.intProperty);
+ }
+
+ public void setIntProperty(int intProperty) {
+ this.intProperty = intProperty;
+ }
+
+
+ /**
+ * A List property accessed as an indexed property.
+ */
+ private static List listIndexed = new ArrayList();
+
+ static {
+ listIndexed.add("String 0");
+ listIndexed.add("String 1");
+ listIndexed.add("String 2");
+ listIndexed.add("String 3");
+ listIndexed.add("String 4");
+ }
+
+ public List getListIndexed() {
+ return (listIndexed);
+ }
+
+
+ /**
+ * A long property.
+ */
+ private long longProperty = 321;
+
+ public long getLongProperty() {
+ return (this.longProperty);
+ }
+
+ public void setLongProperty(long longProperty) {
+ this.longProperty = longProperty;
+ }
+
+
+ /**
+ * A mapped property with only a getter and setter for a Map.
+ */
+ private Map mapProperty = null;
+
+ public Map getMapProperty() {
+ // Create the map the very first time
+ if (mapProperty == null) {
+ mapProperty = new HashMap();
+ mapProperty.put("First Key", "First Value");
+ mapProperty.put("Second Key", "Second Value");
+ }
+ return (mapProperty);
+ }
+
+ public void setMapProperty(Map mapProperty) {
+ // Create the map the very first time
+ if (mapProperty == null) {
+ mapProperty = new HashMap();
+ mapProperty.put("First Key", "First Value");
+ mapProperty.put("Second Key", "Second Value");
+ }
+ this.mapProperty = mapProperty;
+ }
+
+
+ /**
+ * A mapped property that has String keys and Object values.
+ */
+ private HashMap mappedObjects = null;
+
+ public Object getMappedObjects(String key) {
+ // Create the map the very first time
+ if (mappedObjects == null) {
+ mappedObjects = new HashMap();
+ mappedObjects.put("First Key", "First Value");
+ mappedObjects.put("Second Key", "Second Value");
+ }
+ return (mappedObjects.get(key));
+ }
+
+ public void setMappedObjects(String key, Object value) {
+ // Create the map the very first time
+ if (mappedObjects == null) {
+ mappedObjects = new HashMap();
+ mappedObjects.put("First Key", "First Value");
+ mappedObjects.put("Second Key", "Second Value");
+ }
+ mappedObjects.put(key, value);
+ }
+
+
+ /**
+ * A mapped property that has String keys and String values.
+ */
+ private HashMap mappedProperty = null;
+
+ public String getMappedProperty(String key) {
+ // Create the map the very first time
+ if (mappedProperty == null) {
+ mappedProperty = new HashMap();
+ mappedProperty.put("First Key", "First Value");
+ mappedProperty.put("Second Key", "Second Value");
+ }
+ return ((String) mappedProperty.get(key));
+ }
+
+ public void setMappedProperty(String key, String value) {
+ // Create the map the very first time
+ if (mappedProperty == null) {
+ mappedProperty = new HashMap();
+ mappedProperty.put("First Key", "First Value");
+ mappedProperty.put("Second Key", "Second Value");
+ }
+ mappedProperty.put(key, value);
+ }
+
+
+ /**
+ * A mapped property that has String keys and int values.
+ */
+ private HashMap mappedIntProperty = null;
+
+ public int getMappedIntProperty(String key) {
+ // Create the map the very first time
+ if (mappedProperty == null) {
+ mappedProperty = new HashMap();
+ mappedProperty.put("One", new Integer(1));
+ mappedProperty.put("Two", new Integer(2));
+ }
+ Integer x = (Integer) mappedIntProperty.get(key);
+ return ((x == null) ? 0 : x.intValue());
+ }
+
+ public void setMappedIntProperty(String key, int value) {
+ mappedIntProperty.put(key, new Integer(value));
+ }
+
+
+ /**
+ * A nested reference to another test bean (populated as needed).
+ */
+ private TestBean nested = null;
+
+ public TestBean getNested() {
+ if (nested == null)
+ nested = new TestBean();
+ return (nested);
+ }
+
+ /**
+ * Another nested reference to another test bean,
+ */
+ private TestBean anotherNested = null;
+
+ public TestBean getAnotherNested() {
+ return anotherNested;
+ }
+
+ public void setAnotherNested( TestBean anotherNested ) {
+ this.anotherNested = anotherNested;
+ }
+
+ /*
+ * Another nested reference to a bean containing mapp properties
+ */
+ class MappedTestBean {
+ public void setValue(String key,String val) { }
+ public String getValue(String key) { return "Mapped Value"; }
+ }
+
+ private MappedTestBean mappedNested = null;
+
+ public MappedTestBean getMappedNested() {
+ if (mappedNested == null)
+ {
+ mappedNested = new MappedTestBean();
+ }
+ return mappedNested;
+ }
+
+ /**
+ * A String property with an initial value of null.
+ */
+ private String nullProperty = null;
+
+ public String getNullProperty() {
+ return (this.nullProperty);
+ }
+
+ public void setNullProperty(String nullProperty) {
+ this.nullProperty = nullProperty;
+ }
+
+
+ /**
+ * A read-only String property.
+ */
+ private String readOnlyProperty = "Read Only String Property";
+
+ public String getReadOnlyProperty() {
+ return (this.readOnlyProperty);
+ }
+
+
+ /**
+ * A short property.
+ */
+ private short shortProperty = (short) 987;
+
+ public short getShortProperty() {
+ return (this.shortProperty);
+ }
+
+ public void setShortProperty(short shortProperty) {
+ this.shortProperty = shortProperty;
+ }
+
+
+ /**
+ * A String array property accessed as a String.
+ */
+ private String stringArray[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+
+ public String[] getStringArray() {
+ return (this.stringArray);
+ }
+
+ public void setStringArray(String stringArray[]) {
+ this.stringArray = stringArray;
+ }
+
+
+ /**
+ * A String array property accessed as an indexed property.
+ */
+ private String stringIndexed[] =
+ { "String 0", "String 1", "String 2", "String 3", "String 4" };
+
+ public String getStringIndexed(int index) {
+ return (stringIndexed[index]);
+ }
+
+ public void setStringIndexed(int index, String value) {
+ stringIndexed[index] = value;
+ }
+
+
+ /**
+ * A String property.
+ */
+ private String stringProperty = "This is a string";
+
+ public String getStringProperty() {
+ return (this.stringProperty);
+ }
+
+ public void setStringProperty(String stringProperty) {
+ this.stringProperty = stringProperty;
+ }
+
+
+ /**
+ * A write-only String property.
+ */
+ private String writeOnlyProperty = "Write Only String Property";
+
+ public String getWriteOnlyPropertyValue() {
+ return (this.writeOnlyProperty);
+ }
+
+ public void setWriteOnlyProperty(String writeOnlyProperty) {
+ this.writeOnlyProperty = writeOnlyProperty;
+ }
+
+
+ // ------------------------------------------------------ Invalid Properties
+
+
+ /**
+ * <p>An invalid property that has two boolean getters (getInvalidBoolean
+ * and isInvalidBoolean) plus a String setter (setInvalidBoolean). By the
+ * rules described in the JavaBeans Specification, this will be considered
+ * a read-only boolean property, using isInvalidBoolean() as the getter.</p>
+ */
+ private boolean invalidBoolean = false;
+
+ public boolean getInvalidBoolean() {
+ return (this.invalidBoolean);
+ }
+
+ public boolean isInvalidBoolean() {
+ return (this.invalidBoolean);
+ }
+
+ public void setInvalidBoolean(String invalidBoolean) {
+ if ("true".equalsIgnoreCase(invalidBoolean) ||
+ "yes".equalsIgnoreCase(invalidBoolean) ||
+ "1".equalsIgnoreCase(invalidBoolean)) {
+ this.invalidBoolean = true;
+ } else {
+ this.invalidBoolean = false;
+ }
+ }
+
+
+
+ // ------------------------------------------------------- Static Variables
+
+
+ /**
+ * A static variable that is accessed and updated via static methods
+ * for MethodUtils testing.
+ */
+ private static int counter = 0;
+
+
+ /**
+ * Return the current value of the counter.
+ */
+ public static int currentCounter() {
+
+ return (counter);
+
+ }
+
+
+ /**
+ * Increment the current value of the counter by 1.
+ */
+ public static void incrementCounter() {
+
+ incrementCounter(1);
+
+ }
+
+
+ /**
+ * Increment the current value of the counter by the specified amount.
+ *
+ * @param amount Amount to be added to the current counter
+ */
+ public static void incrementCounter(int amount) {
+
+ counter += amount;
+
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/TestBeanPackageSubclass.java b/trunk/src/test/org/apache/commons/beanutils/TestBeanPackageSubclass.java
new file mode 100644
index 0000000..3da64c5
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/TestBeanPackageSubclass.java
@@ -0,0 +1,32 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+/**
+ * This is a package private subclass of TestBean. All of our properties
+ * should still be accessible via reflection.
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.5 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+class TestBeanPackageSubclass extends TestBean {
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/TestBeanPublicSubclass.java b/trunk/src/test/org/apache/commons/beanutils/TestBeanPublicSubclass.java
new file mode 100644
index 0000000..776c112
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/TestBeanPublicSubclass.java
@@ -0,0 +1,32 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+/**
+ * This is a public subclass of TestBean. All of our properties should still
+ * be accessible via reflection.
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.5 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class TestBeanPublicSubclass extends TestBean {
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/TestResultSet.java b/trunk/src/test/org/apache/commons/beanutils/TestResultSet.java
new file mode 100644
index 0000000..61005ce
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/TestResultSet.java
@@ -0,0 +1,864 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.io.InputStream;
+import java.io.Reader;
+import java.math.BigDecimal;
+import java.net.URL;
+import java.sql.Array;
+import java.sql.Blob;
+import java.sql.Clob;
+import java.sql.Date;
+import java.sql.Ref;
+import java.sql.ResultSet;
+import java.sql.ResultSetMetaData;
+import java.sql.SQLException;
+import java.sql.SQLWarning;
+import java.sql.Statement;
+import java.sql.Time;
+import java.sql.Timestamp;
+import java.util.Calendar;
+import java.util.Map;
+
+
+/**
+ * <p>Mock object that implements enough of <code>java.sql.ResultSet</code>
+ * to exercise the {@link ResultSetDyaClass} functionality.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.5 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class TestResultSet implements ResultSet {
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * Current row number (0 means "before the first one").
+ */
+ protected int row = 0;
+
+
+ /**
+ * The constant (per run) value used to initialize date/time/timestamp.
+ */
+ protected long timestamp = System.currentTimeMillis();
+
+
+ // ---------------------------------------------------- Implemented Methods
+
+
+ public void close() throws SQLException {
+ ; // No action required
+ }
+
+
+ public ResultSetMetaData getMetaData() throws SQLException {
+ return (new TestResultSetMetaData());
+ }
+
+
+ public Object getObject(String columnName) throws SQLException {
+ if (row > 5) {
+ throw new SQLException("No current row");
+ }
+ if ("bigdecimalproperty".equals(columnName)) {
+ return (new BigDecimal(123.45));
+ } else if ("booleanproperty".equals(columnName)) {
+ if ((row % 2) == 0) {
+ return (Boolean.TRUE);
+ } else {
+ return (Boolean.FALSE);
+ }
+ } else if ("byteproperty".equals(columnName)) {
+ return (new Byte((byte) row));
+ } else if ("dateproperty".equals(columnName)) {
+ return (new Date(timestamp));
+ } else if ("doubleproperty".equals(columnName)) {
+ return (new Double(321.0));
+ } else if ("floatproperty".equals(columnName)) {
+ return (new Float((float) 123.0));
+ } else if ("intproperty".equals(columnName)) {
+ return (new Integer(100 + row));
+ } else if ("longproperty".equals(columnName)) {
+ return (new Long(200 + row));
+ } else if ("nullproperty".equals(columnName)) {
+ return (null);
+ } else if ("shortproperty".equals(columnName)) {
+ return (new Short((short) (300 + row)));
+ } else if ("stringproperty".equals(columnName)) {
+ return ("This is a string");
+ } else if ("timeproperty".equals(columnName)) {
+ return (new Time(timestamp));
+ } else if ("timestampproperty".equals(columnName)) {
+ return (new Timestamp(timestamp));
+ } else {
+ throw new SQLException("Unknown column name " + columnName);
+ }
+ }
+
+
+ public boolean next() throws SQLException {
+ if (row++ < 5) {
+ return (true);
+ } else {
+ return (false);
+ }
+ }
+
+
+ public void updateObject(String columnName, Object x)
+ throws SQLException {
+ if (row > 5) {
+ throw new SQLException("No current row");
+ }
+ ; // FIXME - updateObject()
+ }
+
+
+ // -------------------------------------------------- Unimplemented Methods
+
+
+ public boolean absolute(int row) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void afterLast() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void beforeFirst() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void cancelRowUpdates() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void clearWarnings() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void deleteRow() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int findColumn(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean first() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Array getArray(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Array getArray(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public InputStream getAsciiStream(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public InputStream getAsciiStream(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public BigDecimal getBigDecimal(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+ /** @deprecated */
+ public BigDecimal getBigDecimal(int columnIndex, int scale)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public BigDecimal getBigDecimal(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ /** @deprecated */
+ public BigDecimal getBigDecimal(String columnName, int scale)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public InputStream getBinaryStream(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public InputStream getBinaryStream(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Blob getBlob(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Blob getBlob(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean getBoolean(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean getBoolean(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public byte getByte(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public byte getByte(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public byte[] getBytes(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public byte[] getBytes(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Reader getCharacterStream(int columnIndex)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Reader getCharacterStream(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Clob getClob(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Clob getClob(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getConcurrency() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public String getCursorName() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Date getDate(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Date getDate(int columnIndex, Calendar cal) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Date getDate(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Date getDate(String columnName, Calendar cal) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public double getDouble(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public double getDouble(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getFetchDirection() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getFetchSize() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public float getFloat(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public float getFloat(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getInt(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getInt(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public long getLong(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public long getLong(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Object getObject(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Object getObject(int columnIndex, Map map) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Object getObject(String columnName, Map map) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Ref getRef(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Ref getRef(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getRow() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public short getShort(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public short getShort(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Statement getStatement() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public String getString(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public String getString(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Time getTime(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Time getTime(int columnIndex, Calendar cal) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Time getTime(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Time getTime(String columnName, Calendar cal) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Timestamp getTimestamp(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Timestamp getTimestamp(int columnIndex, Calendar cal)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Timestamp getTimestamp(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public Timestamp getTimestamp(String columnName, Calendar cal)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getType() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ /** @deprecated */
+ public InputStream getUnicodeStream(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ /** @deprecated */
+ public InputStream getUnicodeStream(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public URL getURL(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public URL getURL(String columnName) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public SQLWarning getWarnings() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void insertRow() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isAfterLast() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isBeforeFirst() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isFirst() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isLast() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean last() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void moveToCurrentRow() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void moveToInsertRow() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean previous() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void refreshRow() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean relative(int rows) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean rowDeleted() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean rowInserted() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean rowUpdated() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void setFetchDirection(int direction) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void setFetchSize(int size) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateArray(int columnPosition, Array x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateArray(String columnName, Array x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateAsciiStream(int columnPosition, InputStream x, int len)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateAsciiStream(String columnName, InputStream x, int len)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBigDecimal(int columnPosition, BigDecimal x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBigDecimal(String columnName, BigDecimal x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBinaryStream(int columnPosition, InputStream x, int len)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBinaryStream(String columnName, InputStream x, int len)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBlob(int columnPosition, Blob x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBlob(String columnName, Blob x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBoolean(int columnPosition, boolean x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBoolean(String columnName, boolean x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateByte(int columnPosition, byte x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateByte(String columnName, byte x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBytes(int columnPosition, byte x[])
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateBytes(String columnName, byte x[])
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateCharacterStream(int columnPosition, Reader x, int len)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateCharacterStream(String columnName, Reader x, int len)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateClob(int columnPosition, Clob x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateClob(String columnName, Clob x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateDate(int columnPosition, Date x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateDate(String columnName, Date x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateDouble(int columnPosition, double x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateDouble(String columnName, double x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateFloat(int columnPosition, float x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateFloat(String columnName, float x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateInt(int columnPosition, int x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateInt(String columnName, int x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateLong(int columnPosition, long x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateLong(String columnName, long x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateNull(int columnPosition)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateNull(String columnName)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateObject(int columnPosition, Object x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateObject(int columnPosition, Object x, int scale)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateObject(String columnName, Object x, int scale)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateRef(int columnPosition, Ref x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateRef(String columnName, Ref x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateRow() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateShort(int columnPosition, short x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateShort(String columnName, short x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateString(int columnPosition, String x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateString(String columnName, String x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateTime(int columnPosition, Time x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateTime(String columnName, Time x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateTimestamp(int columnPosition, Timestamp x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public void updateTimestamp(String columnName, Timestamp x)
+ throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean wasNull() throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/TestResultSetMetaData.java b/trunk/src/test/org/apache/commons/beanutils/TestResultSetMetaData.java
new file mode 100644
index 0000000..8028b86
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/TestResultSetMetaData.java
@@ -0,0 +1,170 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import java.sql.ResultSetMetaData;
+import java.sql.SQLException;
+
+
+/**
+ * <p>Mock object that implements enough of
+ * <code>java.sql.ResultSetMetaData</code>
+ * to exercise the {@link ResultSetDyaClass} functionality.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class TestResultSetMetaData implements ResultSetMetaData {
+
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * <p>Array of column names and class names for metadata.</p>
+ */
+ protected String metadata[][] = {
+ { "bigDecimalProperty", "java.math.BigDecimal" },
+ { "booleanProperty", Boolean.class.getName() },
+ { "byteProperty", Byte.class.getName() },
+ { "dateProperty", "java.sql.Date" },
+ { "doubleProperty", Double.class.getName() },
+ { "floatProperty", Float.class.getName() },
+ { "intProperty", Integer.class.getName() },
+ { "longProperty", Long.class.getName() },
+ { "nullProperty", "java.lang.String" },
+ { "shortProperty", Short.class.getName() },
+ { "stringProperty", "java.lang.String" },
+ { "timeProperty", "java.sql.Time" },
+ { "timestampProperty", "java.sql.Timestamp" },
+ };
+
+
+ // ---------------------------------------------------- Implemented Methods
+
+
+ public String getColumnClassName(int columnIndex) throws SQLException {
+ return (metadata[columnIndex - 1][1]);
+ }
+
+
+ public int getColumnCount() throws SQLException {
+ return (metadata.length);
+ }
+
+ public String getColumnName(int columnIndex) throws SQLException {
+ return (metadata[columnIndex - 1][0]);
+ }
+
+
+ // -------------------------------------------------- Unimplemented Methods
+
+
+ public String getCatalogName(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getColumnDisplaySize(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public String getColumnLabel(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getColumnType(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public String getColumnTypeName(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getPrecision(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int getScale(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public String getSchemaName(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public String getTableName(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isAutoIncrement(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isCaseSensitive(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isCurrency(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isDefinitelyWritable(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public int isNullable(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isReadOnly(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isSearchable(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isSigned(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+ public boolean isWritable(int columnIndex) throws SQLException {
+ throw new UnsupportedOperationException();
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/ThrowExceptionConverter.java b/trunk/src/test/org/apache/commons/beanutils/ThrowExceptionConverter.java
new file mode 100644
index 0000000..635e9a9
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/ThrowExceptionConverter.java
@@ -0,0 +1,35 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+/**
+ * Converter implementation that throws a <code>PassTestException</code>
+ * when convert is called.
+ * The idea is that catching this exception is a clear signal that this method
+ * has been called.
+ *
+ * @author Robert Burrell Donkin
+ */
+
+public class ThrowExceptionConverter implements Converter {
+
+ public Object convert(Class type, Object value) {
+ throw new PassTestException();
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/WrapDynaBeanTestCase.java b/trunk/src/test/org/apache/commons/beanutils/WrapDynaBeanTestCase.java
new file mode 100644
index 0000000..8ab6f2d
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/WrapDynaBeanTestCase.java
@@ -0,0 +1,182 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils;
+
+
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+
+/**
+ * <p>Test Case for the <code>WrapDynaBean</code> implementation class.
+ * These tests were based on the ones in <code>PropertyUtilsTestCase</code>
+ * because the two classes provide similar levels of functionality.</p>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.9 $ $Date: 2004/02/28 13:18:36 $
+ */
+
+public class WrapDynaBeanTestCase extends BasicDynaBeanTestCase {
+
+
+ // ---------------------------------------------------- Instance Variables
+
+
+ // ---------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public WrapDynaBeanTestCase(String name) {
+
+ super(name);
+
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() throws Exception {
+
+ bean = new WrapDynaBean(new TestBean());
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+
+ return (new TestSuite(WrapDynaBeanTestCase.class));
+
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+
+ bean = null;
+
+ }
+
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ /**
+ * The <code>contains()</code> method is not supported by the
+ * <code>WrapDynaBean</code> implementation class.
+ */
+ public void testMappedContains() {
+
+ try {
+ assertTrue("Can see first key",
+ bean.contains("mappedProperty", "First Key"));
+ fail("Should have thrown UnsupportedOperationException");
+ } catch (UnsupportedOperationException t) {
+ ; // Expected result
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+
+ try {
+ assertTrue("Can not see unknown key",
+ !bean.contains("mappedProperty", "Unknown Key"));
+ fail("Should have thrown UnsupportedOperationException");
+ } catch (UnsupportedOperationException t) {
+ ; // Expected result
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * The <code>remove()</code> method is not supported by the
+ * <code>WrapDynaBean</code> implementation class.
+ */
+ public void testMappedRemove() {
+
+ try {
+ assertTrue("Can see first key",
+ bean.contains("mappedProperty", "First Key"));
+ bean.remove("mappedProperty", "First Key");
+ fail("Should have thrown UnsupportedOperationException");
+ // assertTrue("Can not see first key",
+ // !bean.contains("mappedProperty", "First Key"));
+ } catch (UnsupportedOperationException t) {
+ ; // Expected result
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ try {
+ assertTrue("Can not see unknown key",
+ !bean.contains("mappedProperty", "Unknown Key"));
+ bean.remove("mappedProperty", "Unknown Key");
+ fail("Should have thrown UnsupportedOperationException");
+ // assertTrue("Can not see unknown key",
+ // !bean.contains("mappedProperty", "Unknown Key"));
+ } catch (UnsupportedOperationException t) {
+ ; // Expected result
+ } catch (Throwable t) {
+ fail("Exception: " + t);
+ }
+
+ }
+
+
+ /**
+ * Suppress serialization and deserialization tests. WrapDynaClass
+ * is not serializable.
+ */
+ public void testSerialization() { }
+
+ /** Tests getInstance method */
+ public void testGetInstance() {
+ AlphaBean alphaBean = new AlphaBean("Now On Air... John Peel");
+ WrapDynaBean dynaBean = new WrapDynaBean(alphaBean);
+ Object wrappedInstance = dynaBean.getInstance();
+ assertTrue("Object type is AlphaBean", wrappedInstance instanceof AlphaBean);
+ AlphaBean wrappedAlphaBean = (AlphaBean) wrappedInstance;
+ assertTrue("Same Object", wrappedAlphaBean == alphaBean);
+ }
+
+ /** Tests the newInstance implementation for WrapDynaClass */
+ public void testNewInstance() throws Exception {
+ WrapDynaClass dynaClass = WrapDynaClass.createDynaClass(AlphaBean.class);
+ Object createdInstance = dynaClass.newInstance();
+ assertTrue("Object type is WrapDynaBean", createdInstance instanceof WrapDynaBean);
+ WrapDynaBean dynaBean = (WrapDynaBean) createdInstance;
+ assertTrue("Object type is AlphaBean", dynaBean.getInstance() instanceof AlphaBean);
+ }
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/ByteConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/converters/ByteConverterTestCase.java
new file mode 100644
index 0000000..18e2dc4
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/ByteConverterTestCase.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.TestSuite;
+
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * Test Case for the ByteConverter class.
+ *
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class ByteConverterTestCase extends NumberConverterTestBase {
+
+ private Converter converter = null;
+
+ // ------------------------------------------------------------------------
+
+ public ByteConverterTestCase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void setUp() throws Exception {
+ converter = makeConverter();
+ }
+
+ public static TestSuite suite() {
+ return new TestSuite(ByteConverterTestCase.class);
+ }
+
+ public void tearDown() throws Exception {
+ converter = null;
+ }
+
+ // ------------------------------------------------------------------------
+
+ protected Converter makeConverter() {
+ return new ByteConverter();
+ }
+
+ protected Class getExpectedType() {
+ return Byte.class;
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void testSimpleConversion() throws Exception {
+ String[] message= {
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from Byte",
+ "from Short",
+ "from Integer",
+ "from Long",
+ "from Float",
+ "from Double"
+ };
+
+ Object[] input = {
+ String.valueOf(Byte.MIN_VALUE),
+ "-17",
+ "-1",
+ "0",
+ "1",
+ "17",
+ String.valueOf(Byte.MAX_VALUE),
+ new Byte((byte)7),
+ new Short((short)8),
+ new Integer(9),
+ new Long(10),
+ new Float(11.1),
+ new Double(12.2)
+ };
+
+ Byte[] expected = {
+ new Byte(Byte.MIN_VALUE),
+ new Byte((byte)-17),
+ new Byte((byte)-1),
+ new Byte((byte)0),
+ new Byte((byte)1),
+ new Byte((byte)17),
+ new Byte(Byte.MAX_VALUE),
+ new Byte((byte)7),
+ new Byte((byte)8),
+ new Byte((byte)9),
+ new Byte((byte)10),
+ new Byte((byte)11),
+ new Byte((byte)12)
+ };
+
+ for(int i=0;i<expected.length;i++) {
+ assertEquals(message[i] + " to Byte",expected[i],converter.convert(Byte.class,input[i]));
+ assertEquals(message[i] + " to byte",expected[i],converter.convert(Byte.TYPE,input[i]));
+ assertEquals(message[i] + " to null type",expected[i],converter.convert(null,input[i]));
+ }
+ }
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/ConverterTestSuite.java b/trunk/src/test/org/apache/commons/beanutils/converters/ConverterTestSuite.java
new file mode 100644
index 0000000..7e14718
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/ConverterTestSuite.java
@@ -0,0 +1,50 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.Test;
+import junit.framework.TestCase;
+import junit.framework.TestSuite;
+
+/**
+ * <p>
+ * Created a test suite so that new test cases can be added here without having to
+ * edit the build.xml.
+ * </p>
+ *
+ * @author Robert Burrell Donkin
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class ConverterTestSuite {
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ TestSuite testSuite = new TestSuite();
+ testSuite.addTestSuite(ByteConverterTestCase.class);
+ testSuite.addTestSuite(DoubleConverterTestCase.class);
+ testSuite.addTestSuite(FloatConverterTestCase.class);
+ testSuite.addTestSuite(IntegerConverterTestCase.class);
+ testSuite.addTestSuite(LongConverterTestCase.class);
+ testSuite.addTestSuite(ShortConverterTestCase.class);
+ testSuite.addTestSuite(StringArrayConverterTestCase.class);
+ return testSuite;
+ }
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/DoubleConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/converters/DoubleConverterTestCase.java
new file mode 100644
index 0000000..32c8ddd
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/DoubleConverterTestCase.java
@@ -0,0 +1,136 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.TestSuite;
+
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * Test Case for the DoubleConverter class.
+ *
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class DoubleConverterTestCase extends NumberConverterTestBase {
+
+ private Converter converter = null;
+
+ // ------------------------------------------------------------------------
+
+ public DoubleConverterTestCase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void setUp() throws Exception {
+ converter = makeConverter();
+ }
+
+ public static TestSuite suite() {
+ return new TestSuite(DoubleConverterTestCase.class);
+ }
+
+ public void tearDown() throws Exception {
+ converter = null;
+ }
+
+ // ------------------------------------------------------------------------
+
+ protected Converter makeConverter() {
+ return new DoubleConverter();
+ }
+
+ protected Class getExpectedType() {
+ return Double.class;
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void testSimpleConversion() throws Exception {
+ String[] message= {
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from Byte",
+ "from Short",
+ "from Integer",
+ "from Long",
+ "from Float",
+ "from Double"
+ };
+
+ Object[] input = {
+ String.valueOf(Double.MIN_VALUE),
+ "-17.2",
+ "-1.1",
+ "0.0",
+ "1.1",
+ "17.2",
+ String.valueOf(Double.MAX_VALUE),
+ new Byte((byte)7),
+ new Short((short)8),
+ new Integer(9),
+ new Long(10),
+ new Float(11.1),
+ new Double(12.2)
+ };
+
+ Double[] expected = {
+ new Double(Double.MIN_VALUE),
+ new Double(-17.2),
+ new Double(-1.1),
+ new Double(0.0),
+ new Double(1.1),
+ new Double(17.2),
+ new Double(Double.MAX_VALUE),
+ new Double(7),
+ new Double(8),
+ new Double(9),
+ new Double(10),
+ new Double(11.1),
+ new Double(12.2)
+ };
+
+ for(int i=0;i<expected.length;i++) {
+ assertEquals(
+ message[i] + " to Double",
+ expected[i].doubleValue(),
+ ((Double)(converter.convert(Double.class,input[i]))).doubleValue(),
+ 0.00001D);
+ assertEquals(
+ message[i] + " to double",
+ expected[i].doubleValue(),
+ ((Double)(converter.convert(Double.TYPE,input[i]))).doubleValue(),
+ 0.00001D);
+ assertEquals(
+ message[i] + " to null type",
+ expected[i].doubleValue(),
+ ((Double)(converter.convert(null,input[i]))).doubleValue(),
+ 0.00001D);
+ }
+ }
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/FileConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/converters/FileConverterTestCase.java
new file mode 100644
index 0000000..55fa652
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/FileConverterTestCase.java
@@ -0,0 +1,96 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import java.io.File;
+
+import junit.framework.TestCase;
+import junit.framework.TestSuite;
+
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * Test Case for the FileConverter class.
+ *
+ * @author James Strachan
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class FileConverterTestCase extends TestCase {
+
+ private Converter converter = null;
+
+ // ------------------------------------------------------------------------
+
+ public FileConverterTestCase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void setUp() throws Exception {
+ converter = makeConverter();
+ }
+
+ public static TestSuite suite() {
+ return new TestSuite(FileConverterTestCase.class);
+ }
+
+ public void tearDown() throws Exception {
+ converter = null;
+ }
+
+ // ------------------------------------------------------------------------
+
+ protected Converter makeConverter() {
+ return new FileConverter();
+ }
+
+ protected Class getExpectedType() {
+ return File.class;
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void testSimpleConversion() throws Exception {
+ String[] message= {
+ "from String",
+ "from String",
+ "from String"
+ };
+
+ Object[] input = {
+ "/tmp",
+ "/tmp/foo.txt",
+ "/tmp/does/not/exist.foo"
+ };
+
+ File[] expected = {
+ new File("/tmp"),
+ new File("/tmp/foo.txt"),
+ new File("/tmp/does/not/exist.foo")
+ };
+
+ for(int i=0;i<expected.length;i++) {
+ assertEquals(message[i] + " to File",expected[i],converter.convert(File.class,input[i]));
+ assertEquals(message[i] + " to null type",expected[i],converter.convert(null,input[i]));
+ }
+ }
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/FloatConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/converters/FloatConverterTestCase.java
new file mode 100644
index 0000000..4dbeadf
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/FloatConverterTestCase.java
@@ -0,0 +1,136 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.TestSuite;
+
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * Test Case for the FloatConverter class.
+ *
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class FloatConverterTestCase extends NumberConverterTestBase {
+
+ private Converter converter = null;
+
+ // ------------------------------------------------------------------------
+
+ public FloatConverterTestCase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void setUp() throws Exception {
+ converter = makeConverter();
+ }
+
+ public static TestSuite suite() {
+ return new TestSuite(FloatConverterTestCase.class);
+ }
+
+ public void tearDown() throws Exception {
+ converter = null;
+ }
+
+ // ------------------------------------------------------------------------
+
+ protected Converter makeConverter() {
+ return new FloatConverter();
+ }
+
+ protected Class getExpectedType() {
+ return Float.class;
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void testSimpleConversion() throws Exception {
+ String[] message= {
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from Byte",
+ "from Short",
+ "from Integer",
+ "from Long",
+ "from Float",
+ "from Double"
+ };
+
+ Object[] input = {
+ String.valueOf(Float.MIN_VALUE),
+ "-17.2",
+ "-1.1",
+ "0.0",
+ "1.1",
+ "17.2",
+ String.valueOf(Float.MAX_VALUE),
+ new Byte((byte)7),
+ new Short((short)8),
+ new Integer(9),
+ new Long(10),
+ new Float(11.1),
+ new Double(12.2)
+ };
+
+ Float[] expected = {
+ new Float(Float.MIN_VALUE),
+ new Float(-17.2),
+ new Float(-1.1),
+ new Float(0.0),
+ new Float(1.1),
+ new Float(17.2),
+ new Float(Float.MAX_VALUE),
+ new Float(7),
+ new Float(8),
+ new Float(9),
+ new Float(10),
+ new Float(11.1),
+ new Float(12.2)
+ };
+
+ for(int i=0;i<expected.length;i++) {
+ assertEquals(
+ message[i] + " to Float",
+ expected[i].floatValue(),
+ ((Float)(converter.convert(Float.class,input[i]))).floatValue(),
+ 0.00001);
+ assertEquals(
+ message[i] + " to float",
+ expected[i].floatValue(),
+ ((Float)(converter.convert(Float.TYPE,input[i]))).floatValue(),
+ 0.00001);
+ assertEquals(
+ message[i] + " to null type",
+ expected[i].floatValue(),
+ ((Float)(converter.convert(null,input[i]))).floatValue(),
+ 0.00001);
+ }
+ }
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/IntegerConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/converters/IntegerConverterTestCase.java
new file mode 100644
index 0000000..d2ea741
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/IntegerConverterTestCase.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.TestSuite;
+
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * Test Case for the IntegerConverter class.
+ *
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class IntegerConverterTestCase extends NumberConverterTestBase {
+
+ private Converter converter = null;
+
+ // ------------------------------------------------------------------------
+
+ public IntegerConverterTestCase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void setUp() throws Exception {
+ converter = makeConverter();
+ }
+
+ public static TestSuite suite() {
+ return new TestSuite(IntegerConverterTestCase.class);
+ }
+
+ public void tearDown() throws Exception {
+ converter = null;
+ }
+
+ // ------------------------------------------------------------------------
+
+ protected Converter makeConverter() {
+ return new IntegerConverter();
+ }
+
+ protected Class getExpectedType() {
+ return Integer.class;
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void testSimpleConversion() throws Exception {
+ String[] message= {
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from Byte",
+ "from Short",
+ "from Integer",
+ "from Long",
+ "from Float",
+ "from Double"
+ };
+
+ Object[] input = {
+ String.valueOf(Integer.MIN_VALUE),
+ "-17",
+ "-1",
+ "0",
+ "1",
+ "17",
+ String.valueOf(Integer.MAX_VALUE),
+ new Byte((byte)7),
+ new Short((short)8),
+ new Integer(9),
+ new Long(10),
+ new Float(11.1),
+ new Double(12.2)
+ };
+
+ Integer[] expected = {
+ new Integer(Integer.MIN_VALUE),
+ new Integer(-17),
+ new Integer(-1),
+ new Integer(0),
+ new Integer(1),
+ new Integer(17),
+ new Integer(Integer.MAX_VALUE),
+ new Integer(7),
+ new Integer(8),
+ new Integer(9),
+ new Integer(10),
+ new Integer(11),
+ new Integer(12)
+ };
+
+ for(int i=0;i<expected.length;i++) {
+ assertEquals(message[i] + " to Integer",expected[i],converter.convert(Integer.class,input[i]));
+ assertEquals(message[i] + " to int",expected[i],converter.convert(Integer.TYPE,input[i]));
+ assertEquals(message[i] + " to null type",expected[i],converter.convert(null,input[i]));
+ }
+ }
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/LongConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/converters/LongConverterTestCase.java
new file mode 100644
index 0000000..ef05623
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/LongConverterTestCase.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.TestSuite;
+
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * Test Case for the LongConverter class.
+ *
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class LongConverterTestCase extends NumberConverterTestBase {
+
+ private Converter converter = null;
+
+ // ------------------------------------------------------------------------
+
+ public LongConverterTestCase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void setUp() throws Exception {
+ converter = makeConverter();
+ }
+
+ public static TestSuite suite() {
+ return new TestSuite(LongConverterTestCase.class);
+ }
+
+ public void tearDown() throws Exception {
+ converter = null;
+ }
+
+ // ------------------------------------------------------------------------
+
+ protected Converter makeConverter() {
+ return new LongConverter();
+ }
+
+ protected Class getExpectedType() {
+ return Long.class;
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void testSimpleConversion() throws Exception {
+ String[] message= {
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from Byte",
+ "from Short",
+ "from Integer",
+ "from Long",
+ "from Float",
+ "from Double"
+ };
+
+ Object[] input = {
+ String.valueOf(Long.MIN_VALUE),
+ "-17",
+ "-1",
+ "0",
+ "1",
+ "17",
+ String.valueOf(Long.MAX_VALUE),
+ new Byte((byte)7),
+ new Short((short)8),
+ new Integer(9),
+ new Long(10),
+ new Float(11.1),
+ new Double(12.2)
+ };
+
+ Long[] expected = {
+ new Long(Long.MIN_VALUE),
+ new Long(-17),
+ new Long(-1),
+ new Long(0),
+ new Long(1),
+ new Long(17),
+ new Long(Long.MAX_VALUE),
+ new Long(7),
+ new Long(8),
+ new Long(9),
+ new Long(10),
+ new Long(11),
+ new Long(12)
+ };
+
+ for(int i=0;i<expected.length;i++) {
+ assertEquals(message[i] + " to Long",expected[i],converter.convert(Long.class,input[i]));
+ assertEquals(message[i] + " to long",expected[i],converter.convert(Long.TYPE,input[i]));
+ assertEquals(message[i] + " to null type",expected[i],converter.convert(null,input[i]));
+ }
+ }
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/NumberConverterTestBase.java b/trunk/src/test/org/apache/commons/beanutils/converters/NumberConverterTestBase.java
new file mode 100644
index 0000000..e7ca88a
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/NumberConverterTestBase.java
@@ -0,0 +1,91 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.TestCase;
+
+import org.apache.commons.beanutils.ConversionException;
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * Abstract base for <Number>Converter classes.
+ *
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public abstract class NumberConverterTestBase extends TestCase {
+
+ // ------------------------------------------------------------------------
+
+ public NumberConverterTestBase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ protected abstract Converter makeConverter();
+ protected abstract Class getExpectedType();
+
+ // ------------------------------------------------------------------------
+
+ /**
+ * Assumes ConversionException in response to covert(getExpectedType(),null).
+ */
+ public void testConvertNull() throws Exception {
+ try {
+ makeConverter().convert(getExpectedType(),null);
+ fail("Expected ConversionException");
+ } catch(ConversionException e) {
+ // expected
+ }
+ }
+
+ /**
+ * Assumes convert(getExpectedType(),Number) returns some non-null
+ * instance of getExpectedType().
+ */
+ public void testConvertNumber() throws Exception {
+ String[] message= {
+ "from Byte",
+ "from Short",
+ "from Integer",
+ "from Long",
+ "from Float",
+ "from Double"
+ };
+
+ Object[] number = {
+ new Byte((byte)7),
+ new Short((short)8),
+ new Integer(9),
+ new Long(10),
+ new Float(11.1),
+ new Double(12.2)
+ };
+
+ for(int i=0;i<number.length;i++) {
+ Object val = makeConverter().convert(getExpectedType(),number[i]);
+ assertNotNull("Convert " + message[i] + " should not be null",val);
+ assertTrue(
+ "Convert " + message[i] + " should return a " + getExpectedType().getName(),
+ getExpectedType().isInstance(val));
+ }
+ }
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/ShortConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/converters/ShortConverterTestCase.java
new file mode 100644
index 0000000..801a109
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/ShortConverterTestCase.java
@@ -0,0 +1,124 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.TestSuite;
+
+import org.apache.commons.beanutils.Converter;
+
+
+/**
+ * Test Case for the ShortConverter class.
+ *
+ * @author Rodney Waldhoff
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class ShortConverterTestCase extends NumberConverterTestBase {
+
+ private Converter converter = null;
+
+ // ------------------------------------------------------------------------
+
+ public ShortConverterTestCase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void setUp() throws Exception {
+ converter = makeConverter();
+ }
+
+ public static TestSuite suite() {
+ return new TestSuite(ShortConverterTestCase.class);
+ }
+
+ public void tearDown() throws Exception {
+ converter = null;
+ }
+
+ // ------------------------------------------------------------------------
+
+ protected Converter makeConverter() {
+ return new ShortConverter();
+ }
+
+ protected Class getExpectedType() {
+ return Short.class;
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void testSimpleConversion() throws Exception {
+ String[] message= {
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from Byte",
+ "from Short",
+ "from Integer",
+ "from Long",
+ "from Float",
+ "from Double"
+ };
+
+ Object[] input = {
+ String.valueOf(Short.MIN_VALUE),
+ "-17",
+ "-1",
+ "0",
+ "1",
+ "17",
+ String.valueOf(Short.MAX_VALUE),
+ new Byte((byte)7),
+ new Short((short)8),
+ new Integer(9),
+ new Long(10),
+ new Float(11.1),
+ new Double(12.2)
+ };
+
+ Short[] expected = {
+ new Short(Short.MIN_VALUE),
+ new Short((short)-17),
+ new Short((short)-1),
+ new Short((short)0),
+ new Short((short)1),
+ new Short((short)17),
+ new Short(Short.MAX_VALUE),
+ new Short((short)7),
+ new Short((short)8),
+ new Short((short)9),
+ new Short((short)10),
+ new Short((short)11),
+ new Short((short)12)
+ };
+
+ for(int i=0;i<expected.length;i++) {
+ assertEquals(message[i] + " to Short",expected[i],converter.convert(Short.class,input[i]));
+ assertEquals(message[i] + " to short",expected[i],converter.convert(Short.TYPE,input[i]));
+ assertEquals(message[i] + " to null type",expected[i],converter.convert(null,input[i]));
+ }
+ }
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/StringArrayConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/converters/StringArrayConverterTestCase.java
new file mode 100644
index 0000000..54b448f
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/StringArrayConverterTestCase.java
@@ -0,0 +1,48 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.TestCase;
+import junit.framework.TestCase;
+
+/**
+ * Test Case for StringArrayConverter
+ *
+ * @author Robert Burrell Donkin
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class StringArrayConverterTestCase extends TestCase {
+
+ public StringArrayConverterTestCase(String name) {
+ super(name);
+ }
+
+ public void testIntToString() {
+
+ int[] testArray = {1, 2, 3, 4, 5};
+ String[] results = (String []) new StringArrayConverter().convert(String.class, testArray);
+
+ assertEquals("Incorrect results size", 5, results.length);
+ assertEquals("Entry one is wrong", "1", results[0]);
+ assertEquals("Entry two is wrong", "2", results[1]);
+ assertEquals("Entry three is wrong", "3", results[2]);
+ assertEquals("Entry four is wrong", "4", results[3]);
+ assertEquals("Entry five is wrong", "5", results[4]);
+ }
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/converters/URLConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/converters/URLConverterTestCase.java
new file mode 100644
index 0000000..70fe00e
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/converters/URLConverterTestCase.java
@@ -0,0 +1,111 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.converters;
+
+import junit.framework.TestSuite;
+import junit.framework.TestCase;
+
+import org.apache.commons.beanutils.Converter;
+
+import java.net.URL;
+
+
+/**
+ * Test Case for the URLConverter class.
+ *
+ * @author Henri Yandell
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class URLConverterTestCase extends TestCase {
+
+ private Converter converter = null;
+
+ // ------------------------------------------------------------------------
+
+ public URLConverterTestCase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void setUp() throws Exception {
+ converter = makeConverter();
+ }
+
+ public static TestSuite suite() {
+ return new TestSuite(URLConverterTestCase.class);
+ }
+
+ public void tearDown() throws Exception {
+ converter = null;
+ }
+
+ // ------------------------------------------------------------------------
+
+ protected Converter makeConverter() {
+ return new URLConverter();
+ }
+
+ protected Class getExpectedType() {
+ return URL.class;
+ }
+
+ // ------------------------------------------------------------------------
+
+ public void testSimpleConversion() throws Exception {
+ String[] message= {
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ "from String",
+ };
+
+ Object[] input = {
+ "http://www.apache.org",
+ "http://www.apache.org/",
+ "ftp://cvs.apache.org",
+ "file://project.xml",
+ "http://208.185.179.12",
+ "http://www.apache.org:9999/test/thing",
+ "http://user:admin@www.apache.org:50/one/two.three",
+ "http://notreal.apache.org",
+ };
+
+ URL[] expected = {
+ new URL("http://www.apache.org"),
+ new URL("http://www.apache.org/"),
+ new URL("ftp://cvs.apache.org"),
+ new URL("file://project.xml"),
+ new URL("http://208.185.179.12"),
+ new URL("http://www.apache.org:9999/test/thing"),
+ new URL("http://user:admin@www.apache.org:50/one/two.three"),
+ new URL("http://notreal.apache.org")
+ };
+
+ for(int i=0;i<expected.length;i++) {
+ assertEquals(message[i] + " to URL",expected[i],converter.convert(URL.class,input[i]));
+ assertEquals(message[i] + " to null type",expected[i],converter.convert(null,input[i]));
+ }
+ }
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/locale/LocaleBeanificationTestCase.java b/trunk/src/test/org/apache/commons/beanutils/locale/LocaleBeanificationTestCase.java
new file mode 100644
index 0000000..e9a67a1
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/locale/LocaleBeanificationTestCase.java
@@ -0,0 +1,533 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale;
+
+import java.util.*;
+
+import java.lang.ref.WeakReference;
+import java.lang.ref.ReferenceQueue;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+
+import org.apache.commons.collections.ReferenceMap;
+import org.apache.commons.logging.LogFactory;
+
+import org.apache.commons.beanutils.ContextClassLoaderLocal;
+import org.apache.commons.beanutils.PrimitiveBean;
+
+/**
+ * <p>
+ * Test Case for changes made during LocaleBeanutils Beanification.
+ * This is basically a cut-and-correct version of the beanutils beanifications tests.
+ * </p>
+ *
+ * @author Robert Burrell Donkin
+ * @author Juozas Baliuka
+ * @version $Revision: 1.2 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class LocaleBeanificationTestCase extends TestCase {
+
+ // ---------------------------------------------------- Constants
+
+ /** Maximum number of iterations before our test fails */
+ public static final int MAX_GC_ITERATIONS = 50;
+
+ // ---------------------------------------------------- Instance Variables
+
+
+ // ---------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public LocaleBeanificationTestCase(String name) {
+ super(name);
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() {
+
+ LocaleConvertUtils.deregister();
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(LocaleBeanificationTestCase.class));
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ ; // No action required
+ }
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+ /** Test of the methodology we'll use for some of the later tests */
+ public void testMemoryTestMethodology() throws Exception {
+ // test methodology
+ // many thanks to Juozas Baliuka for suggesting this method
+ ClassLoader loader = new ClassLoader() {};
+ WeakReference reference = new WeakReference(loader);
+ Class myClass = loader.loadClass("org.apache.commons.beanutils.BetaBean");
+
+ assertNotNull("Weak reference released early", reference.get());
+
+ // dereference class loader and class:
+ loader = null;
+ myClass = null;
+
+ int iterations = 0;
+ int bytz = 2;
+ while(true) {
+ System.gc();
+ if(iterations++ > MAX_GC_ITERATIONS){
+ fail("Max iterations reached before resource released.");
+ }
+ if( reference.get() == null ) {
+ break;
+
+ } else {
+ // create garbage:
+ byte[] b = new byte[bytz];
+ bytz = bytz * 2;
+ }
+ }
+ }
+
+ /** Tests whether classloaders and beans are released from memory by the map used by beanutils */
+ public void testMemoryLeak2() throws Exception {
+ // tests when the map used by beanutils has the right behaviour
+
+ if (isPre14JVM()) {
+ System.out.println("WARNING: CANNOT TEST MEMORY LEAK ON PRE1.4 JVM");
+ return;
+ }
+
+ // many thanks to Juozas Baliuka for suggesting this methodology
+ TestClassLoader loader = new TestClassLoader();
+ ReferenceQueue queue = new ReferenceQueue();
+ WeakReference loaderReference = new WeakReference(loader, queue);
+ Integer test = new Integer(1);
+
+ WeakReference testReference = new WeakReference(test, queue);
+ //Map map = new ReferenceMap(ReferenceMap.WEAK, ReferenceMap.HARD, true);
+ Map map = new WeakHashMap();
+ map.put(loader, test);
+
+ assertEquals("In map", test, map.get(loader));
+ assertNotNull("Weak reference released early (1)", loaderReference.get());
+ assertNotNull("Weak reference released early (2)", testReference.get());
+
+ // dereference strong references
+ loader = null;
+ test = null;
+
+ int iterations = 0;
+ int bytz = 2;
+ while(true) {
+ System.gc();
+ if(iterations++ > MAX_GC_ITERATIONS){
+ fail("Max iterations reached before resource released.");
+ }
+ map.isEmpty();
+
+ if(
+ loaderReference.get() == null &&
+ testReference.get() == null) {
+ break;
+
+ } else {
+ // create garbage:
+ byte[] b = new byte[bytz];
+ bytz = bytz * 2;
+ }
+ }
+ }
+
+ /** Tests whether classloaders and beans are released from memory */
+ public void testMemoryLeak() throws Exception {
+ if (isPre14JVM()) {
+ System.out.println("WARNING: CANNOT TEST MEMORY LEAK ON PRE1.4 JVM");
+ return;
+ }
+
+ // many thanks to Juozas Baliuka for suggesting this methodology
+ TestClassLoader loader = new TestClassLoader();
+ WeakReference loaderReference = new WeakReference(loader);
+ LocaleBeanUtilsBean.getLocaleBeanUtilsInstance();
+
+ class GetBeanUtilsBeanThread extends Thread {
+
+ LocaleBeanUtilsBean beanUtils;
+ LocaleConvertUtilsBean convertUtils;
+
+ GetBeanUtilsBeanThread() {}
+
+ public void run() {
+ beanUtils = LocaleBeanUtilsBean.getLocaleBeanUtilsInstance();
+ convertUtils = LocaleConvertUtilsBean.getInstance();
+ // XXX Log keeps a reference around!
+ LogFactory.releaseAll();
+ }
+
+ public String toString() {
+ return "GetBeanUtilsBeanThread";
+ }
+ }
+
+
+ GetBeanUtilsBeanThread thread = new GetBeanUtilsBeanThread();
+ WeakReference threadWeakReference = new WeakReference(thread);
+ thread.setContextClassLoader(loader);
+
+ thread.start();
+ thread.join();
+
+ WeakReference beanUtilsReference = new WeakReference(thread.beanUtils);
+ WeakReference convertUtilsReference = new WeakReference(thread.convertUtils);
+
+ assertNotNull("Weak reference released early (1)", loaderReference.get());
+ assertNotNull("Weak reference released early (2)", beanUtilsReference.get());
+ assertNotNull("Weak reference released early (4)", convertUtilsReference.get());
+
+ // dereference strong references
+ loader = null;
+ thread.setContextClassLoader(null);
+ thread = null;
+
+ int iterations = 0;
+ int bytz = 2;
+ while(true) {
+ LocaleBeanUtilsBean.getLocaleBeanUtilsInstance();
+ System.gc();
+ if(iterations++ > MAX_GC_ITERATIONS){
+ fail("Max iterations reached before resource released.");
+ }
+
+ if(
+ loaderReference.get() == null &&
+ beanUtilsReference.get() == null &&
+ convertUtilsReference.get() == null) {
+ break;
+
+ } else {
+ // create garbage:
+ byte[] b = new byte[bytz];
+ bytz = bytz * 2;
+ }
+ }
+ }
+
+ /**
+ * Tests whether difference instances are loaded by different
+ * context classloaders.
+ */
+ public void testGetByContextClassLoader() throws Exception {
+
+ class GetBeanUtilsBeanThread extends Thread {
+
+ private Signal signal;
+
+ GetBeanUtilsBeanThread(Signal signal) {
+ this.signal = signal;
+ }
+
+ public void run() {
+ signal.setSignal(2);
+ signal.setBean(LocaleBeanUtilsBean.getLocaleBeanUtilsInstance());
+ signal.setConvertUtils(LocaleConvertUtilsBean.getInstance());
+ }
+
+ public String toString() {
+ return "GetBeanUtilsBeanThread";
+ }
+ }
+
+ Signal signal = new Signal();
+ signal.setSignal(1);
+
+ GetBeanUtilsBeanThread thread = new GetBeanUtilsBeanThread(signal);
+ thread.setContextClassLoader(new TestClassLoader());
+
+ thread.start();
+ thread.join();
+
+ assertEquals("Signal not set by test thread", 2, signal.getSignal());
+ assertTrue(
+ "Different LocaleBeanUtilsBean instances per context classloader",
+ LocaleBeanUtilsBean.getInstance() != signal.getBean());
+ assertTrue(
+ "Different LocaleConvertUtilsBean instances per context classloader",
+ LocaleConvertUtilsBean.getInstance() != signal.getConvertUtils());
+ }
+
+
+ /**
+ * Tests whether difference instances are loaded by different
+ * context classloaders.
+ */
+ public void testContextClassLoaderLocal() throws Exception {
+
+ class CCLLTesterThread extends Thread {
+
+ private Signal signal;
+ private ContextClassLoaderLocal ccll;
+
+ CCLLTesterThread(Signal signal, ContextClassLoaderLocal ccll) {
+ this.signal = signal;
+ this.ccll = ccll;
+ }
+
+ public void run() {
+ ccll.set(new Integer(1789));
+ signal.setSignal(2);
+ signal.setMarkerObject(ccll.get());
+ }
+
+ public String toString() {
+ return "CCLLTesterThread";
+ }
+ }
+
+ ContextClassLoaderLocal ccll = new ContextClassLoaderLocal();
+ ccll.set(new Integer(1776));
+ assertEquals("Start thread sets value", new Integer(1776), ccll.get());
+
+ Signal signal = new Signal();
+ signal.setSignal(1);
+
+ CCLLTesterThread thread = new CCLLTesterThread(signal, ccll);
+ thread.setContextClassLoader(new TestClassLoader());
+
+ thread.start();
+ thread.join();
+
+ assertEquals("Signal not set by test thread", 2, signal.getSignal());
+ assertEquals("Second thread preserves value", new Integer(1776), ccll.get());
+ assertEquals("Second thread gets value it set", new Integer(1789), signal.getMarkerObject());
+ }
+
+ /** Tests whether calls are independent for different classloaders */
+ public void testContextClassloaderIndependence() throws Exception {
+
+ class TestIndependenceThread extends Thread {
+ private Signal signal;
+ private PrimitiveBean bean;
+
+ TestIndependenceThread(Signal signal, PrimitiveBean bean) {
+ this.signal = signal;
+ this.bean = bean;
+ }
+
+ public void run() {
+ try {
+ signal.setSignal(3);
+ LocaleConvertUtils.register(new LocaleConverter() {
+ public Object convert(Class type, Object value) {
+ return new Integer(9);
+ }
+ public Object convert(Class type, Object value, String pattern) {
+ return new Integer(9);
+ }
+ }, Integer.TYPE, Locale.getDefault());
+ LocaleBeanUtils.setProperty(bean, "int", "1");
+ } catch (Exception e) {
+ e.printStackTrace();
+ signal.setException(e);
+ }
+ }
+
+ public String toString() {
+ return "TestIndependenceThread";
+ }
+ }
+
+ PrimitiveBean bean = new PrimitiveBean();
+ LocaleBeanUtils.setProperty(bean, "int", new Integer(1));
+ assertEquals("Wrong property value (1)", 1, bean.getInt());
+
+ LocaleConvertUtils.register(new LocaleConverter() {
+ public Object convert(Class type, Object value) {
+ return new Integer(5);
+ }
+ public Object convert(Class type, Object value, String pattern) {
+ return new Integer(5);
+ }
+ }, Integer.TYPE, Locale.getDefault());
+ LocaleBeanUtils.setProperty(bean, "int", "1");
+ assertEquals("Wrong property value(2)", 5, bean.getInt());
+
+ Signal signal = new Signal();
+ signal.setSignal(1);
+ TestIndependenceThread thread = new TestIndependenceThread(signal, bean);
+ thread.setContextClassLoader(new TestClassLoader());
+
+ thread.start();
+ thread.join();
+
+ assertNull("Exception thrown by test thread:" + signal.getException(), signal.getException());
+ assertEquals("Signal not set by test thread", 3, signal.getSignal());
+ assertEquals("Wrong property value(3)", 9, bean.getInt());
+
+ }
+
+ /** Tests whether different threads can set beanutils instances correctly */
+ public void testBeanUtilsBeanSetInstance() throws Exception {
+
+ class SetInstanceTesterThread extends Thread {
+
+ private Signal signal;
+ private LocaleBeanUtilsBean bean;
+
+ SetInstanceTesterThread(Signal signal, LocaleBeanUtilsBean bean) {
+ this.signal = signal;
+ this.bean = bean;
+ }
+
+ public void run() {
+ LocaleBeanUtilsBean.setInstance(bean);
+ signal.setSignal(21);
+ signal.setBean(LocaleBeanUtilsBean.getLocaleBeanUtilsInstance());
+ }
+
+ public String toString() {
+ return "SetInstanceTesterThread";
+ }
+ }
+
+ Signal signal = new Signal();
+ signal.setSignal(1);
+
+ LocaleBeanUtilsBean beanOne = new LocaleBeanUtilsBean();
+ LocaleBeanUtilsBean beanTwo = new LocaleBeanUtilsBean();
+
+ SetInstanceTesterThread thread = new SetInstanceTesterThread(signal, beanTwo);
+ thread.setContextClassLoader(new TestClassLoader());
+
+ LocaleBeanUtilsBean.setInstance(beanOne);
+ assertEquals("Start thread gets right instance", beanOne, LocaleBeanUtilsBean.getLocaleBeanUtilsInstance());
+
+ thread.start();
+ thread.join();
+
+ assertEquals("Signal not set by test thread", 21, signal.getSignal());
+ assertEquals("Second thread preserves value", beanOne, LocaleBeanUtilsBean.getLocaleBeanUtilsInstance());
+ assertEquals("Second thread gets value it set", beanTwo, signal.getBean());
+ }
+
+ /** Tests whether the unset method works*/
+ public void testContextClassLoaderUnset() throws Exception {
+ LocaleBeanUtilsBean beanOne = new LocaleBeanUtilsBean();
+ ContextClassLoaderLocal ccll = new ContextClassLoaderLocal();
+ ccll.set(beanOne);
+ assertEquals("Start thread gets right instance", beanOne, ccll.get());
+ ccll.unset();
+ assertTrue("Unset works", !beanOne.equals(ccll.get()));
+ }
+
+ private boolean isPre14JVM() {
+ // some pre 1.4 JVM have buggy WeakHashMap implementations
+ // this is used to test for those JVM
+ String version = System.getProperty("java.specification.version");
+ StringTokenizer tokenizer = new StringTokenizer(version,".");
+ if (tokenizer.nextToken().equals("1")) {
+ String minorVersion = tokenizer.nextToken();
+ if (minorVersion.equals("0")) return true;
+ if (minorVersion.equals("1")) return true;
+ if (minorVersion.equals("2")) return true;
+ if (minorVersion.equals("3")) return true;
+ }
+ return false;
+ }
+
+ // ---- Auxillary classes
+
+ class TestClassLoader extends ClassLoader {
+ public String toString() {
+ return "TestClassLoader";
+ }
+ }
+
+ class Signal {
+ private Exception e;
+ private int signal = 0;
+ private LocaleBeanUtilsBean bean;
+ private LocaleConvertUtilsBean convertUtils;
+ private Object marker;
+
+ public Exception getException() {
+ return e;
+ }
+
+ public void setException(Exception e) {
+ this.e = e;
+ }
+
+ public int getSignal() {
+ return signal;
+ }
+
+ public void setSignal(int signal) {
+ this.signal = signal;
+ }
+
+ public Object getMarkerObject() {
+ return marker;
+ }
+
+ public void setMarkerObject(Object marker) {
+ this.marker = marker;
+ }
+
+ public LocaleBeanUtilsBean getBean() {
+ return bean;
+ }
+
+ public void setBean(LocaleBeanUtilsBean bean) {
+ this.bean = bean;
+ }
+
+ public LocaleConvertUtilsBean getConvertUtils() {
+ return convertUtils;
+ }
+
+ public void setConvertUtils(LocaleConvertUtilsBean convertUtils) {
+ this.convertUtils = convertUtils;
+ }
+ }
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/locale/LocaleConvertTestSuite.java b/trunk/src/test/org/apache/commons/beanutils/locale/LocaleConvertTestSuite.java
new file mode 100644
index 0000000..8c38f01
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/locale/LocaleConvertTestSuite.java
@@ -0,0 +1,48 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale;
+
+import junit.framework.Test;
+import junit.framework.TestCase;
+import junit.framework.TestSuite;
+
+import org.apache.commons.beanutils.locale.converters.DateLocaleConverterTestCase;
+
+/**
+ * <p>
+ * Created a test suite so that new test cases can be added here without having to
+ * edit the build.xml.
+ * </p>
+ *
+ * @author Robert Burrell Donkin
+ * @version $Revision: 1.6 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class LocaleConvertTestSuite {
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ TestSuite testSuite = new TestSuite();
+ testSuite.addTestSuite(LocaleConvertUtilsTestCase.class);
+ testSuite.addTestSuite(DateLocaleConverterTestCase.class);
+ testSuite.addTestSuite(LocaleBeanificationTestCase.class);
+ return testSuite;
+ }
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/locale/LocaleConvertUtilsTestCase.java b/trunk/src/test/org/apache/commons/beanutils/locale/LocaleConvertUtilsTestCase.java
new file mode 100644
index 0000000..2d7bc13
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/locale/LocaleConvertUtilsTestCase.java
@@ -0,0 +1,639 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale;
+
+
+import java.lang.reflect.InvocationTargetException;
+import java.sql.Date;
+import java.sql.Time;
+import java.sql.Timestamp;
+import java.util.HashMap;
+import java.util.Locale;
+import java.text.DecimalFormat;
+import java.text.NumberFormat;
+
+import junit.framework.TestCase;
+import junit.framework.Test;
+import junit.framework.TestSuite;
+import org.apache.commons.beanutils.ConvertUtils;
+import org.apache.commons.beanutils.ConversionException;
+
+
+/**
+ * <p>
+ * Test Case for the LocaleConvertUtils class.
+ * See unimplemented functionality of the convert utils in the method begining with fixme
+ * </p>
+ *
+ * @author Michael Szlapa
+ * @author Paul Hamamnt & Rune Johannesen (pairing) - patches.
+ * @version $Revision: 1.7 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class LocaleConvertUtilsTestCase extends TestCase {
+
+ // ---------------------------------------------------- Instance Variables
+
+ private char m_decimalSeparator;
+
+ // ---------------------------------------------------------- Constructors
+
+
+ /**
+ * Construct a new instance of this test case.
+ *
+ * @param name Name of the test case
+ */
+ public LocaleConvertUtilsTestCase(String name) {
+ super(name);
+ }
+
+
+ // -------------------------------------------------- Overall Test Methods
+
+
+ /**
+ * Set up instance variables required by this test case.
+ */
+ public void setUp() {
+
+ LocaleConvertUtils.deregister();
+
+ NumberFormat nf = DecimalFormat.getNumberInstance();
+ String result = nf.format(1.1);
+
+ // could be commas instead of stops in Europe.
+ m_decimalSeparator = result.charAt(1);
+
+
+ }
+
+
+ /**
+ * Return the tests included in this test suite.
+ */
+ public static Test suite() {
+ return (new TestSuite(LocaleConvertUtilsTestCase.class));
+ }
+
+
+ /**
+ * Tear down instance variables required by this test case.
+ */
+ public void tearDown() {
+ ; // No action required
+ }
+
+
+ // ------------------------------------------------ Individual Test Methods
+
+
+ /**
+ * Negative String to primitive integer array tests.
+ */
+ public void fixmetestNegativeIntegerArray() {
+
+ fail("Array conversions not implemented yet.");
+
+ Object value = null;
+ int intArray[] = new int[0];
+
+ value = LocaleConvertUtils.convert((String) null, intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = LocaleConvertUtils.convert("a", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = LocaleConvertUtils.convert("{ a }", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = LocaleConvertUtils.convert("1a3", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = LocaleConvertUtils.convert("{ 1a3 }", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = LocaleConvertUtils.convert("0,1a3", intArray.getClass());
+ checkIntegerArray(value, intArray);
+ value = LocaleConvertUtils.convert("{ 0, 1a3 }", intArray.getClass());
+ checkIntegerArray(value, intArray);
+
+ }
+
+
+ /**
+ * Negative scalar conversion tests. These rely on the standard
+ * default value conversions in LocaleConvertUtils.
+ */
+ public void fixmetestNegativeScalar() {
+
+ Object value = null;
+
+ /* fixme Boolean converters not implemented at this point
+ value = LocaleConvertUtils.convert("foo", Boolean.TYPE);
+ ...
+
+ value = LocaleConvertUtils.convert("foo", Boolean.class);
+ ...
+ */
+
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Byte.TYPE);
+ fail("Should have thrown conversion exception (1)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Byte.class);
+ fail("Should have thrown conversion exception (2)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ /* fixme - not implemented
+ try {
+ value = LocaleConvertUtils.convert("org.apache.commons.beanutils.Undefined", Class.class);
+ fail("Should have thrown conversion exception");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+ */
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Double.TYPE);
+ fail("Should have thrown conversion exception (3)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Double.class);
+ fail("Should have thrown conversion exception (4)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Float.TYPE);
+ fail("Should have thrown conversion exception (5)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Float.class);
+ fail("Should have thrown conversion exception (6)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Integer.TYPE);
+ fail("Should have thrown conversion exception (7)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Integer.class);
+ fail("Should have thrown conversion exception (8)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Byte.TYPE);
+ fail("Should have thrown conversion exception (9)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Long.class);
+ fail("Should have thrown conversion exception (10)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Short.TYPE);
+ fail("Should have thrown conversion exception (11)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ try {
+ value = LocaleConvertUtils.convert("foo", Short.class);
+ fail("Should have thrown conversion exception (12)");
+ } catch (ConversionException e) {
+ ; // Expected result
+ }
+
+ }
+
+
+ /**
+ * Negative String to String array tests.
+ */
+ public void fixmetestNegativeStringArray() {
+
+ fail("Array conversions not implemented yet.");
+
+ Object value = null;
+ String stringArray[] = new String[0];
+
+ value = LocaleConvertUtils.convert((String) null, stringArray.getClass());
+ checkStringArray(value, stringArray);
+ }
+
+
+ /**
+ * Test conversion of object to string for arrays - .
+ */
+ public void fixmetestObjectToStringArray() {
+
+ fail("Array conversions not implemented yet.");
+ int intArray0[] = new int[0];
+ int intArray1[] = {123};
+ int intArray2[] = {123, 456};
+ String stringArray0[] = new String[0];
+ String stringArray1[] = {"abc"};
+ String stringArray2[] = {"abc", "def"};
+
+ assertEquals("intArray0", null,
+ LocaleConvertUtils.convert(intArray0));
+ assertEquals("intArray1", "123",
+ LocaleConvertUtils.convert(intArray1));
+ assertEquals("intArray2", "123",
+ LocaleConvertUtils.convert(intArray2));
+
+ assertEquals("stringArray0", null,
+ LocaleConvertUtils.convert(stringArray0));
+ assertEquals("stringArray1", "abc",
+ LocaleConvertUtils.convert(stringArray1));
+ assertEquals("stringArray2", "abc",
+ LocaleConvertUtils.convert(stringArray2));
+
+ }
+
+
+ /**
+ * Test conversion of object to string for scalars.
+ */
+ public void testObjectToStringScalar() {
+
+ assertEquals("Boolean->String", "false",
+ LocaleConvertUtils.convert(Boolean.FALSE));
+ assertEquals("Boolean->String", "true",
+ LocaleConvertUtils.convert(Boolean.TRUE));
+ assertEquals("Byte->String", "123",
+ LocaleConvertUtils.convert(new Byte((byte) 123)));
+ assertEquals("Character->String", "a",
+ LocaleConvertUtils.convert(new Character('a')));
+ assertEquals("Double->String", "123" + m_decimalSeparator + "4",
+ LocaleConvertUtils.convert(new Double((double) 123.4)));
+ assertEquals("Float->String", "123" + m_decimalSeparator + "4",
+ LocaleConvertUtils.convert(new Float((float) 123.4)));
+ assertEquals("Integer->String", "123",
+ LocaleConvertUtils.convert(new Integer((int) 123)));
+ assertEquals("Long->String", "123",
+ LocaleConvertUtils.convert(new Long((long) 123)));
+ assertEquals("Short->String", "123",
+ LocaleConvertUtils.convert(new Short((short) 123)));
+ assertEquals("String->String", "abc",
+ LocaleConvertUtils.convert("abc"));
+ assertEquals("String->String null", null,
+ LocaleConvertUtils.convert(null));
+
+ }
+
+
+ /**
+ * Positive array conversion tests.
+ */
+ public void fixmetestPositiveArray() {
+
+ fail("Array conversions not implemented yet.");
+
+ String values1[] = {"10", "20", "30"};
+ Object value = LocaleConvertUtils.convert(values1, Integer.TYPE);
+ int shape[] = new int[0];
+ assertEquals(shape.getClass(), value.getClass());
+ int results1[] = (int[]) value;
+ assertEquals(results1[0], 10);
+ assertEquals(results1[1], 20);
+ assertEquals(results1[2], 30);
+
+ String values2[] = {"100", "200", "300"};
+ value = LocaleConvertUtils.convert(values2, shape.getClass());
+ assertEquals(shape.getClass(), value.getClass());
+ int results2[] = (int[]) value;
+ assertEquals(results2[0], 100);
+ assertEquals(results2[1], 200);
+ assertEquals(results2[2], 300);
+ }
+
+
+ /**
+ * Positive String to primitive integer array tests.
+ */
+ public void fixmetestPositiveIntegerArray() {
+
+ fail("Array conversions not implemented yet.");
+
+ Object value = null;
+ int intArray[] = new int[0];
+ int intArray1[] = new int[]{0};
+ int intArray2[] = new int[]{0, 10};
+
+ value = LocaleConvertUtils.convert("{ }", intArray.getClass());
+ checkIntegerArray(value, intArray);
+
+ value = LocaleConvertUtils.convert("0", intArray.getClass());
+ checkIntegerArray(value, intArray1);
+ value = LocaleConvertUtils.convert(" 0 ", intArray.getClass());
+ checkIntegerArray(value, intArray1);
+ value = LocaleConvertUtils.convert("{ 0 }", intArray.getClass());
+ checkIntegerArray(value, intArray1);
+
+ value = LocaleConvertUtils.convert("0,10", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = LocaleConvertUtils.convert("0 10", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = LocaleConvertUtils.convert("{0,10}", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = LocaleConvertUtils.convert("{0 10}", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = LocaleConvertUtils.convert("{ 0, 10 }", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ value = LocaleConvertUtils.convert("{ 0 10 }", intArray.getClass());
+ checkIntegerArray(value, intArray2);
+ }
+
+
+ /**
+ * Positive scalar conversion tests.
+ */
+ public void testPositiveScalar() {
+ Object value = null;
+
+ /* fixme Boolean converters not implemented
+ value = LocaleConvertUtils.convert("true", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = LocaleConvertUtils.convert("true", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = LocaleConvertUtils.convert("yes", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = LocaleConvertUtils.convert("yes", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = LocaleConvertUtils.convert("y", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = LocaleConvertUtils.convert("y", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = LocaleConvertUtils.convert("on", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = LocaleConvertUtils.convert("on", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), true);
+
+ value = LocaleConvertUtils.convert("false", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = LocaleConvertUtils.convert("false", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = LocaleConvertUtils.convert("no", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = LocaleConvertUtils.convert("no", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = LocaleConvertUtils.convert("n", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = LocaleConvertUtils.convert("n", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = LocaleConvertUtils.convert("off", Boolean.TYPE);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+
+ value = LocaleConvertUtils.convert("off", Boolean.class);
+ assertTrue(value instanceof Boolean);
+ assertEquals(((Boolean) value).booleanValue(), false);
+ */
+
+ value = LocaleConvertUtils.convert("123", Byte.TYPE);
+ assertTrue(value instanceof Byte);
+ assertEquals(((Byte) value).byteValue(), (byte) 123);
+
+ value = LocaleConvertUtils.convert("123", Byte.class);
+ assertTrue(value instanceof Byte);
+ assertEquals(((Byte) value).byteValue(), (byte) 123);
+
+ /*fixme Character conversion not implemented yet
+ value = LocaleConvertUtils.convert("a", Character.TYPE);
+ assertTrue(value instanceof Character);
+ assertEquals(((Character) value).charValue(), 'a');
+
+ value = LocaleConvertUtils.convert("a", Character.class);
+ assertTrue(value instanceof Character);
+ assertEquals(((Character) value).charValue(), 'a');
+ */
+ /* fixme - this is a discrepancy with standard converters ( probably not major issue )
+ value = LocaleConvertUtils.convert("java.lang.String", Class.class);
+ assertTrue(value instanceof Class);
+ assertEquals(String.class, (Class) value);
+ */
+
+ value = LocaleConvertUtils.convert("123" + m_decimalSeparator + "456", Double.TYPE);
+ assertTrue(value instanceof Double);
+ assertEquals(((Double) value).doubleValue(), (double) 123.456,
+ (double) 0.005);
+
+ value = LocaleConvertUtils.convert("123" + m_decimalSeparator + "456", Double.class);
+ assertTrue(value instanceof Double);
+ assertEquals(((Double) value).doubleValue(), (double) 123.456,
+ (double) 0.005);
+
+ value = LocaleConvertUtils.convert("123" + m_decimalSeparator + "456", Float.TYPE);
+ assertTrue(value instanceof Float);
+ assertEquals(((Float) value).floatValue(), (float) 123.456,
+ (float) 0.005);
+
+ value = LocaleConvertUtils.convert("123" + m_decimalSeparator + "456", Float.class);
+ assertTrue(value instanceof Float);
+ assertEquals(((Float) value).floatValue(), (float) 123.456,
+ (float) 0.005);
+
+ value = LocaleConvertUtils.convert("123", Integer.TYPE);
+ assertTrue(value instanceof Integer);
+ assertEquals(((Integer) value).intValue(), (int) 123);
+
+ value = LocaleConvertUtils.convert("123", Integer.class);
+ assertTrue(value instanceof Integer);
+ assertEquals(((Integer) value).intValue(), (int) 123);
+
+ value = LocaleConvertUtils.convert("123", Long.TYPE);
+ assertTrue(value instanceof Long);
+ assertEquals(((Long) value).longValue(), (long) 123);
+
+ value = LocaleConvertUtils.convert("123456", Long.class);
+ assertTrue(value instanceof Long);
+ assertEquals(((Long) value).longValue(), (long) 123456);
+
+ /* fixme - Short conversion not implemented at this point
+ value = LocaleConvertUtils.convert("123", Short.TYPE);
+ assertTrue(value instanceof Short);
+ assertEquals(((Short) value).shortValue(), (short) 123);
+
+ value = LocaleConvertUtils.convert("123", Short.class);
+ assertTrue(value instanceof Short);
+ assertEquals(((Short) value).shortValue(), (short) 123);
+ */
+
+ String input = null;
+
+ input = "2002-03-17";
+ value = LocaleConvertUtils.convert(input, Date.class);
+ assertTrue(value instanceof Date);
+ assertEquals(input, value.toString());
+
+ input = "20:30:40";
+ value = LocaleConvertUtils.convert(input, Time.class);
+ assertTrue(value instanceof Time);
+ assertEquals(input, value.toString());
+
+ input = "2002-03-17 20:30:40.0";
+ value = LocaleConvertUtils.convert(input, Timestamp.class);
+ assertTrue(value instanceof Timestamp);
+ assertEquals(input, value.toString());
+
+ }
+
+
+ /**
+ * Positive String to String array tests.
+ */
+ public void fixmetestPositiveStringArray() {
+
+ fail("Array conversions not implemented yet.");
+
+ Object value = null;
+ String stringArray[] = new String[0];
+ String stringArray1[] = new String[]
+ {"abc"};
+ String stringArray2[] = new String[]
+ {"abc", "de,f"};
+
+ value = LocaleConvertUtils.convert("", stringArray.getClass());
+ checkStringArray(value, stringArray);
+ value = LocaleConvertUtils.convert(" ", stringArray.getClass());
+ checkStringArray(value, stringArray);
+ value = LocaleConvertUtils.convert("{}", stringArray.getClass());
+ checkStringArray(value, stringArray);
+ value = LocaleConvertUtils.convert("{ }", stringArray.getClass());
+ checkStringArray(value, stringArray);
+
+ value = LocaleConvertUtils.convert("abc", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = LocaleConvertUtils.convert("{abc}", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = LocaleConvertUtils.convert("\"abc\"", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = LocaleConvertUtils.convert("{\"abc\"}", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = LocaleConvertUtils.convert("'abc'", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+ value = LocaleConvertUtils.convert("{'abc'}", stringArray.getClass());
+ checkStringArray(value, stringArray1);
+
+ value = LocaleConvertUtils.convert("abc 'de,f'",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = LocaleConvertUtils.convert("{abc, 'de,f'}",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = LocaleConvertUtils.convert("\"abc\",\"de,f\"",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = LocaleConvertUtils.convert("{\"abc\" 'de,f'}",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = LocaleConvertUtils.convert("'abc' 'de,f'",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+ value = LocaleConvertUtils.convert("{'abc', \"de,f\"}",
+ stringArray.getClass());
+ checkStringArray(value, stringArray2);
+
+ }
+
+
+ // -------------------------------------------------------- Private Methods
+
+
+ private void checkIntegerArray(Object value, int intArray[]) {
+
+ assertNotNull("Returned value is not null", value);
+ assertEquals("Returned value is int[]",
+ intArray.getClass(), value.getClass());
+ int results[] = (int[]) value;
+ assertEquals("Returned array length", intArray.length, results.length);
+ for (int i = 0; i < intArray.length; i++) {
+ assertEquals("Returned array value " + i,
+ intArray[i], results[i]);
+ }
+
+ }
+
+
+ private void checkStringArray(Object value, String stringArray[]) {
+
+ assertNotNull("Returned value is not null", value);
+ assertEquals("Returned value is String[]",
+ stringArray.getClass(), value.getClass());
+ String results[] = (String[]) value;
+ assertEquals("Returned array length",
+ stringArray.length, results.length);
+ for (int i = 0; i < stringArray.length; i++) {
+ assertEquals("Returned array value " + i,
+ stringArray[i], results[i]);
+ }
+
+ }
+
+
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/locale/converters/DateLocaleConverterTestCase.java b/trunk/src/test/org/apache/commons/beanutils/locale/converters/DateLocaleConverterTestCase.java
new file mode 100644
index 0000000..3906028
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/locale/converters/DateLocaleConverterTestCase.java
@@ -0,0 +1,143 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package org.apache.commons.beanutils.locale.converters;
+
+import junit.framework.TestSuite;
+import junit.framework.TestCase;
+
+import java.text.SimpleDateFormat;
+import java.text.ParseException;
+
+import java.util.Locale;
+
+import org.apache.commons.beanutils.Converter;
+import org.apache.commons.beanutils.ConversionException;
+
+/**
+ * Test Case for the DateLocaleConverter class.
+ *
+ * @author Robert Burrell Donkin
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class DateLocaleConverterTestCase extends TestCase {
+
+ // ------------------------------------------------------------------------
+
+ public DateLocaleConverterTestCase(String name) {
+ super(name);
+ }
+
+ // ------------------------------------------------------------------------
+
+ public static TestSuite suite() {
+ return new TestSuite(DateLocaleConverterTestCase.class);
+ }
+
+
+ // ------------------------------------------------------------------------
+
+ public void testSetLenient() {
+ // make sure that date format works as expected
+ SimpleDateFormat dateFormat = new SimpleDateFormat("MMM dd, yyyy", Locale.UK);
+
+ // test with no leniency
+ dateFormat.setLenient(false);
+
+ try {
+
+ dateFormat.parse("Feb 10, 2001");
+
+ } catch (ParseException e) {
+ fail("Could not parse date (1) - " + e.getMessage());
+ }
+
+ try {
+
+ dateFormat.parse("Feb 31, 2001");
+ fail("Parsed illegal date (1)");
+
+ } catch (ParseException e) {
+ // that's what we expected
+ }
+
+ // test with leniency
+ dateFormat.setLenient(true);
+
+ try {
+
+ dateFormat.parse("Feb 10, 2001");
+
+ } catch (ParseException e) {
+ fail("Could not parse date (2) - " + e.getMessage());
+ }
+
+ try {
+
+ dateFormat.parse("Feb 31, 2001");
+
+ } catch (ParseException e) {
+ fail("Could not parse date (3) - " + e.getMessage());
+ }
+
+ // now repeat tests for converter
+ DateLocaleConverter converter = new DateLocaleConverter(Locale.UK, "MMM dd, yyyy");
+
+ // test with no leniency
+ converter.setLenient(false);
+ assertEquals("Set lenient failed", converter.isLenient(), false);
+
+ try {
+
+ converter.convert("Feb 10, 2001");
+
+ } catch (ConversionException e) {
+ fail("Could not parse date (4) - " + e.getMessage());
+ }
+
+ try {
+
+ converter.convert("Feb 31, 2001");
+ assertEquals("Set lenient failed", converter.isLenient(), false);
+ fail("Parsed illegal date (2)");
+
+ } catch (ConversionException e) {
+ // that's what we expected
+ }
+
+ // test with leniency
+ converter.setLenient(true);
+ assertEquals("Set lenient failed", converter.isLenient(), true);
+
+ try {
+
+ converter.convert("Feb 10, 2001");
+
+ } catch (ConversionException e) {
+ fail("Could not parse date (5) - " + e.getMessage());
+ }
+
+ try {
+
+ converter.convert("Feb 31, 2001");
+
+ } catch (ConversionException e) {
+ fail("Could not parse date (6) - " + e.getMessage());
+ }
+ }
+}
+
diff --git a/trunk/src/test/org/apache/commons/beanutils/priv/PackageBean.java b/trunk/src/test/org/apache/commons/beanutils/priv/PackageBean.java
new file mode 100644
index 0000000..0256c57
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/priv/PackageBean.java
@@ -0,0 +1,69 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.priv;
+
+
+/**
+ * <p>This class is designed to test the default access jvm problem workaround.
+ * The issue is that public methods of a public subclass contained in a default access
+ * superclass are returned by reflection but an IllegalAccessException is thrown
+ * when they are invoked.</p>
+ *
+ * <p>This is the default access superclass</p>
+ *
+ * @author Robert Burrell Donkin
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+class PackageBean {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Package private constructor - can only use factory method to create
+ * beans.
+ */
+ PackageBean() {
+
+ super();
+
+ }
+
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ */
+ private String bar = "This is bar";
+
+ public String getBar() {
+
+ return (this.bar);
+
+ }
+
+ public void setBar(String bar) {
+
+ this.bar = bar;
+
+ }
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/priv/PrivateBean.java b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateBean.java
new file mode 100644
index 0000000..bb40b84
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateBean.java
@@ -0,0 +1,110 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.priv;
+
+
+/**
+ * Bean that has a private constructor that exposes properties via
+ * various mechanisms (based on property name):
+ * <ul>
+ * <li><strong>foo</strong> - Via direct public method
+ * <li><strong>bar</strong> - Via directly implemented interface
+ * <li><strong>baz</strong> - Via indirectly implemented interface
+ * </ul>
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.6 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+class PrivateBean implements PrivateDirect {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Package private constructor - can only use factory method to create
+ * beans.
+ */
+ PrivateBean() {
+
+ super();
+
+ }
+
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ * A directly implemented property.
+ */
+ private String foo = "This is foo";
+
+ public String getFoo() {
+
+ return (this.foo);
+
+ }
+
+
+ /**
+ * A property accessible via a directly implemented interface.
+ */
+ private String bar = "This is bar";
+
+ public String getBar() {
+
+ return (this.bar);
+
+ }
+
+
+ /**
+ * A method accessible via a directly implemented interface.
+ */
+ public String methodBar(String in) {
+
+ return (in);
+
+ }
+
+
+ /**
+ * A property accessible via an indirectly implemented interface.
+ */
+ private String baz = "This is baz";
+
+ public String getBaz() {
+
+ return (this.baz);
+
+ }
+
+
+ /**
+ * A method accessible via an indirectly implemented interface.
+ */
+ public String methodBaz(String in) {
+
+ return (in);
+
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/priv/PrivateBeanFactory.java b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateBeanFactory.java
new file mode 100644
index 0000000..65ddd3b
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateBeanFactory.java
@@ -0,0 +1,52 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.priv;
+
+
+/**
+ * Factory class for PrivateBean instances.
+ *
+ * @author Craig R. McClanahan
+ * @author Jan Sorensen
+ * @version $Revision: 1.6 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class PrivateBeanFactory {
+
+
+ /**
+ * Factory method to create new beans.
+ */
+ public static PrivateDirect create() {
+
+ return (new PrivateBean());
+
+ }
+
+
+ /**
+ * Factory method to create new beans.
+ */
+ public static PrivateDirect createSubclass() {
+
+ return (new PrivateBeanSubclass());
+
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/priv/PrivateBeanSubclass.java b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateBeanSubclass.java
new file mode 100644
index 0000000..c35e872
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateBeanSubclass.java
@@ -0,0 +1,58 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.priv;
+
+
+/**
+ * Bean that exposes methods defined by an interface that is implemented
+ * in the superclass.
+ *
+ * @author Jan Sorensen
+ * @version $Revision: 1.5 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+class PrivateBeanSubclass extends PrivateBean {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Create a new PrivateBeanSubclass instance.
+ */
+ PrivateBeanSubclass() {
+
+ super();
+
+ }
+
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ * A property accessible via the superclass.
+ */
+ public String getBar() {
+
+ return (super.getBar());
+
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/priv/PrivateDirect.java b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateDirect.java
new file mode 100644
index 0000000..b290069
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateDirect.java
@@ -0,0 +1,45 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.priv;
+
+
+/**
+ * Interface that is directly implemented by PrivateBean.
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.6 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public interface PrivateDirect extends PrivateIndirect {
+
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ * A property accessible via a directly implemented interface.
+ */
+ String getBar();
+
+
+ /**
+ * A method accessible via a directly implemented interface.
+ */
+ String methodBar(String in);
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/priv/PrivateIndirect.java b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateIndirect.java
new file mode 100644
index 0000000..ecddcd4
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/priv/PrivateIndirect.java
@@ -0,0 +1,46 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.priv;
+
+
+/**
+ * Interface that is indirectly implemented by PrivateBean.
+ *
+ * @author Craig R. McClanahan
+ * @version $Revision: 1.6 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public interface PrivateIndirect {
+
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ * A property accessible via an indirectly implemented interface.
+ */
+ public String getBaz();
+
+
+ /**
+ * A method accessible via an indirectly implemented interface.
+ */
+ public String methodBaz(String in);
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/beanutils/priv/PublicSubBean.java b/trunk/src/test/org/apache/commons/beanutils/priv/PublicSubBean.java
new file mode 100644
index 0000000..7a41c20
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/beanutils/priv/PublicSubBean.java
@@ -0,0 +1,69 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+
+package org.apache.commons.beanutils.priv;
+
+
+/**
+ * <p>This class is designed to test the default access jvm problem workaround.
+ * The issue is that public methods of a public subclass contained in a default access
+ * superclass are returned by reflection but an IllegalAccessException is thrown
+ * when they are invoked.</p>
+ *
+ * <p>This is the default access superclass</p>
+ *
+ * @author Robert Burrell Donkin
+ * @version $Revision: 1.4 $ $Date: 2004/02/28 13:18:37 $
+ */
+
+public class PublicSubBean extends PackageBean {
+
+
+ // ----------------------------------------------------------- Constructors
+
+
+ /**
+ * Package private constructor - can only use factory method to create
+ * beans.
+ */
+ public PublicSubBean() {
+
+ super();
+
+ }
+
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ * A directly implemented property.
+ */
+ private String foo = "This is foo";
+
+ public String getFoo() {
+
+ return (this.foo);
+
+ }
+
+ public void setFoo(String foo) {
+
+ this.foo = foo;
+
+ }
+}
diff --git a/trunk/src/test/org/apache/commons/collections/ReferenceMap.java b/trunk/src/test/org/apache/commons/collections/ReferenceMap.java
new file mode 100644
index 0000000..acf54b7
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/collections/ReferenceMap.java
@@ -0,0 +1,957 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.collections;
+
+import java.io.IOException;
+import java.io.ObjectInputStream;
+import java.io.ObjectOutputStream;
+import java.lang.ref.Reference;
+import java.lang.ref.ReferenceQueue;
+import java.lang.ref.SoftReference;
+import java.lang.ref.WeakReference;
+import java.util.AbstractCollection;
+import java.util.AbstractMap;
+import java.util.AbstractSet;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.ConcurrentModificationException;
+import java.util.Iterator;
+import java.util.Map;
+import java.util.NoSuchElementException;
+import java.util.Set;
+
+import org.apache.commons.collections.keyvalue.DefaultMapEntry;
+
+/**
+ * Hash-based {@link Map} implementation that allows
+ * mappings to be removed by the garbage collector.<p>
+ *
+ * When you construct a <code>ReferenceMap</code>, you can
+ * specify what kind of references are used to store the
+ * map's keys and values. If non-hard references are
+ * used, then the garbage collector can remove mappings
+ * if a key or value becomes unreachable, or if the
+ * JVM's memory is running low. For information on how
+ * the different reference types behave, see
+ * {@link Reference}.<p>
+ *
+ * Different types of references can be specified for keys
+ * and values. The keys can be configured to be weak but
+ * the values hard, in which case this class will behave
+ * like a <a href="http://java.sun.com/j2se/1.4/docs/api/java/util/WeakHashMap.html">
+ * <code>WeakHashMap</code></a>. However, you
+ * can also specify hard keys and weak values, or any other
+ * combination. The default constructor uses hard keys
+ * and soft values, providing a memory-sensitive cache.<p>
+ *
+ * The algorithms used are basically the same as those
+ * in {@link java.util.HashMap}. In particular, you
+ * can specify a load factor and capacity to suit your
+ * needs. All optional {@link Map} operations are
+ * supported.<p>
+ *
+ * However, this {@link Map} implementation does <I>not</I>
+ * allow null elements. Attempting to add a null key or
+ * or a null value to the map will raise a
+ * <code>NullPointerException</code>.<p>
+ *
+ * As usual, this implementation is not synchronized. You
+ * can use {@link java.util.Collections#synchronizedMap} to
+ * provide synchronized access to a <code>ReferenceMap</code>.
+ *
+ * @see java.lang.ref.Reference
+ *
+ * @deprecated Moved to map subpackage. Due to be removed in v4.0.
+ * @since Commons Collections 2.1
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 19:52:42 $
+ *
+ * @author Paul Jack
+ */
+public class ReferenceMap extends AbstractMap {
+
+ /**
+ * For serialization.
+ */
+ final private static long serialVersionUID = -3370601314380922368L;
+
+
+ /**
+ * Constant indicating that hard references should be used.
+ */
+ final public static int HARD = 0;
+
+
+ /**
+ * Constant indicating that soft references should be used.
+ */
+ final public static int SOFT = 1;
+
+
+ /**
+ * Constant indicating that weak references should be used.
+ */
+ final public static int WEAK = 2;
+
+
+ // --- serialized instance variables:
+
+
+ /**
+ * The reference type for keys. Must be HARD, SOFT, WEAK.
+ * Note: I originally marked this field as final, but then this class
+ * didn't compile under JDK1.2.2.
+ * @serial
+ */
+ private int keyType;
+
+
+ /**
+ * The reference type for values. Must be HARD, SOFT, WEAK.
+ * Note: I originally marked this field as final, but then this class
+ * didn't compile under JDK1.2.2.
+ * @serial
+ */
+ private int valueType;
+
+
+ /**
+ * The threshold variable is calculated by multiplying
+ * table.length and loadFactor.
+ * Note: I originally marked this field as final, but then this class
+ * didn't compile under JDK1.2.2.
+ * @serial
+ */
+ private float loadFactor;
+
+ /**
+ * Should the value be automatically purged when the associated key has been collected?
+ */
+ private boolean purgeValues = false;
+
+
+ // -- Non-serialized instance variables
+
+ /**
+ * ReferenceQueue used to eliminate stale mappings.
+ * See purge.
+ */
+ private transient ReferenceQueue queue = new ReferenceQueue();
+
+
+ /**
+ * The hash table. Its length is always a power of two.
+ */
+ private transient Entry[] table;
+
+
+ /**
+ * Number of mappings in this map.
+ */
+ private transient int size;
+
+
+ /**
+ * When size reaches threshold, the map is resized.
+ * See resize().
+ */
+ private transient int threshold;
+
+
+ /**
+ * Number of times this map has been modified.
+ */
+ private transient volatile int modCount;
+
+
+ /**
+ * Cached key set. May be null if key set is never accessed.
+ */
+ private transient Set keySet;
+
+
+ /**
+ * Cached entry set. May be null if entry set is never accessed.
+ */
+ private transient Set entrySet;
+
+
+ /**
+ * Cached values. May be null if values() is never accessed.
+ */
+ private transient Collection values;
+
+
+ /**
+ * Constructs a new <code>ReferenceMap</code> that will
+ * use hard references to keys and soft references to values.
+ */
+ public ReferenceMap() {
+ this(HARD, SOFT);
+ }
+
+ /**
+ * Constructs a new <code>ReferenceMap</code> that will
+ * use the specified types of references.
+ *
+ * @param keyType the type of reference to use for keys;
+ * must be {@link #HARD}, {@link #SOFT}, {@link #WEAK}
+ * @param valueType the type of reference to use for values;
+ * must be {@link #HARD}, {@link #SOFT}, {@link #WEAK}
+ * @param purgeValues should the value be automatically purged when the
+ * key is garbage collected
+ */
+ public ReferenceMap(int keyType, int valueType, boolean purgeValues) {
+ this(keyType, valueType);
+ this.purgeValues = purgeValues;
+ }
+
+ /**
+ * Constructs a new <code>ReferenceMap</code> that will
+ * use the specified types of references.
+ *
+ * @param keyType the type of reference to use for keys;
+ * must be {@link #HARD}, {@link #SOFT}, {@link #WEAK}
+ * @param valueType the type of reference to use for values;
+ * must be {@link #HARD}, {@link #SOFT}, {@link #WEAK}
+ */
+ public ReferenceMap(int keyType, int valueType) {
+ this(keyType, valueType, 16, 0.75f);
+ }
+
+ /**
+ * Constructs a new <code>ReferenceMap</code> with the
+ * specified reference types, load factor and initial
+ * capacity.
+ *
+ * @param keyType the type of reference to use for keys;
+ * must be {@link #HARD}, {@link #SOFT}, {@link #WEAK}
+ * @param valueType the type of reference to use for values;
+ * must be {@link #HARD}, {@link #SOFT}, {@link #WEAK}
+ * @param capacity the initial capacity for the map
+ * @param loadFactor the load factor for the map
+ * @param purgeValues should the value be automatically purged when the
+ * key is garbage collected
+ */
+ public ReferenceMap(
+ int keyType,
+ int valueType,
+ int capacity,
+ float loadFactor,
+ boolean purgeValues) {
+ this(keyType, valueType, capacity, loadFactor);
+ this.purgeValues = purgeValues;
+ }
+
+ /**
+ * Constructs a new <code>ReferenceMap</code> with the
+ * specified reference types, load factor and initial
+ * capacity.
+ *
+ * @param keyType the type of reference to use for keys;
+ * must be {@link #HARD}, {@link #SOFT}, {@link #WEAK}
+ * @param valueType the type of reference to use for values;
+ * must be {@link #HARD}, {@link #SOFT}, {@link #WEAK}
+ * @param capacity the initial capacity for the map
+ * @param loadFactor the load factor for the map
+ */
+ public ReferenceMap(int keyType, int valueType, int capacity, float loadFactor) {
+ super();
+
+ verify("keyType", keyType);
+ verify("valueType", valueType);
+
+ if (capacity <= 0) {
+ throw new IllegalArgumentException("capacity must be positive");
+ }
+ if ((loadFactor <= 0.0f) || (loadFactor >= 1.0f)) {
+ throw new IllegalArgumentException("Load factor must be greater than 0 and less than 1.");
+ }
+
+ this.keyType = keyType;
+ this.valueType = valueType;
+
+ int v = 1;
+ while (v < capacity) v *= 2;
+
+ this.table = new Entry[v];
+ this.loadFactor = loadFactor;
+ this.threshold = (int)(v * loadFactor);
+ }
+
+
+ // used by constructor
+ private static void verify(String name, int type) {
+ if ((type < HARD) || (type > WEAK)) {
+ throw new IllegalArgumentException(name +
+ " must be HARD, SOFT, WEAK.");
+ }
+ }
+
+
+ /**
+ * Writes this object to the given output stream.
+ *
+ * @param out the output stream to write to
+ * @throws IOException if the stream raises it
+ */
+ private void writeObject(ObjectOutputStream out) throws IOException {
+ out.defaultWriteObject();
+ out.writeInt(table.length);
+
+ // Have to use null-terminated list because size might shrink
+ // during iteration
+
+ for (Iterator iter = entrySet().iterator(); iter.hasNext();) {
+ Map.Entry entry = (Map.Entry)iter.next();
+ out.writeObject(entry.getKey());
+ out.writeObject(entry.getValue());
+ }
+ out.writeObject(null);
+ }
+
+
+ /**
+ * Reads the contents of this object from the given input stream.
+ *
+ * @param inp the input stream to read from
+ * @throws IOException if the stream raises it
+ * @throws ClassNotFoundException if the stream raises it
+ */
+ private void readObject(ObjectInputStream inp) throws IOException, ClassNotFoundException {
+ inp.defaultReadObject();
+ table = new Entry[inp.readInt()];
+ threshold = (int)(table.length * loadFactor);
+ queue = new ReferenceQueue();
+ Object key = inp.readObject();
+ while (key != null) {
+ Object value = inp.readObject();
+ put(key, value);
+ key = inp.readObject();
+ }
+ }
+
+
+ /**
+ * Constructs a reference of the given type to the given
+ * referent. The reference is registered with the queue
+ * for later purging.
+ *
+ * @param type HARD, SOFT or WEAK
+ * @param referent the object to refer to
+ * @param hash the hash code of the <I>key</I> of the mapping;
+ * this number might be different from referent.hashCode() if
+ * the referent represents a value and not a key
+ */
+ private Object toReference(int type, Object referent, int hash) {
+ switch (type) {
+ case HARD: return referent;
+ case SOFT: return new SoftRef(hash, referent, queue);
+ case WEAK: return new WeakRef(hash, referent, queue);
+ default: throw new Error();
+ }
+ }
+
+
+ /**
+ * Returns the entry associated with the given key.
+ *
+ * @param key the key of the entry to look up
+ * @return the entry associated with that key, or null
+ * if the key is not in this map
+ */
+ private Entry getEntry(Object key) {
+ if (key == null) return null;
+ int hash = key.hashCode();
+ int index = indexFor(hash);
+ for (Entry entry = table[index]; entry != null; entry = entry.next) {
+ if ((entry.hash == hash) && key.equals(entry.getKey())) {
+ return entry;
+ }
+ }
+ return null;
+ }
+
+
+ /**
+ * Converts the given hash code into an index into the
+ * hash table.
+ */
+ private int indexFor(int hash) {
+ // mix the bits to avoid bucket collisions...
+ hash += ~(hash << 15);
+ hash ^= (hash >>> 10);
+ hash += (hash << 3);
+ hash ^= (hash >>> 6);
+ hash += ~(hash << 11);
+ hash ^= (hash >>> 16);
+ return hash & (table.length - 1);
+ }
+
+
+
+ /**
+ * Resizes this hash table by doubling its capacity.
+ * This is an expensive operation, as entries must
+ * be copied from the old smaller table to the new
+ * bigger table.
+ */
+ private void resize() {
+ Entry[] old = table;
+ table = new Entry[old.length * 2];
+
+ for (int i = 0; i < old.length; i++) {
+ Entry next = old[i];
+ while (next != null) {
+ Entry entry = next;
+ next = next.next;
+ int index = indexFor(entry.hash);
+ entry.next = table[index];
+ table[index] = entry;
+ }
+ old[i] = null;
+ }
+ threshold = (int)(table.length * loadFactor);
+ }
+
+
+
+ /**
+ * Purges stale mappings from this map.
+ * <p>
+ * Ordinarily, stale mappings are only removed during
+ * a write operation, although this method is called for both
+ * read and write operations to maintain a consistent state.
+ * <p>
+ * Note that this method is not synchronized! Special
+ * care must be taken if, for instance, you want stale
+ * mappings to be removed on a periodic basis by some
+ * background thread.
+ */
+ private void purge() {
+ Reference ref = queue.poll();
+ while (ref != null) {
+ purge(ref);
+ ref = queue.poll();
+ }
+ }
+
+
+ private void purge(Reference ref) {
+ // The hashCode of the reference is the hashCode of the
+ // mapping key, even if the reference refers to the
+ // mapping value...
+ int hash = ref.hashCode();
+ int index = indexFor(hash);
+ Entry previous = null;
+ Entry entry = table[index];
+ while (entry != null) {
+ if (entry.purge(ref)) {
+ if (previous == null) table[index] = entry.next;
+ else previous.next = entry.next;
+ this.size--;
+ return;
+ }
+ previous = entry;
+ entry = entry.next;
+ }
+
+ }
+
+
+ /**
+ * Returns the size of this map.
+ *
+ * @return the size of this map
+ */
+ public int size() {
+ purge();
+ return size;
+ }
+
+
+ /**
+ * Returns <code>true</code> if this map is empty.
+ *
+ * @return <code>true</code> if this map is empty
+ */
+ public boolean isEmpty() {
+ purge();
+ return size == 0;
+ }
+
+
+ /**
+ * Returns <code>true</code> if this map contains the given key.
+ *
+ * @return true if the given key is in this map
+ */
+ public boolean containsKey(Object key) {
+ purge();
+ Entry entry = getEntry(key);
+ if (entry == null) return false;
+ return entry.getValue() != null;
+ }
+
+
+ /**
+ * Returns the value associated with the given key, if any.
+ *
+ * @return the value associated with the given key, or <code>null</code>
+ * if the key maps to no value
+ */
+ public Object get(Object key) {
+ purge();
+ Entry entry = getEntry(key);
+ if (entry == null) return null;
+ return entry.getValue();
+ }
+
+
+ /**
+ * Associates the given key with the given value.<p>
+ * Neither the key nor the value may be null.
+ *
+ * @param key the key of the mapping
+ * @param value the value of the mapping
+ * @return the last value associated with that key, or
+ * null if no value was associated with the key
+ * @throws NullPointerException if either the key or value
+ * is null
+ */
+ public Object put(Object key, Object value) {
+ if (key == null) throw new NullPointerException("null keys not allowed");
+ if (value == null) throw new NullPointerException("null values not allowed");
+
+ purge();
+ if (size + 1 > threshold) resize();
+
+ int hash = key.hashCode();
+ int index = indexFor(hash);
+ Entry entry = table[index];
+ while (entry != null) {
+ if ((hash == entry.hash) && key.equals(entry.getKey())) {
+ Object result = entry.getValue();
+ entry.setValue(value);
+ return result;
+ }
+ entry = entry.next;
+ }
+ this.size++;
+ modCount++;
+ key = toReference(keyType, key, hash);
+ value = toReference(valueType, value, hash);
+ table[index] = new Entry(key, hash, value, table[index]);
+ return null;
+ }
+
+
+ /**
+ * Removes the key and its associated value from this map.
+ *
+ * @param key the key to remove
+ * @return the value associated with that key, or null if
+ * the key was not in the map
+ */
+ public Object remove(Object key) {
+ if (key == null) return null;
+ purge();
+ int hash = key.hashCode();
+ int index = indexFor(hash);
+ Entry previous = null;
+ Entry entry = table[index];
+ while (entry != null) {
+ if ((hash == entry.hash) && key.equals(entry.getKey())) {
+ if (previous == null) table[index] = entry.next;
+ else previous.next = entry.next;
+ this.size--;
+ modCount++;
+ return entry.getValue();
+ }
+ previous = entry;
+ entry = entry.next;
+ }
+ return null;
+ }
+
+
+ /**
+ * Clears this map.
+ */
+ public void clear() {
+ Arrays.fill(table, null);
+ size = 0;
+ while (queue.poll() != null); // drain the queue
+ }
+
+
+ /**
+ * Returns a set view of this map's entries.
+ *
+ * @return a set view of this map's entries
+ */
+ public Set entrySet() {
+ if (entrySet != null) {
+ return entrySet;
+ }
+ entrySet = new AbstractSet() {
+ public int size() {
+ return ReferenceMap.this.size();
+ }
+
+ public void clear() {
+ ReferenceMap.this.clear();
+ }
+
+ public boolean contains(Object o) {
+ if (o == null) return false;
+ if (!(o instanceof Map.Entry)) return false;
+ Map.Entry e = (Map.Entry)o;
+ Entry e2 = getEntry(e.getKey());
+ return (e2 != null) && e.equals(e2);
+ }
+
+ public boolean remove(Object o) {
+ boolean r = contains(o);
+ if (r) {
+ Map.Entry e = (Map.Entry)o;
+ ReferenceMap.this.remove(e.getKey());
+ }
+ return r;
+ }
+
+ public Iterator iterator() {
+ return new EntryIterator();
+ }
+
+ public Object[] toArray() {
+ return toArray(new Object[0]);
+ }
+
+ public Object[] toArray(Object[] arr) {
+ ArrayList list = new ArrayList();
+ Iterator iterator = iterator();
+ while (iterator.hasNext()) {
+ Entry e = (Entry)iterator.next();
+ list.add(new DefaultMapEntry(e.getKey(), e.getValue()));
+ }
+ return list.toArray(arr);
+ }
+ };
+ return entrySet;
+ }
+
+
+ /**
+ * Returns a set view of this map's keys.
+ *
+ * @return a set view of this map's keys
+ */
+ public Set keySet() {
+ if (keySet != null) return keySet;
+ keySet = new AbstractSet() {
+ public int size() {
+ return ReferenceMap.this.size();
+ }
+
+ public Iterator iterator() {
+ return new KeyIterator();
+ }
+
+ public boolean contains(Object o) {
+ return containsKey(o);
+ }
+
+
+ public boolean remove(Object o) {
+ Object r = ReferenceMap.this.remove(o);
+ return r != null;
+ }
+
+ public void clear() {
+ ReferenceMap.this.clear();
+ }
+
+ public Object[] toArray() {
+ return toArray(new Object[0]);
+ }
+
+ public Object[] toArray(Object[] array) {
+ Collection c = new ArrayList(size());
+ for (Iterator it = iterator(); it.hasNext(); ) {
+ c.add(it.next());
+ }
+ return c.toArray(array);
+ }
+ };
+ return keySet;
+ }
+
+
+ /**
+ * Returns a collection view of this map's values.
+ *
+ * @return a collection view of this map's values.
+ */
+ public Collection values() {
+ if (values != null) return values;
+ values = new AbstractCollection() {
+ public int size() {
+ return ReferenceMap.this.size();
+ }
+
+ public void clear() {
+ ReferenceMap.this.clear();
+ }
+
+ public Iterator iterator() {
+ return new ValueIterator();
+ }
+
+ public Object[] toArray() {
+ return toArray(new Object[0]);
+ }
+
+ public Object[] toArray(Object[] array) {
+ Collection c = new ArrayList(size());
+ for (Iterator it = iterator(); it.hasNext(); ) {
+ c.add(it.next());
+ }
+ return c.toArray(array);
+ }
+ };
+ return values;
+ }
+
+
+ // If getKey() or getValue() returns null, it means
+ // the mapping is stale and should be removed.
+ private class Entry implements Map.Entry, KeyValue {
+
+ Object key;
+ Object value;
+ int hash;
+ Entry next;
+
+
+ public Entry(Object key, int hash, Object value, Entry next) {
+ this.key = key;
+ this.hash = hash;
+ this.value = value;
+ this.next = next;
+ }
+
+
+ public Object getKey() {
+ return (keyType > HARD) ? ((Reference)key).get() : key;
+ }
+
+
+ public Object getValue() {
+ return (valueType > HARD) ? ((Reference)value).get() : value;
+ }
+
+
+ public Object setValue(Object object) {
+ Object old = getValue();
+ if (valueType > HARD) ((Reference)value).clear();
+ value = toReference(valueType, object, hash);
+ return old;
+ }
+
+
+ public boolean equals(Object o) {
+ if (o == null) return false;
+ if (o == this) return true;
+ if (!(o instanceof Map.Entry)) return false;
+
+ Map.Entry entry = (Map.Entry)o;
+ Object key = entry.getKey();
+ Object value = entry.getValue();
+ if ((key == null) || (value == null)) return false;
+ return key.equals(getKey()) && value.equals(getValue());
+ }
+
+
+ public int hashCode() {
+ Object v = getValue();
+ return hash ^ ((v == null) ? 0 : v.hashCode());
+ }
+
+
+ public String toString() {
+ return getKey() + "=" + getValue();
+ }
+
+
+ boolean purge(Reference ref) {
+ boolean r = (keyType > HARD) && (key == ref);
+ r = r || ((valueType > HARD) && (value == ref));
+ if (r) {
+ if (keyType > HARD) ((Reference)key).clear();
+ if (valueType > HARD) {
+ ((Reference)value).clear();
+ } else if (purgeValues) {
+ value = null;
+ }
+ }
+ return r;
+ }
+ }
+
+
+ private class EntryIterator implements Iterator {
+ // These fields keep track of where we are in the table.
+ int index;
+ Entry entry;
+ Entry previous;
+
+ // These Object fields provide hard references to the
+ // current and next entry; this assures that if hasNext()
+ // returns true, next() will actually return a valid element.
+ Object nextKey, nextValue;
+ Object currentKey, currentValue;
+
+ int expectedModCount;
+
+
+ public EntryIterator() {
+ index = (size() != 0 ? table.length : 0);
+ // have to do this here! size() invocation above
+ // may have altered the modCount.
+ expectedModCount = modCount;
+ }
+
+
+ public boolean hasNext() {
+ checkMod();
+ while (nextNull()) {
+ Entry e = entry;
+ int i = index;
+ while ((e == null) && (i > 0)) {
+ i--;
+ e = table[i];
+ }
+ entry = e;
+ index = i;
+ if (e == null) {
+ currentKey = null;
+ currentValue = null;
+ return false;
+ }
+ nextKey = e.getKey();
+ nextValue = e.getValue();
+ if (nextNull()) entry = entry.next;
+ }
+ return true;
+ }
+
+
+ private void checkMod() {
+ if (modCount != expectedModCount) {
+ throw new ConcurrentModificationException();
+ }
+ }
+
+
+ private boolean nextNull() {
+ return (nextKey == null) || (nextValue == null);
+ }
+
+ protected Entry nextEntry() {
+ checkMod();
+ if (nextNull() && !hasNext()) throw new NoSuchElementException();
+ previous = entry;
+ entry = entry.next;
+ currentKey = nextKey;
+ currentValue = nextValue;
+ nextKey = null;
+ nextValue = null;
+ return previous;
+ }
+
+
+ public Object next() {
+ return nextEntry();
+ }
+
+
+ public void remove() {
+ checkMod();
+ if (previous == null) throw new IllegalStateException();
+ ReferenceMap.this.remove(currentKey);
+ previous = null;
+ currentKey = null;
+ currentValue = null;
+ expectedModCount = modCount;
+ }
+
+ }
+
+
+ private class ValueIterator extends EntryIterator {
+ public Object next() {
+ return nextEntry().getValue();
+ }
+ }
+
+
+ private class KeyIterator extends EntryIterator {
+ public Object next() {
+ return nextEntry().getKey();
+ }
+ }
+
+
+
+ // These two classes store the hashCode of the key of
+ // of the mapping, so that after they're dequeued a quick
+ // lookup of the bucket in the table can occur.
+
+
+ private static class SoftRef extends SoftReference {
+ private int hash;
+
+
+ public SoftRef(int hash, Object r, ReferenceQueue q) {
+ super(r, q);
+ this.hash = hash;
+ }
+
+
+ public int hashCode() {
+ return hash;
+ }
+ }
+
+
+ private static class WeakRef extends WeakReference {
+ private int hash;
+
+
+ public WeakRef(int hash, Object r, ReferenceQueue q) {
+ super(r, q);
+ this.hash = hash;
+ }
+
+
+ public int hashCode() {
+ return hash;
+ }
+ }
+
+
+}
diff --git a/trunk/src/test/org/apache/commons/collections/keyvalue/AbstractKeyValue.java b/trunk/src/test/org/apache/commons/collections/keyvalue/AbstractKeyValue.java
new file mode 100644
index 0000000..1130df2
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/collections/keyvalue/AbstractKeyValue.java
@@ -0,0 +1,81 @@
+/*
+ * Copyright 2003-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.collections.keyvalue;
+
+import org.apache.commons.collections.KeyValue;
+
+/**
+ * Abstract pair class to assist with creating KeyValue and MapEntry implementations.
+ *
+ * @since Commons Collections 3.0
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 19:53:54 $
+ *
+ * @author James Strachan
+ * @author Michael A. Smith
+ * @author Neil O'Toole
+ * @author Stephen Colebourne
+ */
+public abstract class AbstractKeyValue implements KeyValue {
+
+ /** The key */
+ protected Object key;
+ /** The value */
+ protected Object value;
+
+ /**
+ * Constructs a new pair with the specified key and given value.
+ *
+ * @param key the key for the entry, may be null
+ * @param value the value for the entry, may be null
+ */
+ protected AbstractKeyValue(Object key, Object value) {
+ super();
+ this.key = key;
+ this.value = value;
+ }
+
+ /**
+ * Gets the key from the pair.
+ *
+ * @return the key
+ */
+ public Object getKey() {
+ return key;
+ }
+
+ /**
+ * Gets the value from the pair.
+ *
+ * @return the value
+ */
+ public Object getValue() {
+ return value;
+ }
+
+ /**
+ * Gets a debugging String view of the pair.
+ *
+ * @return a String view of the entry
+ */
+ public String toString() {
+ return new StringBuffer()
+ .append(getKey())
+ .append('=')
+ .append(getValue())
+ .toString();
+ }
+
+}
diff --git a/trunk/src/test/org/apache/commons/collections/keyvalue/AbstractMapEntry.java b/trunk/src/test/org/apache/commons/collections/keyvalue/AbstractMapEntry.java
new file mode 100644
index 0000000..e88bc53
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/collections/keyvalue/AbstractMapEntry.java
@@ -0,0 +1,92 @@
+/*
+ * Copyright 2003-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.collections.keyvalue;
+
+import java.util.Map;
+
+/**
+ * Abstract Pair class to assist with creating correct Map Entry implementations.
+ *
+ * @since Commons Collections 3.0
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 19:53:36 $
+ *
+ * @author James Strachan
+ * @author Michael A. Smith
+ * @author Neil O'Toole
+ * @author Stephen Colebourne
+ */
+public abstract class AbstractMapEntry extends AbstractKeyValue implements Map.Entry {
+
+ /**
+ * Constructs a new entry with the given key and given value.
+ *
+ * @param key the key for the entry, may be null
+ * @param value the value for the entry, may be null
+ */
+ protected AbstractMapEntry(Object key, Object value) {
+ super(key, value);
+ }
+
+ // Map.Entry interface
+ //-------------------------------------------------------------------------
+ /**
+ * Sets the value stored in this Map Entry.
+ * <p>
+ * This Map Entry is not connected to a Map, so only the local data is changed.
+ *
+ * @param value the new value
+ * @return the previous value
+ */
+ public Object setValue(Object value) {
+ Object answer = this.value;
+ this.value = value;
+ return answer;
+ }
+
+ /**
+ * Compares this Map Entry with another Map Entry.
+ * <p>
+ * Implemented per API documentation of {@link java.util.Map.Entry#equals(Object)}
+ *
+ * @param obj the object to compare to
+ * @return true if equal key and value
+ */
+ public boolean equals(Object obj) {
+ if (obj == this) {
+ return true;
+ }
+ if (obj instanceof Map.Entry == false) {
+ return false;
+ }
+ Map.Entry other = (Map.Entry) obj;
+ return
+ (getKey() == null ? other.getKey() == null : getKey().equals(other.getKey())) &&
+ (getValue() == null ? other.getValue() == null : getValue().equals(other.getValue()));
+ }
+
+ /**
+ * Gets a hashCode compatible with the equals method.
+ * <p>
+ * Implemented per API documentation of {@link java.util.Map.Entry#hashCode()}
+ *
+ * @return a suitable hash code
+ */
+ public int hashCode() {
+ return (getKey() == null ? 0 : getKey().hashCode()) ^
+ (getValue() == null ? 0 : getValue().hashCode());
+ }
+
+}
diff --git a/trunk/src/test/org/apache/commons/collections/keyvalue/DefaultMapEntry.java b/trunk/src/test/org/apache/commons/collections/keyvalue/DefaultMapEntry.java
new file mode 100644
index 0000000..4c5342e
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/collections/keyvalue/DefaultMapEntry.java
@@ -0,0 +1,66 @@
+/*
+ * Copyright 2001-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.collections.keyvalue;
+
+import java.util.Map;
+
+import org.apache.commons.collections.KeyValue;
+
+/**
+ * A restricted implementation of {@link java.util.Map.Entry} that prevents
+ * the MapEntry contract from being broken.
+ *
+ * @since Commons Collections 3.0
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 19:53:16 $
+ *
+ * @author James Strachan
+ * @author Michael A. Smith
+ * @author Neil O'Toole
+ * @author Stephen Colebourne
+ */
+public final class DefaultMapEntry extends AbstractMapEntry {
+
+ /**
+ * Constructs a new entry with the specified key and given value.
+ *
+ * @param key the key for the entry, may be null
+ * @param value the value for the entry, may be null
+ */
+ public DefaultMapEntry(final Object key, final Object value) {
+ super(key, value);
+ }
+
+ /**
+ * Constructs a new entry from the specified KeyValue.
+ *
+ * @param pair the pair to copy, must not be null
+ * @throws NullPointerException if the entry is null
+ */
+ public DefaultMapEntry(final KeyValue pair) {
+ super(pair.getKey(), pair.getValue());
+ }
+
+ /**
+ * Constructs a new entry from the specified MapEntry.
+ *
+ * @param entry the entry to copy, must not be null
+ * @throws NullPointerException if the entry is null
+ */
+ public DefaultMapEntry(final Map.Entry entry) {
+ super(entry.getKey(), entry.getValue());
+ }
+
+}
diff --git a/trunk/src/test/org/apache/commons/collections/keyvalue/KeyValue.java b/trunk/src/test/org/apache/commons/collections/keyvalue/KeyValue.java
new file mode 100644
index 0000000..e872d77
--- /dev/null
+++ b/trunk/src/test/org/apache/commons/collections/keyvalue/KeyValue.java
@@ -0,0 +1,46 @@
+/*
+ * Copyright 2003-2004 The Apache Software Foundation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.collections;
+
+/**
+ * Defines a simple key value pair.
+ * <p>
+ * A Map Entry has considerable additional semantics over and above a simple
+ * key-value pair. This interface defines the minimum key value, with just the
+ * two get methods.
+ *
+ * @since Commons Collections 3.0
+ * @version $Revision: 1.1 $ $Date: 2004/05/10 19:52:59 $
+ *
+ * @author Stephen Colebourne
+ */
+public interface KeyValue {
+
+ /**
+ * Gets the key from the pair.
+ *
+ * @return the key
+ */
+ Object getKey();
+
+ /**
+ * Gets the value from the pair.
+ *
+ * @return the value
+ */
+ Object getValue();
+
+}
diff --git a/trunk/xdocs/.cvsignore b/trunk/xdocs/.cvsignore
new file mode 100644
index 0000000..cb6131b
--- /dev/null
+++ b/trunk/xdocs/.cvsignore
@@ -0,0 +1 @@
+stylesheets
diff --git a/trunk/xdocs/images/logo.png b/trunk/xdocs/images/logo.png
new file mode 100644
index 0000000..4f600cf
Binary files /dev/null and b/trunk/xdocs/images/logo.png differ
diff --git a/trunk/xdocs/index.xml b/trunk/xdocs/index.xml
new file mode 100644
index 0000000..44900a9
--- /dev/null
+++ b/trunk/xdocs/index.xml
@@ -0,0 +1,153 @@
+<?xml version="1.0"?>
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+
+<document>
+
+ <properties>
+ <title>Commons</title>
+ <author email="commons-dev at jakarta.apache.org">Commons Documentation Team</author>
+ </properties>
+
+ <body>
+
+<section name="Commons BeanUtils">
+
+<p>
+Most Java developers are used to creating Java classes that conform to the
+JavaBeans naming patterns for property getters and setters. It is natural to
+then access these methods directly, using calls to the corresponding
+<code>getXxx</code> and <code>setXxx</code> methods. However, there are some
+occasions where dynamic access to Java object properties (without compiled-in
+knowledge of the property getter and setter methods to be called) is needed.
+Example use cases include:</p>
+<ul>
+<li>Building scripting languages that interact with the Java object model
+ (such as the Bean Scripting Framework).</li>
+<li>Building template language processors for web presentation and similar
+ uses (such as JSP or Velocity).</li>
+<li>Building custom tag libraries for JSP and XSP environments (such as Jakarta
+ Taglibs, Struts, Cocoon).</li>
+<li>Consuming XML-based configuration resources (such as Ant build scripts, web
+ application deployment descriptors, Tomcat's <code>server.xml</code>
+ file).</li>
+</ul>
+
+<p>
+The Java language provides <em>Reflection</em> and <em>Introspection</em>
+APIs (see the <code>java.lang.reflect</code> and <code>java.beans</code>
+packages in the JDK Javadocs). However, these APIs can be quite complex to
+understand and utilize. The <em>BeanUtils</em> component provides
+easy-to-use wrappers around these capabilities.
+</p>
+ <subsection name='BeanUtils Core And Modules'>
+ <p>
+The next release will see a departure from the usual packaging strategy.
+Rather than just a single jar, BeanUtils will distribute a
+<code>commons-beanutils-core.jar</code> containing the essential parts of BeanUtils
+with minimal dependencies (for this release just
+<a href='http://jakarta.apache.org/commons/logging'>Commons Logging</a>).
+Other parts of BeanUtils (typically specialized modules) will be distributed as separate,
+modular jars with additional dependencies. Hopefully this will allow library and framework
+creators more finely grained control over their dependencies. It should also help with
+compatibility issues.
+ </p>
+ <p>
+For everyone else, don't worry :) There will be a single commons-beanutils-all.jar
+containing everything. Drop it in and add any dependencies your application needs.
+ </p>
+ </subsection>
+ <subsection name='Bean Collections'>
+ <p>
+Bean collections is a library combining BeanUtils with
+<a href='http://jakarta.apache.org/commons/collections'>Commons Collections</a>
+to provide services for collections of beans. Once class (<code>BeanComparator</code>)
+was previously released, the rest are new. This new distribution strategy should allow
+this sub-component to evolve naturally without the concerns about size and scope
+that might otherwise happen.
+ </p>
+ <p>
+Bean Collections has an additional dependency on
+<a href='http://jakarta.apache.org/commons/collections'>Commons Collections</a>.
+ </p>
+ </subsection>
+</section>
+
+
+<section name="Documentation">
+
+<p>The <a href="http://jakarta.apache.org/commons/beanutils/RELEASE-NOTES.txt">
+Release Notes</a> document the new features and bug fixes that have been
+included in this release.</p>
+
+<p>The <a href="http://jakarta.apache.org/commons/beanutils/api/index.html">
+JavaDoc API documents</a> are available online. In particular, you should
+note the property reference syntax options described in the
+<code>PropertyUtils</code> class description.</p>
+
+</section>
+
+
+<section name="Releases">
+ <subsection name='Mirrored Releases'>
+<p>
+Releases after 1.5 should be downloaded from a mirror. Please remember to verify the
+sigature of the release from the
+<a href='http://www.apache.org/dist/jakarta/commons/beanutils/'>main apache site</a>
+when downloading from a mirror.
+</p>
+<p>
+ Binary releases are available
+ <a href="http://jakarta.apache.org/site/binindex.cgi">here</a>.
+ Source releases are available
+ <a href="http://jakarta.apache.org/site/sourceindex.cgi">here</a>
+</p>
+</subsection>
+<subsection name='1.7.x Releases (Mirrored)'>
+ <p>
+<strong>BeanUtils 1.7.0</strong> is a service release which removes the dependency
+upon a specific commons-collection library version. It may be safely used together
+with either the 2.x or 3.x series of commons-collections releases.
+It also introduces a number of important enhancements. It is backward compatible
+with the 1.6 release.
+ </p>
+</subsection>
+<subsection name='1.6.x Releases (Mirrored)'>
+<ul>
+ <li>Version 1.6.1 (latest) - 18 Feb 2003</li>
+ <li>Version 1.6 - 21 Jan 2003</li>
+</ul>
+ </subsection>
+ <subsection name='Older Releases (Not Mirrored)'>
+ <p>
+ <ul>
+ <li><a href="http://jakarta.apache.org/builds/jakarta-commons/release/commons-beanutils/v1.5/">Version 1.5 </a> - 23 Oct 2002</li>
+ <li><a href="http://jakarta.apache.org/builds/jakarta-commons/release/commons-beanutils/v1.4.1/">Version 1.4.1</a> - 28 Aug 2002</li>
+ <li><a href="http://jakarta.apache.org/builds/jakarta-commons/release/commons-beanutils/v1.4/">Version 1.4</a> - 13 Aug 2002
+</li>
+ <li><a href="http://jakarta.apache.org/builds/jakarta-commons/release/commons-beanutils/v1.3/">Version 1.3</a> - 29 Apr 2002</li>
+ <li><a href="http://jakarta.apache.org/builds/jakarta-commons/release/commons-beanutils/v1.2/">Version 1.2</a> - 24 Dec 2001</li>
+ <li><a href="http://jakarta.apache.org/builds/jakarta-commons/release/commons-beanutils/v1.1/">Version 1.1</a> - 22 Sep 2001</li>
+ <li><a href="http://jakarta.apache.org/builds/jakarta-commons/release/commons-beanutils/v1.0/">Version 1.0</a> - 14 July 2001</li>
+ </ul>
+ </p>
+ </subsection>
+</section>
+
+
+
+</body>
+</document>
diff --git a/trunk/xdocs/navigation.xml b/trunk/xdocs/navigation.xml
new file mode 100644
index 0000000..6b8bb7b
--- /dev/null
+++ b/trunk/xdocs/navigation.xml
@@ -0,0 +1,48 @@
+<?xml version="1.0" encoding="ISO-8859-1"?>
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<!DOCTYPE org.apache.commons.menus SYSTEM '../../commons-build/menus/menus.dtd'>
+<project name="BeanUtils">
+
+ <title>BeanUtils</title>
+ <organizationLogo href="/images/jakarta-logo-blue.gif">
+ Jakarta
+ </organizationLogo>
+
+ <body>
+ <links>
+ <item name="Jakarta Commons"
+ href="http://jakarta.apache.org/commons/"/>
+ </links>
+
+ <menu name="Commons BeanUtils">
+ <item name="Overview" href="/index.html"/>
+ <item name="Javadoc (Release)" href="api/index.html"/>
+ <item name="Mailing lists" href="/mail-lists.html"/>
+ <item name="Team" href="/team-list.html"/>
+ <!--item name="Tasks" href="/tasks.html"/-->
+ <item name="CVS" href="http://cvs.apache.org/viewcvs/jakarta-commons/beanutils/"/>
+ <item name="Javadoc (CVS latest)" href="apidocs/index.html"/>
+ </menu>
+
+ <menu name='Modules'>
+ <item name="Bean Collections" href="bean-collections/index.html"/>
+ </menu>
+
+ &common-menus;
+
+ </body>
+</project>
diff --git a/trunk/xdocs/proposal.xml b/trunk/xdocs/proposal.xml
new file mode 100644
index 0000000..20e4929
--- /dev/null
+++ b/trunk/xdocs/proposal.xml
@@ -0,0 +1,136 @@
+<!--
+ Copyright 2001-2004 The Apache Software Foundation
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+-->
+<document>
+<properties>
+<title>Proposal for BeanUtils Package</title>
+</properties>
+<body>
+
+
+<section name="Proposal for BeanUtils Package">
+
+
+
+<subsection name="(0) Rationale">
+
+<p>Most Java developers are used to creating Java classes that conform to the
+JavaBeans naming patterns for property getters and setters. It is natural to
+then access these methods directly, using calls to the corresponding
+<code>getXxx</code> and <code>setXxx</code> methods. However, there are some
+occasions where dynamic access to Java object properties (without compiled-in
+knowledge of the property getter and setter methods to be called) is needed.
+Example use cases include:</p>
+<ul>
+<li>Building scripting languages that interact with the Java object model
+ (such as the Bean Scripting Framework).</li>
+<li>Building template language processors for web presentation and similar
+ uses (such as JSP or Velocity).</li>
+<li>Building custom tag libraries for JSP and XSP environments (such as Jakarta
+ Taglibs, Struts, Cocoon).</li>
+<li>Consuming XML-based configuration resources (such as Ant build scripts, web
+ application deployment descriptors, Tomcat's <code>server.xml</code>
+ file).</li>
+</ul>
+
+<p>The Java language provides <em>Reflection</em> and <em>Introspection</em>
+APIs (see the <code>java.lang.reflect</code> and <code>java.beans</code>
+packages in the JDK Javadocs). However, these APIs can be quite complex to
+understand and utilize. The proposed <em>BeanUtils</em> component provides
+easy-to-use wrappers around these capabilities.</p>
+
+
+</subsection>
+<subsection name="(1) Scope of the Package">
+
+<p>This proposal is to create a package of Java utility methods for accessing
+and modifying the properties of arbitrary JavaBeans. No dependencies outside
+of the JDK are required, so the use of this package is very lightweight.</p>
+
+<p>In addition to wrapping the reflection and introspection APIs of the
+standard JDK, <em>BeanUtils</em> components shall support a syntax for directly
+accessing <strong>nested</strong> and <strong>indexed</strong> properties, in a
+manner that will be familar to users of scripting languages like JavaScript.
+For example, the following property accessor expressions are supported:</p>
+<ul>
+<li><strong>customer</strong> - Equivalent to <code>getCustomer()</code>.</li>
+<li><strong>customer.address</strong> - Equivalent to
+ <code>getCustomer().getAddress()</code>.</li>
+<li><strong>customer.address[2].street</strong> - Equivalent to
+ <code>getCustomer().getAddress(2).getStreet()</code> (access to indexed
+ properties also works if the underlying property is an array rather than
+ providing indexed getter and setter methods).</li>
+</ul>
+
+
+</subsection>
+<subsection name="(1.5) Interaction With Other Packages">
+
+<p><em>BeanUtils</em> relies only on standard JDK 1.2 (or later) APIs for
+production deployment. It utilizes the JUnit unit testing framework for
+developing and executing unit tests, but this is of interest only to
+developers of the component. BeanUtils will also be a dependency for
+several future proposed components for the Jakarta Commons subproject.</p>
+
+<p>No external configuration files are utilized.</p>
+
+
+</subsection>
+<subsection name="(2) Initial Source of the Package">
+
+<p>The three original Java classes (<code>BeanUtils</code>,
+<code>ConvertUtils</code>, and <code>PropertyUtils</code>) are an integral
+part of the <a href="http://jakarta.apache.org/struts">Struts Framework</a>.
+However, they have very few dependencies on other aspects of Struts, and
+those dependencies have been removed in the proposed code base.
+Once accepted and released as a Jakarta Commons component, Struts will
+be modified to use the Commons version of these classes, and its internal
+versions will be deprecated.</p>
+
+<p>The proposed package name for the new component is
+<code>org.apache.commons.beanutils</code>.</p>
+
+
+</subsection>
+<subsection name="(3) Required Jakarta-Commons Resources">
+
+<ul>
+<li>CVS Repository - New directory <code>beanutils</code> in the
+ <code>jakarta-commons</code> CVS repository. All initial committers
+ are already committers on <code>jakarta-commons</code>, so no
+ additional user setups are required.</li>
+<li>Mailing List - Discussions will take place on the general
+ <em>jakarta-commons at jakarta.apache.org</em> mailing list. To help
+ list subscribers identify messages of interest, it is suggested that
+ the message subject of messages about this component be prefixed with
+ [BeanUtils].</li>
+<li>Bugzilla - New component "BeanUtils" under the "Commons" product
+ category, with appropriate version identifiers as needed.</li>
+<li>Jyve FAQ - New category "commons-beanutils" (when available). </li>
+</ul>
+
+
+</subsection>
+<subsection name="(4) Initial Committers">
+
+<p>The initial committers on the BeanUtils component shall be Craig
+McClanahan and Geir Magnusson Jr.</p>
+
+
+
+</subsection>
+</section>
+</body>
+</document>
diff --git a/trunk/xdocs/style/project.css b/trunk/xdocs/style/project.css
new file mode 100644
index 0000000..031480c
--- /dev/null
+++ b/trunk/xdocs/style/project.css
@@ -0,0 +1,5 @@
+#banner, #banner td {
+ background: #fff;
+ color: #000;
+}
+
--
Alioth's /usr/local/bin/git-commit-notice on /srv/git.debian.org/git/pkg-java/commons-beanutils.git
More information about the pkg-java-commits
mailing list