| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You 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 |
| * |
| * https://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.configuration2.beanutils; |
| |
| import static org.junit.jupiter.api.Assertions.assertEquals; |
| import static org.junit.jupiter.api.Assertions.assertFalse; |
| import static org.junit.jupiter.api.Assertions.assertInstanceOf; |
| import static org.junit.jupiter.api.Assertions.assertNotNull; |
| import static org.junit.jupiter.api.Assertions.assertNull; |
| import static org.junit.jupiter.api.Assertions.assertThrows; |
| import static org.junit.jupiter.api.Assertions.assertTrue; |
| |
| import java.util.Collection; |
| import java.util.HashMap; |
| import java.util.Iterator; |
| import java.util.Map; |
| |
| import org.apache.commons.configuration2.BaseHierarchicalConfiguration; |
| import org.apache.commons.configuration2.HierarchicalConfiguration; |
| import org.apache.commons.configuration2.ex.ConfigurationRuntimeException; |
| import org.junit.jupiter.api.Test; |
| |
| /** |
| * Test class for XMLBeanDeclaration. |
| */ |
| public class TestXMLBeanDeclaration { |
| |
| /** |
| * A helper class used for testing the createBeanDeclaration() factory method. |
| */ |
| private static final class XMLBeanDeclarationTestImpl extends XMLBeanDeclaration { |
| public XMLBeanDeclarationTestImpl(final HierarchicalConfiguration<?> config, final NodeData<?> node) { |
| super(config, node); |
| } |
| } |
| |
| /** An array with some test properties. */ |
| private static final String[] TEST_PROPS = {"firstName", "lastName", "department", "age", "hobby"}; |
| |
| /** An array with the values for the test properties. */ |
| private static final String[] TEST_VALUES = {"John", "Smith", "Engineering", "42", "TV"}; |
| |
| /** An array with the names of nested (complex) properties. */ |
| private static final String[] COMPLEX_PROPS = {"address", "car"}; |
| |
| /** An array with the names of the classes of the complex properties. */ |
| private static final String[] COMPLEX_CLASSES = {"org.apache.commons.configuration.test.AddressTest", "org.apache.commons.configuration.test.CarTest"}; |
| |
| /** An array with the property names of the complex properties. */ |
| private static final String[][] COMPLEX_ATTRIBUTES = {{"street", "zip", "city", "country"}, {"brand", "color"}}; |
| |
| /** An array with the values of the complex properties. */ |
| private static final String[][] COMPLEX_VALUES = {{"Baker Street", "12354", "London", "UK"}, {"Bentley", "silver"}}; |
| |
| /** An array with property names for a complex constructor argument. */ |
| private static final String[] CTOR_COMPLEX_ATTRIBUTES = {"secCode", "validTo"}; |
| |
| /** An array with values of a complex constructor argument. */ |
| private static final String[] CTOR_COMPLEX_VALUES = {"20121110181559", "2015-01-31"}; |
| |
| /** Constant for an ID value passed as constructor argument. */ |
| private static final String CTOR_ID = "20121110182006"; |
| |
| /** Constant for the key with the bean declaration. */ |
| private static final String KEY = "myBean"; |
| |
| /** Constant for the section with the variables. */ |
| private static final String VARS = "variables."; |
| |
| /** |
| * Checks the properties returned by a bean declaration. |
| * |
| * @param beanDecl the bean declaration |
| * @param names an array with the expected property names |
| * @param values an array with the expected property values |
| */ |
| private static void checkProperties(final BeanDeclaration beanDecl, final String[] names, final String[] values) { |
| final Map<String, Object> props = beanDecl.getBeanProperties(); |
| |
| final Map<String, String> expected = new HashMap<>(); |
| for (int i = 0; i < names.length; i++) { |
| expected.put(names[i], values[i]); |
| } |
| assertEquals(expected, props); |
| } |
| |
| /** |
| * Creates a configuration with data for testing nested bean declarations including constructor arguments. |
| * |
| * @return the initialized test configuration |
| */ |
| private static BaseHierarchicalConfiguration prepareNestedBeanDeclarations() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| setupBeanDeclaration(config, KEY, TEST_PROPS, TEST_VALUES); |
| final String keyCtorArg = KEY + ".config-constrarg"; |
| setupBeanDeclaration(config, keyCtorArg, CTOR_COMPLEX_ATTRIBUTES, CTOR_COMPLEX_VALUES); |
| config.addProperty(keyCtorArg + "[@config-class]", "TestClass"); |
| config.addProperty(keyCtorArg + "(-1)[@config-value]", CTOR_ID); |
| config.addProperty(keyCtorArg + "[@config-type]", "long"); |
| for (int i = 0; i < COMPLEX_PROPS.length; i++) { |
| setupBeanDeclaration(config, KEY + '.' + COMPLEX_PROPS[i], COMPLEX_ATTRIBUTES[i], COMPLEX_VALUES[i]); |
| config.addProperty(KEY + '.' + COMPLEX_PROPS[i] + "[@config-class]", COMPLEX_CLASSES[i]); |
| } |
| return config; |
| } |
| |
| /** |
| * Initializes a configuration object with a bean declaration. Under the specified key the given properties will be |
| * added. |
| * |
| * @param config the configuration to initialize |
| * @param key the key of the bean declaration |
| * @param names an array with the names of the properties |
| * @param values an array with the corresponding values |
| */ |
| private static void setupBeanDeclaration(final HierarchicalConfiguration<?> config, final String key, final String[] names, final String[] values) { |
| for (int i = 0; i < names.length; i++) { |
| config.addProperty(key + "[@" + names[i] + "]", values[i]); |
| } |
| } |
| |
| /** |
| * Tests fetching the bean's class name. |
| */ |
| @Test |
| void testGetBeanClassName() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.addProperty(KEY + "[@config-class]", getClass().getName()); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| assertEquals(getClass().getName(), decl.getBeanClassName()); |
| } |
| |
| /** |
| * Tests whether a default bean class name is taken into account. |
| */ |
| @Test |
| void testGetBeanClassNameFromDefault() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.addProperty(KEY + "[@someProperty]", Boolean.TRUE); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY, false, getClass().getName()); |
| assertEquals(getClass().getName(), decl.getBeanClassName()); |
| } |
| |
| /** |
| * Tests fetching the bean's class name if it is undefined. |
| */ |
| @Test |
| void testGetBeanClassNameUndefined() { |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(new BaseHierarchicalConfiguration()); |
| assertNull(decl.getBeanClassName()); |
| } |
| |
| /** |
| * Tests that a missing bean class name does not cause an exception. |
| */ |
| @Test |
| void testGetBeanClassNameUndefinedWithEx() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.setThrowExceptionOnMissing(true); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config); |
| assertNull(decl.getBeanClassName()); |
| } |
| |
| /** |
| * Tests fetching the name of the bean factory. |
| */ |
| @Test |
| void testGetBeanFactoryName() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.addProperty(KEY + "[@config-factory]", "myFactory"); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| assertEquals("myFactory", decl.getBeanFactoryName()); |
| } |
| |
| /** |
| * Tests fetching the name of the bean factory if it is undefined. |
| */ |
| @Test |
| void testGetBeanFactoryNameUndefined() { |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(new BaseHierarchicalConfiguration()); |
| assertNull(decl.getBeanFactoryName()); |
| } |
| |
| /** |
| * Tests that a missing bean factory name does not throw an exception. |
| */ |
| @Test |
| void testGetBeanFactoryNameUndefinedWithEx() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.setThrowExceptionOnMissing(true); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config); |
| assertNull(decl.getBeanFactoryName()); |
| } |
| |
| /** |
| * Tests fetching the parameter for the bean factory. |
| */ |
| @Test |
| void testGetBeanFactoryParameter() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.addProperty(KEY + "[@config-factoryParam]", "myFactoryParameter"); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| assertEquals("myFactoryParameter", decl.getBeanFactoryParameter()); |
| } |
| |
| /** |
| * Tests fetching the parameter for the bean factory if it is undefined. |
| */ |
| @Test |
| void testGetBeanFactoryParameterUndefined() { |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(new BaseHierarchicalConfiguration()); |
| assertNull(decl.getBeanFactoryParameter()); |
| } |
| |
| /** |
| * Tests that an undefined bean factory parameter does not cause an exception. |
| */ |
| @Test |
| void testGetBeanFactoryParameterUndefinedWithEx() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.setThrowExceptionOnMissing(true); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config); |
| assertNull(decl.getBeanFactoryParameter()); |
| } |
| |
| /** |
| * Tests if the bean's properties are correctly extracted from the configuration object. |
| */ |
| @Test |
| void testGetBeanProperties() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| setupBeanDeclaration(config, KEY, TEST_PROPS, TEST_VALUES); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| checkProperties(decl, TEST_PROPS, TEST_VALUES); |
| } |
| |
| /** |
| * Tests fetching properties if none are defined. |
| */ |
| @Test |
| void testGetBeanPropertiesEmpty() { |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(new BaseHierarchicalConfiguration()); |
| final Map<String, Object> props = decl.getBeanProperties(); |
| assertTrue(props == null || props.isEmpty()); |
| } |
| |
| /** |
| * Tests obtaining the bean's properties when reserved attributes are involved. These should be ignored. |
| */ |
| @Test |
| void testGetBeanPropertiesWithReservedAttributes() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| setupBeanDeclaration(config, KEY, TEST_PROPS, TEST_VALUES); |
| config.addProperty(KEY + "[@config-testattr]", "yes"); |
| config.addProperty(KEY + "[@config-anothertest]", "this, too"); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| checkProperties(decl, TEST_PROPS, TEST_VALUES); |
| } |
| |
| /** |
| * Tests whether constructor arguments can be queried. |
| */ |
| @Test |
| void testGetConstructorArgs() { |
| final BaseHierarchicalConfiguration config = prepareNestedBeanDeclarations(); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| final Collection<ConstructorArg> args = decl.getConstructorArgs(); |
| assertEquals(2, args.size()); |
| final Iterator<ConstructorArg> it = args.iterator(); |
| final ConstructorArg arg1 = it.next(); |
| assertTrue(arg1.isNestedBeanDeclaration()); |
| checkProperties(arg1.getBeanDeclaration(), CTOR_COMPLEX_ATTRIBUTES, CTOR_COMPLEX_VALUES); |
| assertNull(arg1.getTypeName()); |
| assertEquals("TestClass", arg1.getBeanDeclaration().getBeanClassName()); |
| final ConstructorArg arg2 = it.next(); |
| assertFalse(arg2.isNestedBeanDeclaration()); |
| assertEquals(CTOR_ID, arg2.getValue()); |
| assertEquals("long", arg2.getTypeName()); |
| } |
| |
| /** |
| * Tests whether a constructor argument with a null value can be defined. |
| */ |
| @Test |
| void testGetConstructorArgsNullArg() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| setupBeanDeclaration(config, KEY, TEST_PROPS, TEST_VALUES); |
| config.addProperty(KEY + ".config-constrarg", ""); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| final Collection<ConstructorArg> args = decl.getConstructorArgs(); |
| assertEquals(1, args.size()); |
| final ConstructorArg arg = args.iterator().next(); |
| assertFalse(arg.isNestedBeanDeclaration()); |
| assertNull(arg.getValue()); |
| } |
| |
| /** |
| * Tests whether interpolation of bean properties works. |
| */ |
| @Test |
| void testGetInterpolatedBeanProperties() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| final String[] varValues = new String[TEST_PROPS.length]; |
| for (int i = 0; i < TEST_PROPS.length; i++) { |
| varValues[i] = "${" + VARS + TEST_PROPS[i] + "}"; |
| config.addProperty(VARS + TEST_PROPS[i], TEST_VALUES[i]); |
| } |
| setupBeanDeclaration(config, KEY, TEST_PROPS, varValues); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| checkProperties(decl, TEST_PROPS, TEST_VALUES); |
| } |
| |
| /** |
| * Tests whether interpolation is done on constructor arguments. |
| */ |
| @Test |
| void testGetInterpolatedConstructorArgs() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| final String expectedValue = "ctorArg"; |
| config.addProperty("value", expectedValue); |
| setupBeanDeclaration(config, KEY, TEST_PROPS, TEST_VALUES); |
| config.addProperty(KEY + ".config-constrarg[@config-value]", "${value}"); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| final Collection<ConstructorArg> args = decl.getConstructorArgs(); |
| final ConstructorArg arg = args.iterator().next(); |
| assertEquals(expectedValue, arg.getValue()); |
| } |
| |
| /** |
| * Tests fetching nested bean declarations. |
| */ |
| @Test |
| void testGetNestedBeanDeclarations() { |
| final BaseHierarchicalConfiguration config = prepareNestedBeanDeclarations(); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| checkProperties(decl, TEST_PROPS, TEST_VALUES); |
| |
| final Map<String, Object> nested = decl.getNestedBeanDeclarations(); |
| assertEquals(COMPLEX_PROPS.length, nested.size()); |
| for (int i = 0; i < COMPLEX_PROPS.length; i++) { |
| final XMLBeanDeclaration d = (XMLBeanDeclaration) nested.get(COMPLEX_PROPS[i]); |
| assertNotNull(d, "No declaration found for " + COMPLEX_PROPS[i]); |
| checkProperties(d, COMPLEX_ATTRIBUTES[i], COMPLEX_VALUES[i]); |
| assertEquals(COMPLEX_CLASSES[i], d.getBeanClassName()); |
| } |
| } |
| |
| /** |
| * Tests fetching nested bean declarations if none are defined. |
| */ |
| @Test |
| void testGetNestedBeanDeclarationsEmpty() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| setupBeanDeclaration(config, KEY, TEST_PROPS, TEST_VALUES); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| final Map<String, Object> nested = decl.getNestedBeanDeclarations(); |
| assertTrue(nested == null || nested.isEmpty()); |
| } |
| |
| /** |
| * Tests whether the factory method for creating nested bean declarations gets called. |
| */ |
| @Test |
| void testGetNestedBeanDeclarationsFactoryMethod() { |
| final BaseHierarchicalConfiguration config = prepareNestedBeanDeclarations(); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY) { |
| @Override |
| BeanDeclaration createBeanDeclaration(final NodeData<?> node) { |
| return new XMLBeanDeclarationTestImpl(getConfiguration().configurationAt(node.nodeName()), node); |
| } |
| }; |
| final Map<String, Object> nested = decl.getNestedBeanDeclarations(); |
| for (final String element : COMPLEX_PROPS) { |
| final Object d = nested.get(element); |
| assertInstanceOf(XMLBeanDeclarationTestImpl.class, d, "Wrong class for bean declaration: " + d); |
| } |
| } |
| |
| /** |
| * Tests whether reserved characters in the node names of nested bean declarations are handled correctly. This is |
| * related to CONFIGURATION-567. |
| */ |
| @Test |
| void testGetNestedBeanDeclarationsReservedCharacter() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| final String key = KEY + ".address..private"; |
| setupBeanDeclaration(config, key, COMPLEX_ATTRIBUTES[0], COMPLEX_VALUES[0]); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| |
| final Map<String, Object> nested = decl.getNestedBeanDeclarations(); |
| assertTrue(nested.containsKey("address.private")); |
| } |
| |
| /** |
| * Tests constructing a bean declaration from a key with multiple values. This should cause an exception because keys |
| * must be unique. |
| */ |
| @Test |
| void testInitFromMultiValueKey() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.addProperty(KEY, "myFirstKey"); |
| config.addProperty(KEY, "mySecondKey"); |
| assertThrows(ConfigurationRuntimeException.class, () -> new XMLBeanDeclaration(config, KEY)); |
| } |
| |
| /** |
| * Tests creating a declaration from a null configuration. This should cause an exception. |
| */ |
| @Test |
| void testInitFromNullConfiguration() { |
| assertThrows(IllegalArgumentException.class, () -> new XMLBeanDeclaration(null)); |
| } |
| |
| /** |
| * Tests creating a declaration from a null configuration with a key. This should cause an exception. |
| */ |
| @Test |
| void testInitFromNullConfigurationAndKey() { |
| assertThrows(IllegalArgumentException.class, () -> new XMLBeanDeclaration(null, KEY)); |
| } |
| |
| /** |
| * Tests constructing a bean declaration from an undefined key. This should cause an exception. |
| */ |
| @Test |
| void testInitFromUndefinedKey() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| setupBeanDeclaration(config, KEY, TEST_PROPS, TEST_VALUES); |
| assertThrows(ConfigurationRuntimeException.class, () -> new XMLBeanDeclaration(config, "undefined_key")); |
| } |
| |
| /** |
| * Tests constructing a bean declaration from a key, which is undefined when the optional flag is set. In this case an |
| * empty declaration should be created, which can be used for creating beans as long as a default class is provided. |
| */ |
| @Test |
| void testInitFromUndefinedKeyOptional() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| setupBeanDeclaration(config, KEY, TEST_PROPS, TEST_VALUES); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, "undefined_key", true); |
| assertNull(decl.getBeanClassName()); |
| } |
| |
| /** |
| * Tests interpolate() if no ConfigurationInterpolator is available. |
| */ |
| @Test |
| void testInterpolateNoInterpolator() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.addProperty("value", "expectedValue"); |
| setupBeanDeclaration(config, KEY, TEST_PROPS, TEST_VALUES); |
| final String value = "${value}"; |
| config.addProperty(KEY + ".config-constrarg[@config-value]", value); |
| config.setInterpolator(null); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY); |
| final Collection<ConstructorArg> args = decl.getConstructorArgs(); |
| final ConstructorArg arg = args.iterator().next(); |
| assertEquals(value, arg.getValue()); |
| } |
| |
| /** |
| * Tests whether a default bean class name is overridden by a value in the configuration. |
| */ |
| @Test |
| public void tetGetBeanClassNameDefaultOverride() { |
| final BaseHierarchicalConfiguration config = new BaseHierarchicalConfiguration(); |
| config.addProperty(KEY + "[@config-class]", getClass().getName()); |
| final XMLBeanDeclaration decl = new XMLBeanDeclaration(config, KEY, false, "someDefaultClassName"); |
| assertEquals(getClass().getName(), decl.getBeanClassName()); |
| } |
| } |