View Javadoc
1   /*
2    * Licensed to the Apache Software Foundation (ASF) under one or more
3    * contributor license agreements.  See the NOTICE file distributed with
4    * this work for additional information regarding copyright ownership.
5    * The ASF licenses this file to You under the Apache License, Version 2.0
6    * (the "License"); you may not use this file except in compliance with
7    * the License.  You may obtain a copy of the License at
8    *
9    *     http://www.apache.org/licenses/LICENSE-2.0
10   *
11   * Unless required by applicable law or agreed to in writing, software
12   * distributed under the License is distributed on an "AS IS" BASIS,
13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14   * See the License for the specific language governing permissions and
15   * limitations under the License.
16   */
17  
18  package org.apache.commons.configuration2;
19  
20  import static org.junit.jupiter.api.Assertions.assertEquals;
21  import static org.junit.jupiter.api.Assertions.assertFalse;
22  import static org.junit.jupiter.api.Assertions.assertNull;
23  import static org.junit.jupiter.api.Assertions.assertSame;
24  import static org.junit.jupiter.api.Assertions.assertThrows;
25  import static org.junit.jupiter.api.Assertions.assertTrue;
26  import static org.mockito.Mockito.mock;
27  
28  import java.lang.reflect.Constructor;
29  import java.util.ArrayList;
30  import java.util.HashSet;
31  import java.util.Iterator;
32  import java.util.List;
33  import java.util.NoSuchElementException;
34  import java.util.Set;
35  
36  import org.apache.commons.configuration2.builder.FileBasedBuilderParametersImpl;
37  import org.apache.commons.configuration2.builder.combined.CombinedConfigurationBuilder;
38  import org.apache.commons.configuration2.convert.DefaultListDelimiterHandler;
39  import org.apache.commons.configuration2.convert.ListDelimiterHandler;
40  import org.apache.commons.configuration2.interpol.ConfigurationInterpolator;
41  import org.junit.jupiter.api.Test;
42  
43  /**
44   * Test case for the {@link SubsetConfiguration} class.
45   */
46  public class TestSubsetConfiguration {
47      static final String TEST_DIR = ConfigurationAssert.TEST_DIR_NAME;
48      static final String TEST_FILE = "testDigesterConfiguration2.xml";
49  
50      @Test
51      public void testClear() {
52          final Configuration config = new BaseConfiguration();
53          config.setProperty("test.key1", "value1");
54          config.setProperty("testing.key2", "value1");
55  
56          final Configuration subset = config.subset("test");
57          subset.clear();
58  
59          assertTrue(subset.isEmpty());
60          assertFalse(config.isEmpty());
61      }
62  
63      /**
64       * Tries to create an instance without a parent configuration.
65       */
66      @Test
67      public void testConstructNullParent() {
68          assertThrows(NullPointerException.class, () -> new SubsetConfiguration(null, ""));
69      }
70  
71      @Test
72      public void testGetChildKey() {
73          final Configuration conf = new BaseConfiguration();
74          // subset with delimiter
75          SubsetConfiguration subset = new SubsetConfiguration(conf, "prefix", ".");
76          assertEquals("key", subset.getChildKey("prefix.key"));
77          assertEquals("", subset.getChildKey("prefix"));
78  
79          // subset without delimiter
80          subset = new SubsetConfiguration(conf, "prefix", null);
81          assertEquals("key", subset.getChildKey("prefixkey"));
82          assertEquals("", subset.getChildKey("prefix"));
83      }
84  
85      @Test
86      public void testGetKeys() {
87          final Configuration conf = new BaseConfiguration();
88          conf.setProperty("test", "value0");
89          conf.setProperty("test.key1", "value1");
90          conf.setProperty("testing.key2", "value1");
91  
92          final Configuration subset = new SubsetConfiguration(conf, "test", ".");
93  
94          final Iterator<String> it = subset.getKeys();
95          assertEquals("", it.next());
96          assertEquals("key1", it.next());
97          assertFalse(it.hasNext());
98      }
99  
100     @Test
101     public void testGetKeysWithPrefix() {
102         final Configuration conf = new BaseConfiguration();
103         conf.setProperty("test.abc", "value0");
104         conf.setProperty("test.abc.key1", "value1");
105         conf.setProperty("test.abcdef.key2", "value1");
106 
107         final Configuration subset = new SubsetConfiguration(conf, "test", ".");
108 
109         final Iterator<String> it = subset.getKeys("abc");
110         assertEquals("abc", it.next());
111         assertEquals("abc.key1", it.next());
112         assertFalse(it.hasNext());
113     }
114 
115     @Test
116     public void testGetList() {
117         final BaseConfiguration conf = new BaseConfiguration();
118         conf.setListDelimiterHandler(new DefaultListDelimiterHandler(','));
119         conf.setProperty("test.abc", "value0,value1");
120         conf.addProperty("test.abc", "value3");
121 
122         final Configuration subset = new SubsetConfiguration(conf, "test", ".");
123         final List<Object> list = subset.getList("abc", new ArrayList<>());
124         assertEquals(3, list.size());
125     }
126 
127     /**
128      * Tests whether the list delimiter handler from the parent configuration is used.
129      */
130     @Test
131     public void testGetListDelimiterHandlerFromParent() {
132         final BaseConfiguration config = new BaseConfiguration();
133         final AbstractConfiguration subset = (AbstractConfiguration) config.subset("prefix");
134         final ListDelimiterHandler listHandler = new DefaultListDelimiterHandler(',');
135         config.setListDelimiterHandler(listHandler);
136         assertSame(listHandler, subset.getListDelimiterHandler());
137     }
138 
139     @Test
140     public void testGetParent() {
141         final Configuration conf = new BaseConfiguration();
142         final SubsetConfiguration subset = new SubsetConfiguration(conf, "prefix", ".");
143 
144         assertEquals(conf, subset.getParent());
145     }
146 
147     @Test
148     public void testGetParentKey() {
149         final Configuration conf = new BaseConfiguration();
150         // subset with delimiter
151         SubsetConfiguration subset = new SubsetConfiguration(conf, "prefix", ".");
152         assertEquals("prefix.key", subset.getParentKey("key"));
153         assertEquals("prefix", subset.getParentKey(""));
154 
155         // subset without delimiter
156         subset = new SubsetConfiguration(conf, "prefix", null);
157         assertEquals("prefixkey", subset.getParentKey("key"));
158         assertEquals("prefix", subset.getParentKey(""));
159     }
160 
161     @Test
162     public void testGetPrefix() {
163         final Configuration conf = new BaseConfiguration();
164         final SubsetConfiguration subset = new SubsetConfiguration(conf, "prefix", ".");
165 
166         assertEquals("prefix", subset.getPrefix());
167     }
168 
169     @Test
170     public void testGetProperty() {
171         final Configuration conf = new BaseConfiguration();
172         conf.setProperty("test.key1", "value1");
173         conf.setProperty("testing.key2", "value1");
174 
175         final Configuration subset = new SubsetConfiguration(conf, "test", ".");
176         assertFalse(subset.isEmpty());
177         assertTrue(subset.containsKey("key1"));
178         assertFalse(subset.containsKey("ng.key2"));
179     }
180 
181     @Test
182     public void testInterpolationForKeysOfTheParent() {
183         final BaseConfiguration config = new BaseConfiguration();
184         config.setProperty("test", "junit");
185         config.setProperty("prefix.key", "${test}");
186         final AbstractConfiguration subset = (AbstractConfiguration) config.subset("prefix");
187         assertEquals("junit", subset.getString("key", ""));
188     }
189 
190     /**
191      * Tests manipulating the interpolator.
192      */
193     @Test
194     public void testInterpolator() {
195         final BaseConfiguration config = new BaseConfiguration();
196         final AbstractConfiguration subset = (AbstractConfiguration) config.subset("prefix");
197         InterpolationTestHelper.testGetInterpolator(subset);
198     }
199 
200     /**
201      * Tests whether a list delimiter handler is used correctly.
202      */
203     @Test
204     public void testListDelimiterHandling() {
205         final BaseConfiguration config = new BaseConfiguration();
206         final Configuration subset = config.subset("prefix");
207         config.setListDelimiterHandler(new DefaultListDelimiterHandler('/'));
208         subset.addProperty("list", "a/b/c");
209         assertEquals(3, config.getList("prefix.list").size());
210 
211         ((AbstractConfiguration) subset).setListDelimiterHandler(new DefaultListDelimiterHandler(';'));
212         subset.addProperty("list2", "a;b;c");
213         assertEquals(3, config.getList("prefix.list2").size());
214     }
215 
216     @Test
217     public void testLocalLookupsInInterpolatorAreInherited() {
218         final BaseConfiguration config = new BaseConfiguration();
219         final ConfigurationInterpolator interpolator = config.getInterpolator();
220         interpolator.registerLookup("brackets", key -> "(" + key + ")");
221         config.setProperty("prefix.var", "${brackets:x}");
222         final AbstractConfiguration subset = (AbstractConfiguration) config.subset("prefix");
223         assertEquals("(x)", subset.getString("var", ""));
224     }
225 
226     @Test
227     public void testNested() throws Exception {
228         final CombinedConfigurationBuilder builder = new CombinedConfigurationBuilder();
229         builder.configure(new FileBasedBuilderParametersImpl().setFile(ConfigurationAssert.getTestFile(TEST_FILE)));
230         final Configuration config = builder.getConfiguration();
231         final Configuration subConf = config.subset("tables.table(0)");
232         assertTrue(subConf.getKeys().hasNext());
233         final Configuration subSubConf = subConf.subset("fields.field(1)");
234         final Iterator<String> itKeys = subSubConf.getKeys();
235         final Set<String> keys = new HashSet<>();
236         keys.add("name");
237         keys.add("type");
238         while (itKeys.hasNext()) {
239             final String k = itKeys.next();
240             assertTrue(keys.contains(k));
241             keys.remove(k);
242         }
243         assertTrue(keys.isEmpty());
244     }
245 
246     @Test
247     public void testPrefixDelimiter(){
248         final BaseConfiguration config = new BaseConfiguration();
249         config.setProperty("part1.part2@test.key1", "value1");
250         config.setProperty("part1.part2", "value2");
251         config.setProperty("part3.part4@testing.key2", "value3");
252 
253         final SubsetConfiguration subset = new SubsetConfiguration(config, "part1.part2", "@");
254         // Check subset properties
255         assertEquals("value1", subset.getString("test.key1"));
256         assertEquals("value2", subset.getString(""));
257         assertNull(subset.getString("testing.key2"));
258 
259         // Check for empty subset configuration and iterator
260         assertEquals(2, subset.size());
261         assertFalse(subset.isEmpty());
262         assertTrue(subset.getKeys().hasNext());
263     }
264 
265     @Test
266     public void testPrefixDelimiterNegativeTest(){
267         final BaseConfiguration config = new BaseConfiguration();
268         config.setProperty("part1.part2@test.key1", "value1");
269         config.setProperty("part3.part4@testing.key2", "value2");
270 
271         final SubsetConfiguration subset = new SubsetConfiguration(config, "part1.part2", "@") {
272             // Anonymous inner class declaration to override SubsetConfiguration.getKeysInternal() - Call
273             // ImutableConfiguration.getKeys(String) on the parent configuration of the SubsetConfiguration in order to
274             // not consequently pass the prefix delimiter
275             @Override
276             protected Iterator<String> getKeysInternal() {
277                 Class<?> subsetIteratorClass;
278                 try {
279                     subsetIteratorClass = Class
280                             .forName("org.apache.commons.configuration2.SubsetConfiguration$SubsetIterator");
281                     final Constructor<?> ctor = subsetIteratorClass.getDeclaredConstructor(SubsetConfiguration.class,
282                             Iterator.class);
283                     ctor.setAccessible(true);
284 
285                     return (Iterator<String>) ctor.newInstance(this, parent.getKeys("part1.part2"));
286                 } catch (final Exception ex) {
287                     throw new IllegalArgumentException(ex);
288                 }
289             }
290         };
291 
292         // Check subset properties - contains one property
293         assertEquals("value1", subset.getString("test.key1"));
294         assertNull(subset.getString("testing.key2"));
295 
296         // Check for empty subset configuration and iterator - even if the SubsetConfiguration contains properties, like
297         // checked previously its states that it is empty
298         assertEquals(0, subset.size());
299         assertTrue(subset.isEmpty());
300         assertFalse(subset.getKeys().hasNext());
301     }
302 
303     /**
304      * Tests whether the list delimiter handler is also set for the parent configuration.
305      */
306     @Test
307     public void testSetListDelimiterHandlerInParent() {
308         final BaseConfiguration config = new BaseConfiguration();
309         final AbstractConfiguration subset = (AbstractConfiguration) config.subset("prefix");
310         final ListDelimiterHandler listHandler = new DefaultListDelimiterHandler(',');
311         subset.setListDelimiterHandler(listHandler);
312         assertSame(listHandler, config.getListDelimiterHandler());
313     }
314 
315     /**
316      * Tests the case that the parent configuration is not derived from AbstractConfiguration and thus does not support a
317      * list delimiter handler.
318      */
319     @Test
320     public void testSetListDelimiterHandlerParentNotSupported() {
321         final Configuration config = mock(Configuration.class);
322         final SubsetConfiguration subset = new SubsetConfiguration(config, "prefix");
323         final ListDelimiterHandler listHandler = new DefaultListDelimiterHandler(',');
324         subset.setListDelimiterHandler(listHandler);
325         assertSame(listHandler, subset.getListDelimiterHandler());
326     }
327 
328     @Test
329     public void testSetPrefix() {
330         final Configuration conf = new BaseConfiguration();
331         final SubsetConfiguration subset = new SubsetConfiguration(conf, null, ".");
332         subset.setPrefix("prefix");
333 
334         assertEquals("prefix", subset.getPrefix());
335     }
336 
337     @Test
338     public void testSetProperty() {
339         final Configuration conf = new BaseConfiguration();
340         final Configuration subset = new SubsetConfiguration(conf, "test", ".");
341 
342         // set a property in the subset and check the parent
343         subset.setProperty("key1", "value1");
344         assertEquals("value1", subset.getProperty("key1"));
345         assertEquals("value1", conf.getProperty("test.key1"));
346 
347         // set a property in the parent and check in the subset
348         conf.setProperty("test.key2", "value2");
349         assertEquals("value2", conf.getProperty("test.key2"));
350         assertEquals("value2", subset.getProperty("key2"));
351     }
352 
353     @Test
354     public void testThrowExceptionOnMissing() {
355         final BaseConfiguration config = new BaseConfiguration();
356         config.setThrowExceptionOnMissing(true);
357 
358         final SubsetConfiguration subset = new SubsetConfiguration(config, "prefix");
359 
360         assertThrows(NoSuchElementException.class, () -> subset.getString("foo"));
361 
362         config.setThrowExceptionOnMissing(false);
363         assertNull(subset.getString("foo"));
364 
365         subset.setThrowExceptionOnMissing(true);
366         assertThrows(NoSuchElementException.class, () -> config.getString("foo"));
367     }
368 }