blob: bdfecb26cde5b77ec8a13085dff09680805c6992 [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
*
* 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.configuration2;
import static org.hamcrest.CoreMatchers.containsString;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertNull;
import static org.junit.Assert.assertThat;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.TransformerFactoryConfigurationError;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.StringReader;
import java.io.StringWriter;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import org.apache.commons.configuration2.SynchronizerTestImpl.Methods;
import org.apache.commons.configuration2.builder.FileBasedBuilderParametersImpl;
import org.apache.commons.configuration2.builder.FileBasedConfigurationBuilder;
import org.apache.commons.configuration2.convert.DefaultListDelimiterHandler;
import org.apache.commons.configuration2.convert.DisabledListDelimiterHandler;
import org.apache.commons.configuration2.ex.ConfigurationException;
import org.apache.commons.configuration2.io.FileHandler;
import org.apache.commons.configuration2.resolver.CatalogResolver;
import org.apache.commons.configuration2.tree.ImmutableNode;
import org.apache.commons.configuration2.tree.NodeStructureHelper;
import org.apache.commons.configuration2.tree.xpath.XPathExpressionEngine;
import org.junit.Before;
import org.junit.Test;
import org.xml.sax.SAXException;
import org.xml.sax.SAXParseException;
import org.xml.sax.helpers.DefaultHandler;
/**
* test for loading and saving xml properties files
*
* @version $Id$
*/
public class TestXMLConfiguration
{
/** XML Catalog */
private static final String CATALOG_FILES = ConfigurationAssert
.getTestFile("catalog.xml").getAbsolutePath();
/** Constant for the used encoding.*/
static final String ENCODING = "ISO-8859-1";
/** Constant for the test system ID.*/
static final String SYSTEM_ID = "properties.dtd";
/** Constant for the test public ID.*/
static final String PUBLIC_ID = "-//Commons Configuration//DTD Test Configuration 1.3//EN";
/** Constant for the DOCTYPE declaration.*/
static final String DOCTYPE_DECL = " PUBLIC \"" + PUBLIC_ID + "\" \"" + SYSTEM_ID + "\">";
/** Constant for the DOCTYPE prefix.*/
static final String DOCTYPE = "<!DOCTYPE ";
/** Constant for the transformer factory property.*/
static final String PROP_FACTORY = "javax.xml.transform.TransformerFactory";
/** The File that we test with */
private final String testProperties = ConfigurationAssert.getTestFile("test.xml").getAbsolutePath();
private final String testProperties2 = ConfigurationAssert.getTestFile("testDigesterConfigurationInclude1.xml").getAbsolutePath();
private final File testSaveConf = ConfigurationAssert.getOutFile("testsave.xml");
private final File testSaveFile = ConfigurationAssert.getOutFile("testsample2.xml");
private final String testFile2 = ConfigurationAssert.getTestFile("sample.xml").getAbsolutePath();
/** Constant for the number of test threads. */
private static final int THREAD_COUNT = 5;
/** Constant for the number of loops in tests with multiple threads. */
private static final int LOOP_COUNT = 100;
private XMLConfiguration conf;
@Before
public void setUp() throws Exception
{
conf = createFromFile(testProperties);
removeTestFile();
}
/**
* Helper method for loading the specified configuration file.
*
* @param config the configuration
* @param fileName the name of the file to be loaded
* @throws ConfigurationException if an error occurs
*/
private static void load(XMLConfiguration config, String fileName)
throws ConfigurationException
{
FileHandler handler = new FileHandler(config);
handler.setFileName(fileName);
handler.load();
}
/**
* Creates a new XMLConfiguration and loads the specified file.
*
* @param fileName the name of the file to be loaded
* @return the newly created configuration instance
* @throws ConfigurationException if an error occurs
*/
private static XMLConfiguration createFromFile(String fileName)
throws ConfigurationException
{
XMLConfiguration config = new XMLConfiguration();
config.setListDelimiterHandler(new DefaultListDelimiterHandler(','));
load(config, fileName);
return config;
}
@Test
public void testGetProperty()
{
assertEquals("value", conf.getProperty("element"));
}
@Test
public void testGetCommentedProperty()
{
assertEquals("", conf.getProperty("test.comment"));
}
@Test
public void testGetPropertyWithXMLEntity()
{
assertEquals("1<2", conf.getProperty("test.entity"));
}
@Test
public void testClearPropertyNotExisting()
{
String key = "clearly";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
}
@Test
public void testClearPropertySingleElement()
{
String key = "clear.element";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
}
@Test
public void testClearPropertySingleElementWithAttribute()
{
String key = "clear.element2";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
key = "clear.element2[@id]";
assertNotNull(key, conf.getProperty(key));
assertNotNull(key, conf.getProperty(key));
}
@Test
public void testClearPropertyNonText()
{
String key = "clear.comment";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
}
@Test
public void testClearPropertyCData()
{
String key = "clear.cdata";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
}
@Test
public void testClearPropertyMultipleSiblings()
{
String key = "clear.list.item";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
key = "clear.list.item[@id]";
assertNotNull(key, conf.getProperty(key));
assertNotNull(key, conf.getProperty(key));
}
@Test
public void testClearPropertyMultipleDisjoined() throws Exception
{
String key = "list.item";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
}
@Test
public void testgetProperty() {
// test non-leaf element
Object property = conf.getProperty("clear");
assertNull(property);
// test non-existent element
property = conf.getProperty("e");
assertNull(property);
// test non-existent element
property = conf.getProperty("element3[@n]");
assertNull(property);
// test single element
property = conf.getProperty("element");
assertNotNull(property);
assertTrue(property instanceof String);
assertEquals("value", property);
// test single attribute
property = conf.getProperty("element3[@name]");
assertNotNull(property);
assertTrue(property instanceof String);
assertEquals("foo", property);
// test non-text/cdata element
property = conf.getProperty("test.comment");
assertEquals("", property);
// test cdata element
property = conf.getProperty("test.cdata");
assertNotNull(property);
assertTrue(property instanceof String);
assertEquals("<cdata value>", property);
// test multiple sibling elements
property = conf.getProperty("list.sublist.item");
assertNotNull(property);
assertTrue(property instanceof List);
List<?> list = (List<?>) property;
assertEquals(2, list.size());
assertEquals("five", list.get(0));
assertEquals("six", list.get(1));
// test multiple, disjoined elements
property = conf.getProperty("list.item");
assertNotNull(property);
assertTrue(property instanceof List);
list = (List<?>) property;
assertEquals(4, list.size());
assertEquals("one", list.get(0));
assertEquals("two", list.get(1));
assertEquals("three", list.get(2));
assertEquals("four", list.get(3));
// test multiple, disjoined attributes
property = conf.getProperty("list.item[@name]");
assertNotNull(property);
assertTrue(property instanceof List);
list = (List<?>) property;
assertEquals(2, list.size());
assertEquals("one", list.get(0));
assertEquals("three", list.get(1));
}
@Test
public void testGetAttribute()
{
assertEquals("element3[@name]", "foo", conf.getProperty("element3[@name]"));
}
@Test
public void testClearAttributeNonExisting()
{
String key = "clear[@id]";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
}
@Test
public void testClearAttributeSingle()
{
String key = "clear.element2[@id]";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
key = "clear.element2";
assertNotNull(key, conf.getProperty(key));
assertNotNull(key, conf.getProperty(key));
}
@Test
public void testClearAttributeMultipleDisjoined() throws Exception
{
String key = "clear.list.item[@id]";
conf.clearProperty(key);
assertNull(key, conf.getProperty(key));
assertNull(key, conf.getProperty(key));
key = "clear.list.item";
assertNotNull(key, conf.getProperty(key));
assertNotNull(key, conf.getProperty(key));
}
@Test
public void testSetAttribute()
{
// replace an existing attribute
conf.setProperty("element3[@name]", "bar");
assertEquals("element3[@name]", "bar", conf.getProperty("element3[@name]"));
// set a new attribute
conf.setProperty("foo[@bar]", "value");
assertEquals("foo[@bar]", "value", conf.getProperty("foo[@bar]"));
conf.setProperty("name1", "value1");
assertEquals("value1", conf.getProperty("name1"));
}
/**
* Tests whether an attribute value can be overridden.
*/
@Test
public void testOverrideAttribute()
{
conf.addProperty("element3[@name]", "bar");
List<Object> list = conf.getList("element3[@name]");
assertNotNull("null list", list);
assertTrue("'bar' element missing", list.contains("bar"));
assertEquals("list size", 1, list.size());
}
@Test
public void testAddObjectAttribute()
{
conf.addProperty("test.boolean[@value]", Boolean.TRUE);
assertTrue("test.boolean[@value]", conf.getBoolean("test.boolean[@value]"));
}
/**
* Tests setting an attribute on the root element.
*/
@Test
public void testSetRootAttribute() throws ConfigurationException
{
conf.setProperty("[@test]", "true");
assertEquals("Root attribute not set", "true", conf
.getString("[@test]"));
saveTestConfig();
XMLConfiguration checkConf = checkSavedConfig();
assertTrue("Attribute not found after save", checkConf
.containsKey("[@test]"));
checkConf.setProperty("[@test]", "newValue");
conf = checkConf;
saveTestConfig();
checkConf = checkSavedConfig();
assertEquals("Attribute not modified after save", "newValue", checkConf
.getString("[@test]"));
}
@Test
public void testAddList()
{
conf.addProperty("test.array", "value1");
conf.addProperty("test.array", "value2");
List<Object> list = conf.getList("test.array");
assertNotNull("null list", list);
assertTrue("'value1' element missing", list.contains("value1"));
assertTrue("'value2' element missing", list.contains("value2"));
assertEquals("list size", 2, list.size());
}
@Test
public void testGetComplexProperty()
{
assertEquals("I'm complex!", conf.getProperty("element2.subelement.subsubelement"));
}
/**
* Tests constructing an XMLConfiguration from a non existing file and later
* saving to this file.
*/
@Test
public void testLoadAndSaveFromFile() throws Exception
{
// If the file does not exist, an empty config is created
assertFalse("File exists", testSaveConf.exists());
FileBasedConfigurationBuilder<XMLConfiguration> builder =
new FileBasedConfigurationBuilder<XMLConfiguration>(
XMLConfiguration.class, null, true);
builder.configure(new FileBasedBuilderParametersImpl()
.setFile(testSaveConf));
conf = builder.getConfiguration();
assertTrue(conf.isEmpty());
conf.addProperty("test", "yes");
builder.save();
XMLConfiguration checkConfig =
createFromFile(testSaveConf.getAbsolutePath());
assertEquals("yes", checkConfig.getString("test"));
}
/**
* Tests loading from a stream.
*/
@Test
public void testLoadFromStream() throws Exception
{
String xml = "<?xml version=\"1.0\"?><config><test>1</test></config>";
conf = new XMLConfiguration();
FileHandler handler = new FileHandler(conf);
handler.load(new ByteArrayInputStream(xml.getBytes()));
assertEquals(1, conf.getInt("test"));
conf = new XMLConfiguration();
handler = new FileHandler(conf);
handler.load(new ByteArrayInputStream(xml.getBytes()), "UTF8");
assertEquals(1, conf.getInt("test"));
}
/**
* Tests loading a non well formed XML from a string.
*/
@Test(expected = ConfigurationException.class)
public void testLoadInvalidXML() throws Exception
{
String xml = "<?xml version=\"1.0\"?><config><test>1</rest></config>";
conf = new XMLConfiguration();
FileHandler handler = new FileHandler(conf);
handler.load(new StringReader(xml));
}
@Test
public void testSetProperty() throws Exception
{
conf.setProperty("element.string", "hello");
assertEquals("'element.string'", "hello", conf.getString("element.string"));
assertEquals("XML value of element.string", "hello", conf.getProperty("element.string"));
}
@Test
public void testAddProperty()
{
// add a property to a non initialized xml configuration
XMLConfiguration config = new XMLConfiguration();
config.addProperty("test.string", "hello");
assertEquals("'test.string'", "hello", config.getString("test.string"));
}
@Test
public void testAddObjectProperty()
{
// add a non string property
conf.addProperty("test.boolean", Boolean.TRUE);
assertTrue("'test.boolean'", conf.getBoolean("test.boolean"));
}
@Test
public void testSave() throws Exception
{
// add an array of strings to the configuration
conf.addProperty("string", "value1");
for (int i = 1; i < 5; i++)
{
conf.addProperty("test.array", "value" + i);
}
// add comma delimited lists with escaped delimiters
conf.addProperty("split.list5", "a\\,b\\,c");
conf.setProperty("element3", "value\\,value1\\,value2");
conf.setProperty("element3[@name]", "foo\\,bar");
// save the configuration
saveTestConfig();
// read the configuration and compare the properties
checkSavedConfig();
}
/**
* Tests saving to a URL.
*/
@Test
public void testSaveToURL() throws Exception
{
FileHandler handler = new FileHandler(conf);
handler.save(testSaveConf.toURI().toURL());
checkSavedConfig(testSaveConf);
}
/**
* Tests saving to a stream.
*/
@Test
public void testSaveToStream() throws ConfigurationException, IOException
{
FileOutputStream out = null;
FileHandler handler = new FileHandler(conf);
try
{
out = new FileOutputStream(testSaveConf);
handler.save(out, "UTF8");
}
finally
{
if(out != null)
{
out.close();
}
}
checkSavedConfig(testSaveConf);
}
/**
* Tests whether a configuration can be saved to a stream with a specific encoding.
*/
@Test
public void testSaveToStreamWithEncoding() throws ConfigurationException, IOException
{
FileHandler handler = new FileHandler(conf);
handler.setEncoding("UTF8");
FileOutputStream out = null;
try
{
out = new FileOutputStream(testSaveConf);
handler.save(out);
}
finally
{
if(out != null)
{
out.close();
}
}
checkSavedConfig(testSaveConf);
}
/**
* Tests if a second file can be appended to a first.
*/
@Test
public void testAppend() throws Exception
{
load(conf, testProperties2);
assertEquals("value", conf.getString("element"));
assertEquals("tasks", conf.getString("table.name"));
saveTestConfig();
conf = createFromFile(testSaveConf.getAbsolutePath());
assertEquals("value", conf.getString("element"));
assertEquals("tasks", conf.getString("table.name"));
assertEquals("application", conf.getString("table[@tableType]"));
}
/**
* Tests saving attributes (related to issue 34442).
*/
@Test
public void testSaveAttributes() throws Exception
{
conf.clear();
load(conf, testProperties);
saveTestConfig();
conf = new XMLConfiguration();
load(conf, testSaveConf.getAbsolutePath());
assertEquals("foo", conf.getString("element3[@name]"));
}
/**
* Tests access to tag names with delimiter characters.
*/
@Test
public void testComplexNames()
{
assertEquals("Name with dot", conf.getString("complexNames.my..elem"));
assertEquals("Another dot", conf.getString("complexNames.my..elem.sub..elem"));
}
/**
* Creates a validating document builder.
* @return the document builder
* @throws ParserConfigurationException if an error occurs
*/
private DocumentBuilder createValidatingDocBuilder()
throws ParserConfigurationException
{
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
factory.setValidating(true);
DocumentBuilder builder = factory.newDocumentBuilder();
builder.setErrorHandler(new DefaultHandler() {
@Override
public void error(SAXParseException ex) throws SAXException
{
throw ex;
}
});
return builder;
}
/**
* Tests setting a custom document builder.
*/
@Test
public void testCustomDocBuilder() throws Exception
{
// Load an invalid XML file with the default (non validating)
// doc builder. This should work...
conf = new XMLConfiguration();
load(conf, ConfigurationAssert.getTestFile("testValidateInvalid.xml")
.getAbsolutePath());
assertEquals("customers", conf.getString("table.name"));
assertFalse(conf.containsKey("table.fields.field(1).type"));
}
/**
* Tests whether a validating document builder detects a validation error.
*/
@Test(expected = ConfigurationException.class)
public void testCustomDocBuilderValidationError() throws Exception
{
DocumentBuilder builder = createValidatingDocBuilder();
conf = new XMLConfiguration();
conf.setDocumentBuilder(builder);
load(conf, ConfigurationAssert.getTestFile("testValidateInvalid.xml")
.getAbsolutePath());
}
/**
* Tests whether a valid document can be loaded with a validating document builder.
*/
@Test
public void testCustomDocBuilderValidationSuccess() throws Exception
{
DocumentBuilder builder = createValidatingDocBuilder();
conf = new XMLConfiguration();
conf.setDocumentBuilder(builder);
load(conf, ConfigurationAssert.getTestFile("testValidateValid.xml")
.getAbsolutePath());
assertTrue(conf.containsKey("table.fields.field(1).type"));
}
/**
* Tests the clone() method.
*/
@Test
public void testClone()
{
Configuration c = (Configuration) conf.clone();
assertTrue(c instanceof XMLConfiguration);
XMLConfiguration copy = (XMLConfiguration) c;
assertNotNull(conf.getDocument());
assertNull(copy.getDocument());
copy.setProperty("element3", "clonedValue");
assertEquals("value", conf.getString("element3"));
conf.setProperty("element3[@name]", "originalFoo");
assertEquals("foo", copy.getString("element3[@name]"));
}
/**
* Tests saving a configuration after cloning to ensure that the clone and
* the original are completely detached.
*/
@Test
public void testCloneWithSave() throws ConfigurationException
{
XMLConfiguration c = (XMLConfiguration) conf.clone();
c.addProperty("test.newProperty", Boolean.TRUE);
conf.addProperty("test.orgProperty", Boolean.TRUE);
new FileHandler(c).save(testSaveConf);
XMLConfiguration c2 = new XMLConfiguration();
load(c2, testSaveConf.getAbsolutePath());
assertTrue("New property after clone() was not saved", c2
.getBoolean("test.newProperty"));
assertFalse("Property of original config was saved", c2
.containsKey("test.orgProperty"));
}
/**
* Tests the subset() method. There was a bug that calling subset() had
* undesired side effects.
*/
@Test
public void testSubset() throws ConfigurationException
{
conf = new XMLConfiguration();
load(conf, "testHierarchicalXMLConfiguration.xml");
conf.subset("tables.table(0)");
saveTestConfig();
conf = new XMLConfiguration();
load(conf, "testHierarchicalXMLConfiguration.xml");
assertEquals("users", conf.getString("tables.table(0).name"));
}
/**
* Tests string properties with list delimiters and escaped delimiters.
*/
@Test
public void testSplitLists()
{
assertEquals("a,b,c", conf.getString("split.list3[@values]"));
assertEquals(0, conf.getMaxIndex("split.list3[@values]"));
assertEquals("a\\,b\\,c", conf.getString("split.list4[@values]"));
assertEquals("a", conf.getString("split.list1"));
assertEquals(2, conf.getMaxIndex("split.list1"));
assertEquals("a,b,c", conf.getString("split.list2"));
}
/**
* Tests string properties with list delimiters when delimiter parsing
* is disabled
*/
@Test
public void testDelimiterParsingDisabled() throws ConfigurationException {
XMLConfiguration conf2 = new XMLConfiguration();
load(conf2, testProperties);
assertEquals("a,b,c", conf2.getString("split.list3[@values]"));
assertEquals(0, conf2.getMaxIndex("split.list3[@values]"));
assertEquals("a\\,b\\,c", conf2.getString("split.list4[@values]"));
assertEquals("a,b,c", conf2.getString("split.list1"));
assertEquals(0, conf2.getMaxIndex("split.list1"));
assertEquals("a\\,b\\,c", conf2.getString("split.list2"));
}
/**
* Tests whether string properties with list delimiters can be accessed if
* delimiter parsing is disabled and the XPath expression engine is used.
*/
@Test
public void testDelimiterParsingDisabledXPath() throws ConfigurationException
{
XMLConfiguration conf2 = new XMLConfiguration();
conf2.setExpressionEngine(new XPathExpressionEngine());
load(conf2, testProperties);
assertEquals("a,b,c", conf2.getString("split/list3/@values"));
assertEquals(0, conf2.getMaxIndex("split/list3/@values"));
assertEquals("a\\,b\\,c", conf2.getString("split/list4/@values"));
assertEquals("a,b,c", conf2.getString("split/list1"));
assertEquals(0, conf2.getMaxIndex("split/list1"));
assertEquals("a\\,b\\,c", conf2.getString("split/list2"));
}
/**
* Tests string properties with list delimiters when delimiter parsing
* is disabled
*/
@Test
public void testSaveWithDelimiterParsingDisabled() throws ConfigurationException {
conf = new XMLConfiguration();
conf.setExpressionEngine(new XPathExpressionEngine());
load(conf, testProperties);
assertEquals("a,b,c", conf.getString("split/list3/@values"));
assertEquals(0, conf.getMaxIndex("split/list3/@values"));
assertEquals("a\\,b\\,c", conf.getString("split/list4/@values"));
assertEquals("a,b,c", conf.getString("split/list1"));
assertEquals(0, conf.getMaxIndex("split/list1"));
assertEquals("a\\,b\\,c", conf.getString("split/list2"));
// save the configuration
saveTestConfig();
XMLConfiguration config = new XMLConfiguration();
//config.setExpressionEngine(new XPathExpressionEngine());
load(config, testFile2);
config.setProperty("Employee[@attr1]", "3,2,1");
assertEquals("3,2,1", config.getString("Employee[@attr1]"));
new FileHandler(config).save(testSaveFile);
config = new XMLConfiguration();
//config.setExpressionEngine(new XPathExpressionEngine());
load(config, testSaveFile.getAbsolutePath());
config.setProperty("Employee[@attr1]", "1,2,3");
assertEquals("1,2,3", config.getString("Employee[@attr1]"));
config.setProperty("Employee[@attr2]", "one, two, three");
assertEquals("one, two, three", config.getString("Employee[@attr2]"));
config.setProperty("Employee.text", "a,b,d");
assertEquals("a,b,d", config.getString("Employee.text"));
config.setProperty("Employee.Salary", "100,000");
assertEquals("100,000", config.getString("Employee.Salary"));
new FileHandler(config).save(testSaveFile);
XMLConfiguration checkConfig = new XMLConfiguration();
checkConfig.setExpressionEngine(new XPathExpressionEngine());
load(checkConfig, testSaveFile.getAbsolutePath());
assertEquals("1,2,3", checkConfig.getString("Employee/@attr1"));
assertEquals("one, two, three", checkConfig.getString("Employee/@attr2"));
assertEquals("a,b,d", checkConfig.getString("Employee/text"));
assertEquals("100,000", checkConfig.getString("Employee/Salary"));
}
/**
* Tests whether a DTD can be accessed.
*/
@Test
public void testDtd() throws ConfigurationException
{
conf = new XMLConfiguration();
load(conf, "testDtd.xml");
assertEquals("value1", conf.getString("entry(0)"));
assertEquals("test2", conf.getString("entry(1)[@key]"));
}
/**
* Tests DTD validation using the setValidating() method.
*/
@Test
public void testValidating() throws ConfigurationException
{
File nonValidFile = ConfigurationAssert.getTestFile("testValidateInvalid.xml");
conf = new XMLConfiguration();
assertFalse(conf.isValidating());
// Load a non valid XML document. Should work for isValidating() == false
load(conf, nonValidFile.getAbsolutePath());
assertEquals("customers", conf.getString("table.name"));
assertFalse(conf.containsKey("table.fields.field(1).type"));
}
/**
* Tests whether an invalid file is detected when validating is enabled.
*/
@Test(expected = ConfigurationException.class)
public void testValidatingInvalidFile() throws ConfigurationException
{
conf = new XMLConfiguration();
conf.setValidating(true);
load(conf, "testValidateInvalid.xml");
}
/**
* Tests handling of empty elements.
*/
@Test
public void testEmptyElements() throws ConfigurationException
{
assertTrue(conf.containsKey("empty"));
assertEquals("", conf.getString("empty"));
conf.addProperty("empty2", "");
conf.setProperty("empty", "no more empty");
saveTestConfig();
conf = new XMLConfiguration();
load(conf, testSaveConf.getAbsolutePath());
assertEquals("no more empty", conf.getString("empty"));
assertEquals("", conf.getProperty("empty2"));
}
/**
* Tests the isEmpty() method for an empty configuration that was reloaded.
*/
@Test
public void testEmptyReload() throws ConfigurationException
{
conf = new XMLConfiguration();
assertTrue("Newly created configuration not empty", conf.isEmpty());
saveTestConfig();
load(conf, testSaveConf.getAbsolutePath());
assertTrue("Reloaded configuration not empty", conf.isEmpty());
}
/**
* Tests whether the encoding is correctly detected by the XML parser. This
* is done by loading an XML file with the encoding "UTF-16". If this
* encoding is not detected correctly, an exception will be thrown that
* "Content is not allowed in prolog". This test case is related to issue
* 34204.
*/
@Test
public void testLoadWithEncoding() throws ConfigurationException
{
conf = new XMLConfiguration();
new FileHandler(conf).load(ConfigurationAssert.getTestFile("testEncoding.xml"));
assertEquals("test3_yoge", conf.getString("yoge"));
}
/**
* Tests whether the encoding is written to the generated XML file.
*/
@Test
public void testSaveWithEncoding() throws ConfigurationException
{
conf = new XMLConfiguration();
conf.setProperty("test", "a value");
FileHandler handler = new FileHandler(conf);
handler.setEncoding(ENCODING);
StringWriter out = new StringWriter();
handler.save(out);
assertThat("Encoding was not written to file", out.toString(),
containsString("encoding=\"" + ENCODING + "\""));
}
/**
* Tests whether a default encoding is used if no specific encoding is set.
* According to the XSLT specification (http://www.w3.org/TR/xslt#output)
* this should be either UTF-8 or UTF-16.
*/
@Test
public void testSaveWithNullEncoding() throws ConfigurationException
{
conf = new XMLConfiguration();
conf.setProperty("testNoEncoding", "yes");
FileHandler handler = new FileHandler(conf);
StringWriter out = new StringWriter();
handler.save(out);
assertThat("Encoding was written to file", out.toString(),
containsString("encoding=\"UTF-"));
}
/**
* Tests whether the DOCTYPE survives a save operation.
*/
@Test
public void testSaveWithDoctype() throws ConfigurationException
{
conf = new XMLConfiguration();
load(conf, "testDtdPublic.xml");
assertEquals("Wrong public ID", PUBLIC_ID, conf.getPublicID());
assertEquals("Wrong system ID", SYSTEM_ID, conf.getSystemID());
StringWriter out = new StringWriter();
new FileHandler(conf).save(out);
assertThat("Did not find DOCTYPE", out.toString(),
containsString(DOCTYPE));
}
/**
* Tests setting public and system IDs for the DOCTYPE and then saving the
* configuration. This should generate a DOCTYPE declaration.
*/
@Test
public void testSaveWithDoctypeIDs() throws ConfigurationException
{
assertNull("A public ID was found", conf.getPublicID());
assertNull("A system ID was found", conf.getSystemID());
conf.setPublicID(PUBLIC_ID);
conf.setSystemID(SYSTEM_ID);
StringWriter out = new StringWriter();
new FileHandler(conf).save(out);
assertThat("Did not find DOCTYPE", out.toString(), containsString(
DOCTYPE + "testconfig" + DOCTYPE_DECL));
}
/**
* Tests saving a configuration if an invalid transformer factory is
* specified. In this case an error is thrown by the transformer factory.
* XMLConfiguration should not catch this error.
*/
@Test
public void testSaveWithInvalidTransformerFactory() throws ConfigurationException {
System.setProperty(PROP_FACTORY, "an.invalid.Class");
try
{
saveTestConfig();
fail("Could save with invalid TransformerFactory!");
}
catch (TransformerFactoryConfigurationError cex)
{
// ok
}
finally
{
System.getProperties().remove(PROP_FACTORY);
}
}
/**
* Tests accessing properties when the XPATH expression engine is set.
*/
@Test
public void testXPathExpressionEngine()
{
conf.setExpressionEngine(new XPathExpressionEngine());
assertEquals("Wrong attribute value", "foo\"bar", conf
.getString("test[1]/entity/@name"));
conf.clear();
assertNull(conf.getString("test[1]/entity/@name"));
}
/**
* Tests the copy constructor.
*/
@Test
public void testInitCopy() throws ConfigurationException
{
XMLConfiguration copy = new XMLConfiguration(conf);
copy.setListDelimiterHandler(new DefaultListDelimiterHandler(','));
assertEquals("value", copy.getProperty("element"));
assertNull("Document was copied, too", copy.getDocument());
new FileHandler(copy).save(testSaveConf);
checkSavedConfig();
}
/**
* Tests setting text of the root element.
*/
@Test
public void testSetTextRootElement() throws ConfigurationException
{
conf.setProperty("", "Root text");
saveTestConfig();
checkSavedConfig();
}
/**
* Tests removing the text of the root element.
*/
@Test
public void testClearTextRootElement() throws ConfigurationException
{
final String xml = "<e a=\"v\">text</e>";
conf.clear();
StringReader in = new StringReader(xml);
FileHandler handler = new FileHandler(conf);
handler.load(in);
assertEquals("Wrong text of root", "text", conf.getString(""));
conf.clearProperty("");
saveTestConfig();
checkSavedConfig();
}
/**
* Tests list nodes with multiple values and attributes.
*/
@Test
public void testListWithAttributes()
{
assertEquals("Wrong number of <a> elements", 6, conf.getList(
"attrList.a").size());
assertEquals("Wrong value of first element", "ABC", conf
.getString("attrList.a(0)"));
assertEquals("Wrong value of first name attribute", "x", conf
.getString("attrList.a(0)[@name]"));
assertEquals("Wrong number of name attributes", 6, conf.getList(
"attrList.a[@name]").size());
}
/**
* Tests a list node with attributes that has multiple values separated by
* the list delimiter. In this scenario the attribute should be added to all
* list nodes.
*/
@Test
public void testListWithAttributesMultiValue()
{
assertEquals("Wrong value of 2nd element", "1",
conf.getString("attrList.a(1)"));
assertEquals("Wrong value of 2nd name attribute", "y",
conf.getString("attrList.a(1)[@name]"));
for (int i = 1; i <= 3; i++)
{
assertEquals("Wrong value of element " + (i + 1), i,
conf.getInt("attrList.a(" + i + ")"));
assertEquals("Wrong name attribute for element " + (i), "y",
conf.getString("attrList.a(" + i + ")[@name]"));
}
}
/**
* Tests a list node with multiple values and multiple attributes. All
* attribute values should be assigned to all list nodes.
*/
@Test
public void testListWithMultipleAttributesMultiValue()
{
for (int i = 1; i <= 2; i++)
{
String idxStr = String.format("(%d)", Integer.valueOf(i + 3));
String nodeKey = "attrList.a" + idxStr;
assertEquals("Wrong value of multi-valued node", "value" + i,
conf.getString(nodeKey));
assertEquals("Wrong name attribute at " + i, "u",
conf.getString(nodeKey + "[@name]"));
assertEquals("Wrong test attribute at " + i, "yes",
conf.getString(nodeKey + "[@test]"));
}
}
/**
* Tests whether the auto save mechanism is triggered by changes at a
* subnode configuration.
*/
@Test
public void testAutoSaveWithSubnodeConfig() throws ConfigurationException
{
FileBasedConfigurationBuilder<XMLConfiguration> builder =
new FileBasedConfigurationBuilder<XMLConfiguration>(
XMLConfiguration.class);
builder.configure(new FileBasedBuilderParametersImpl()
.setFileName(testProperties));
conf = builder.getConfiguration();
builder.getFileHandler().setFile(testSaveConf);
builder.setAutoSave(true);
final String newValue = "I am autosaved";
Configuration sub = conf.configurationAt("element2.subelement", true);
sub.setProperty("subsubelement", newValue);
assertEquals("Change not visible to parent", newValue,
conf.getString("element2.subelement.subsubelement"));
XMLConfiguration conf2 = new XMLConfiguration();
load(conf2, testSaveConf.getAbsolutePath());
assertEquals("Change was not saved", newValue,
conf2.getString("element2.subelement.subsubelement"));
}
/**
* Tests whether a subnode configuration created from another subnode
* configuration of a XMLConfiguration can trigger the auto save mechanism.
*/
@Test
public void testAutoSaveWithSubSubnodeConfig() throws ConfigurationException
{
FileBasedConfigurationBuilder<XMLConfiguration> builder =
new FileBasedConfigurationBuilder<XMLConfiguration>(
XMLConfiguration.class);
builder.configure(new FileBasedBuilderParametersImpl()
.setFileName(testProperties));
conf = builder.getConfiguration();
builder.getFileHandler().setFile(testSaveConf);
builder.setAutoSave(true);
final String newValue = "I am autosaved";
HierarchicalConfiguration<?> sub1 = conf.configurationAt("element2", true);
HierarchicalConfiguration<?> sub2 = sub1.configurationAt("subelement", true);
sub2.setProperty("subsubelement", newValue);
assertEquals("Change not visible to parent", newValue, conf
.getString("element2.subelement.subsubelement"));
XMLConfiguration conf2 = new XMLConfiguration();
load(conf2, testSaveConf.getAbsolutePath());
assertEquals("Change was not saved", newValue, conf2
.getString("element2.subelement.subsubelement"));
}
/**
* Tests saving and loading a configuration when delimiter parsing is
* disabled.
*/
@Test
public void testSaveDelimiterParsingDisabled()
throws ConfigurationException
{
checkSaveDelimiterParsingDisabled("list.delimiter.test");
}
/**
* Helper method for testing saving and loading a configuration when
* delimiter parsing is disabled.
*
* @param key the key to be checked
* @throws ConfigurationException if an error occurs
*/
private void checkSaveDelimiterParsingDisabled(String key)
throws ConfigurationException
{
conf.clear();
conf.setListDelimiterHandler(new DisabledListDelimiterHandler());
load(conf, testProperties);
conf.setProperty(key, "C:\\Temp\\,C:\\Data\\");
conf.addProperty(key, "a,b,c");
saveTestConfig();
XMLConfiguration checkConf = new XMLConfiguration();
checkConf.setListDelimiterHandler(conf.getListDelimiterHandler());
load(checkConf, testSaveConf.getAbsolutePath());
ConfigurationAssert.assertConfigurationEquals(conf, checkConf);
}
/**
* Tests that attribute values are not split.
*/
@Test
public void testNoDelimiterParsingInAttrValues() throws ConfigurationException
{
conf.clear();
load(conf, testProperties);
List<Object> expr = conf.getList("expressions[@value]");
assertEquals("Wrong list size", 1, expr.size());
assertEquals("Wrong element 1", "a || (b && c) | !d", expr.get(0));
}
/**
* Tries to create an attribute with multiple values. Only the first value
* is taken into account.
*/
@Test
public void testAttributeKeyWithMultipleValues()
throws ConfigurationException
{
conf.addProperty("errorTest[@multiAttr]", Arrays.asList("v1", "v2"));
saveTestConfig();
XMLConfiguration checkConfig = new XMLConfiguration();
load(checkConfig, testSaveConf.getAbsolutePath());
assertEquals("Wrong attribute value", "v1",
checkConfig.getString("errorTest[@multiAttr]"));
}
/**
* Tests adding nodes from another configuration.
*/
@Test
public void testAddNodesCopy() throws ConfigurationException
{
XMLConfiguration c2 = new XMLConfiguration();
load(c2, testProperties2);
conf.addNodes("copiedProperties", c2.getModel().getNodeHandler()
.getRootNode().getChildren());
saveTestConfig();
checkSavedConfig();
}
/**
* Tests whether the addNodes() method triggers an auto save.
*/
@Test
public void testAutoSaveAddNodes() throws ConfigurationException
{
FileBasedConfigurationBuilder<XMLConfiguration> builder =
new FileBasedConfigurationBuilder<XMLConfiguration>(
XMLConfiguration.class);
builder.configure(new FileBasedBuilderParametersImpl()
.setFileName(testProperties));
conf = builder.getConfiguration();
builder.getFileHandler().setFile(testSaveConf);
builder.setAutoSave(true);
ImmutableNode node = NodeStructureHelper.createNode(
"addNodesTest", Boolean.TRUE);
Collection<ImmutableNode> nodes = new ArrayList<ImmutableNode>(1);
nodes.add(node);
conf.addNodes("test.autosave", nodes);
XMLConfiguration c2 = new XMLConfiguration();
load(c2, testSaveConf.getAbsolutePath());
assertTrue("Added nodes are not saved", c2
.getBoolean("test.autosave.addNodesTest"));
}
/**
* Tests saving a configuration after a node was added. Test for
* CONFIGURATION-294.
*/
@Test
public void testAddNodesAndSave() throws ConfigurationException
{
ImmutableNode.Builder bldrNode = new ImmutableNode.Builder(1);
bldrNode.addChild(NodeStructureHelper.createNode("child", null));
bldrNode.addAttribute("attr", "");
ImmutableNode node2 = NodeStructureHelper.createNode("test2", null);
conf.addNodes("add.nodes",
Arrays.asList(bldrNode.name("test").create(), node2));
saveTestConfig();
conf.setProperty("add.nodes.test", "true");
conf.setProperty("add.nodes.test.child", "yes");
conf.setProperty("add.nodes.test[@attr]", "existing");
conf.setProperty("add.nodes.test2", "anotherValue");
saveTestConfig();
XMLConfiguration c2 = new XMLConfiguration();
load(c2, testSaveConf.getAbsolutePath());
assertEquals("Value was not saved", "true", c2
.getString("add.nodes.test"));
assertEquals("Child value was not saved", "yes", c2
.getString("add.nodes.test.child"));
assertEquals("Attr value was not saved", "existing", c2
.getString("add.nodes.test[@attr]"));
assertEquals("Node2 not saved", "anotherValue", c2
.getString("add.nodes.test2"));
}
/**
* Tests saving a configuration that was created from a hierarchical
* configuration. This test exposes bug CONFIGURATION-301.
*/
@Test
public void testSaveAfterCreateWithCopyConstructor()
throws ConfigurationException
{
HierarchicalConfiguration<ImmutableNode> hc =
conf.configurationAt("element2");
conf = new XMLConfiguration(hc);
saveTestConfig();
XMLConfiguration checkConfig = checkSavedConfig();
assertEquals("Wrong name of root element", "element2", checkConfig
.getRootElementName());
}
/**
* Tests whether the name of the root element is copied when a configuration
* is created using the copy constructor.
*/
@Test
public void testCopyRootName() throws ConfigurationException
{
final String rootName = "rootElement";
final String xml = "<" + rootName + "><test>true</test></" + rootName
+ ">";
conf.clear();
new FileHandler(conf).load(new StringReader(xml));
XMLConfiguration copy = new XMLConfiguration(conf);
assertEquals("Wrong name of root element", rootName, copy
.getRootElementName());
new FileHandler(copy).save(testSaveConf);
copy = new XMLConfiguration();
load(copy, testSaveConf.getAbsolutePath());
assertEquals("Wrong name of root element after save", rootName, copy
.getRootElementName());
}
/**
* Tests whether the name of the root element is copied for a configuration
* for which not yet a document exists.
*/
@Test
public void testCopyRootNameNoDocument() throws ConfigurationException
{
final String rootName = "rootElement";
conf = new XMLConfiguration();
conf.setRootElementName(rootName);
conf.setProperty("test", Boolean.TRUE);
XMLConfiguration copy = new XMLConfiguration(conf);
assertEquals("Wrong name of root element", rootName, copy
.getRootElementName());
new FileHandler(copy).save(testSaveConf);
load(copy, testSaveConf.getAbsolutePath());
assertEquals("Wrong name of root element after save", rootName, copy
.getRootElementName());
}
/**
* Tests the copy constructor for null input.
*/
@Test
public void testCopyNull()
{
conf = new XMLConfiguration(null);
assertTrue("Not empty", conf.isEmpty());
assertEquals("Wrong root element name", "configuration",
conf.getRootElementName());
}
/**
* Tests whether spaces are preserved when the xml:space attribute is set.
*/
@Test
public void testPreserveSpace()
{
assertEquals("Wrong value of blanc", " ", conf.getString("space.blanc"));
assertEquals("Wrong value of stars", " * * ", conf
.getString("space.stars"));
}
/**
* Tests whether the xml:space attribute works directly on the current
* element. This test is related to CONFIGURATION-555.
*/
@Test
public void testPreserveSpaceOnElement()
{
assertEquals("Wrong value", " preserved ", conf.getString("spaceElement"));
}
/**
* Tests whether the xml:space attribute can be overridden in nested
* elements.
*/
@Test
public void testPreserveSpaceOverride()
{
assertEquals("Not trimmed", "Some text", conf
.getString("space.description"));
}
/**
* Tests an xml:space attribute with an invalid value. This will be
* interpreted as default.
*/
@Test
public void testPreserveSpaceInvalid()
{
assertEquals("Invalid not trimmed", "Some other text", conf
.getString("space.testInvalid"));
}
/**
* Tests modifying an XML document and saving it with schema validation enabled.
*/
@Test
public void testSaveWithValidation() throws Exception
{
CatalogResolver resolver = new CatalogResolver();
resolver.setCatalogFiles(CATALOG_FILES);
conf = new XMLConfiguration();
conf.setEntityResolver(resolver);
conf.setSchemaValidation(true);
load(conf, testFile2);
conf.setProperty("Employee.SSN", "123456789");
SynchronizerTestImpl sync = new SynchronizerTestImpl();
conf.setSynchronizer(sync);
conf.validate();
sync.verify(Methods.BEGIN_WRITE, Methods.END_WRITE);
saveTestConfig();
conf = new XMLConfiguration();
load(conf, testSaveConf.getAbsolutePath());
assertEquals("123456789", conf.getString("Employee.SSN"));
}
/**
* Tests modifying an XML document and validating it against the schema.
*/
@Test
public void testSaveWithValidationFailure() throws Exception
{
CatalogResolver resolver = new CatalogResolver();
resolver.setCatalogFiles(CATALOG_FILES);
conf = new XMLConfiguration();
conf.setEntityResolver(resolver);
conf.setSchemaValidation(true);
load(conf, testFile2);
conf.setProperty("Employee.Email", "JohnDoe@apache.org");
try
{
conf.validate();
fail("No validation failure on save");
}
catch (Exception e)
{
Throwable cause = e.getCause();
assertNotNull("No cause for exception on save", cause);
assertTrue("Incorrect exception on save", cause instanceof SAXParseException);
}
}
@Test
public void testConcurrentGetAndReload() throws ConfigurationException,
InterruptedException
{
FileBasedConfigurationBuilder<XMLConfiguration> builder =
new FileBasedConfigurationBuilder<XMLConfiguration>(
XMLConfiguration.class);
builder.configure(new FileBasedBuilderParametersImpl()
.setFileName(testProperties));
XMLConfiguration config = builder.getConfiguration();
assertTrue("Property not found",
config.getProperty("test.short") != null);
Thread testThreads[] = new Thread[THREAD_COUNT];
for (int i = 0; i < testThreads.length; ++i)
{
testThreads[i] = new ReloadThread(builder);
testThreads[i].start();
}
for (int i = 0; i < LOOP_COUNT; i++)
{
config = builder.getConfiguration();
assertTrue("Property not found", config.getProperty("test.short") != null);
}
for (Thread testThread : testThreads) {
testThread.join();
}
}
/**
* Tests whether a windows path can be saved correctly. This test is related
* to CONFIGURATION-428.
*/
@Test
public void testSaveWindowsPath() throws ConfigurationException
{
conf.clear();
conf.setListDelimiterHandler(new DisabledListDelimiterHandler());
conf.addProperty("path", "C:\\Temp");
StringWriter writer = new StringWriter();
new FileHandler(conf).save(writer);
String content = writer.toString();
assertThat("Path not found: ", content,
containsString("<path>C:\\Temp</path>"));
saveTestConfig();
XMLConfiguration conf2 = new XMLConfiguration();
load(conf2, testSaveConf.getAbsolutePath());
assertEquals("Wrong windows path", "C:\\Temp",
conf2.getString("path"));
}
/**
* Tests whether an attribute can be set to an empty string. This test is
* related to CONFIGURATION-446.
*/
@Test
public void testEmptyAttribute() throws ConfigurationException
{
String key = "element3[@value]";
conf.setProperty(key, "");
assertTrue("Key not found", conf.containsKey(key));
assertEquals("Wrong value", "", conf.getString(key));
saveTestConfig();
conf = new XMLConfiguration();
load(conf, testSaveConf.getAbsolutePath());
assertTrue("Key not found after save", conf.containsKey(key));
assertEquals("Wrong value after save", "", conf.getString(key));
}
/**
* Tests whether it is possible to add nodes to a XMLConfiguration through a
* SubnodeConfiguration and whether these nodes have the correct type. This
* test is related to CONFIGURATION-472.
*/
@Test
public void testAddNodesToSubnodeConfiguration() throws Exception
{
HierarchicalConfiguration<ImmutableNode> sub =
conf.configurationAt("element2", true);
sub.addProperty("newKey", "newvalue");
assertEquals("Property not added", "newvalue",
conf.getString("element2.newKey"));
}
/**
* Tests whether list properties are set correctly if delimiter
* parsing is disabled. This test is related to CONFIGURATION-495.
*/
@Test
public void testSetPropertyListWithDelimiterParsingDisabled()
throws ConfigurationException
{
String prop = "delimiterListProp";
List<String> list = Arrays.asList("val", "val2", "val3");
conf.setProperty(prop, list);
saveTestConfig();
XMLConfiguration conf2 = new XMLConfiguration();
load(conf2, testSaveConf.getAbsolutePath());
assertEquals("Wrong list property", list, conf2.getProperty(prop));
}
/**
* Tests whether list properties are added correctly if delimiter parsing is
* disabled. This test is related to CONFIGURATION-495.
*/
@Test
public void testAddPropertyListWithDelimiterParsingDisabled()
throws ConfigurationException
{
String prop = "delimiterListProp";
conf.setListDelimiterHandler(DisabledListDelimiterHandler.INSTANCE);
List<String> list = Arrays.asList("val", "val2", "val3");
conf.addProperty(prop, list);
saveTestConfig();
XMLConfiguration conf2 = new XMLConfiguration();
load(conf2, testSaveConf.getAbsolutePath());
assertEquals("Wrong list property", list, conf2.getProperty(prop));
}
/**
* Tests whether the system ID is accessed in a synchronized manner.
*/
@Test
public void testSystemIdSynchronized()
{
SynchronizerTestImpl sync = new SynchronizerTestImpl();
conf.setSynchronizer(sync);
conf.setSystemID(SYSTEM_ID);
assertEquals("SystemID not set", SYSTEM_ID, conf.getSystemID());
sync.verify(Methods.BEGIN_WRITE, Methods.END_WRITE, Methods.BEGIN_READ,
Methods.END_READ);
}
/**
* Tests whether the public ID is accessed in a synchronized manner.
*/
@Test
public void testPublicIdSynchronized()
{
SynchronizerTestImpl sync = new SynchronizerTestImpl();
conf.setSynchronizer(sync);
conf.setPublicID(PUBLIC_ID);
assertEquals("PublicID not set", PUBLIC_ID, conf.getPublicID());
sync.verify(Methods.BEGIN_WRITE, Methods.END_WRITE, Methods.BEGIN_READ,
Methods.END_READ);
}
/**
* Removes the test output file if it exists.
*/
private void removeTestFile()
{
if (testSaveConf.exists())
{
assertTrue(testSaveConf.delete());
}
}
/**
* Helper method for saving the test configuration to the default output
* file.
*
* @throws ConfigurationException if an error occurs
*/
private void saveTestConfig() throws ConfigurationException
{
FileHandler handler = new FileHandler(conf);
handler.save(testSaveConf);
}
/**
* Tests whether the saved configuration file matches the original data.
*
* @param saveFile the saved configuration file
* @return the newly loaded configuration
* @throws ConfigurationException if an error occurs
*/
private XMLConfiguration checkSavedConfig(File saveFile)
throws ConfigurationException
{
XMLConfiguration config = createFromFile(saveFile.getAbsolutePath());
ConfigurationAssert.assertConfigurationEquals(conf, config);
return config;
}
/**
* Helper method for testing whether a configuration was correctly saved to
* the default output file.
*
* @return the newly loaded configuration
* @throws ConfigurationException if an error occurs
*/
private XMLConfiguration checkSavedConfig() throws ConfigurationException
{
return checkSavedConfig(testSaveConf);
}
/**
* A thread used for testing concurrent access to a builder.
*/
private class ReloadThread extends Thread
{
private final FileBasedConfigurationBuilder<?> builder;
ReloadThread(FileBasedConfigurationBuilder<?> confBulder)
{
builder = confBulder;
}
@Override
public void run()
{
for (int i = 0; i < LOOP_COUNT; i++)
{
builder.resetResult();
}
}
}
}