blob: bcf3831a9eaf8d317aa75cb117ad5e835b8aea6e [file] [log] [blame]
/*
* 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());
}
}