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 * https://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 java.util.ArrayList;
21 import java.util.Collection;
22 import java.util.Iterator;
23 import java.util.List;
24 import java.util.Map;
25 import java.util.Objects;
26 import java.util.Properties;
27
28 import org.apache.commons.configuration2.ex.ConfigurationRuntimeException;
29
30 /**
31 * <p>
32 * A Map based Configuration.
33 * </p>
34 * <p>
35 * This implementation of the {@code Configuration} interface is initialized with a {@link java.util.Map}. The methods
36 * of the {@code Configuration} interface are implemented on top of the content of this map. The following storage
37 * scheme is used:
38 * </p>
39 * <p>
40 * Property keys are directly mapped to map keys, i.e. the {@code getProperty()} method directly performs a
41 * {@code get()} on the map. Analogously, {@code setProperty()} or {@code addProperty()} operations write new data into
42 * the map. If a value is added to an existing property, a {@link java.util.List} is created, which stores the values of
43 * this property.
44 * </p>
45 * <p>
46 * An important use case of this class is to treat a map as a {@code Configuration} allowing access to its data through
47 * the richer interface. This can be a bit problematic in some cases because the map may contain values that need not
48 * adhere to the default storage scheme used by typical configuration implementations, for example regarding lists. In such
49 * cases care must be taken when manipulating the data through the {@code Configuration} interface, for example by calling
50 * {@code addProperty()}; results may be different than expected.
51 * </p>
52 * <p>
53 * The handling of list delimiters is a bit different for this configuration implementation: When a property of type
54 * String is queried, it is passed to the current {@link org.apache.commons.configuration2.convert.ListDelimiterHandler
55 * ListDelimiterHandler} which may generate multiple values. Note that per default a list delimiter handler is set which
56 * does not do any list splitting, so this feature is disabled. It can be enabled by setting a properly configured
57 * {@code ListDelimiterHandler} implementation, for example a
58 * {@link org.apache.commons.configuration2.convert.DefaultListDelimiterHandler DefaultListDelimiterHandler} object.
59 * </p>
60 * <p>
61 * Notice that list splitting is only performed for single string values. If a property has multiple values, the single
62 * values are not split even if they contain the list delimiter character.
63 * </p>
64 * <p>
65 * As the underlying {@code Map} is directly used as store of the property values, the thread-safety of this
66 * {@code Configuration} implementation depends on the map passed to the constructor.
67 * </p>
68 * <p>
69 * Notes about type safety: For properties with multiple values this implementation creates lists of type {@code Object}
70 * and stores them. If a property is assigned another value, the value is added to the list. This can cause problems if
71 * the map passed to the constructor already contains lists of other types. This should be avoided, otherwise it cannot
72 * be guaranteed that the application might throw {@code ClassCastException} exceptions later.
73 * </p>
74 *
75 * @since 1.1
76 */
77 public class MapConfiguration extends AbstractConfiguration implements Cloneable {
78 /**
79 * Helper method for converting the type of the {@code Properties} object to a supported map type. As stated by the
80 * comment of the constructor, we expect the {@code Properties} object to contain only String key; therefore, it is safe
81 * to do this cast.
82 *
83 * @param props the {@code Properties} to be copied.
84 * @return a newly created map with all string keys of the properties.
85 */
86 @SuppressWarnings("unchecked")
87 private static Map<String, Object> toMap(final Properties props) {
88 @SuppressWarnings("rawtypes")
89 final Map map = props;
90 return map;
91 }
92
93 /** The Map decorated by this configuration. */
94 protected Map<String, Object> map;
95
96 /** A flag whether trimming of property values should be disabled. */
97 private boolean trimmingDisabled;
98
99 /**
100 * Create a Configuration decorator around the specified Map. The map is used to store the configuration properties, any
101 * change will also affect the Map.
102 *
103 * @param map the map.
104 */
105 public MapConfiguration(final Map<String, ?> map) {
106 this.map = (Map<String, Object>) Objects.requireNonNull(map, "map");
107 }
108
109 /**
110 * Creates a new instance of {@code MapConfiguration} which uses the specified {@code Properties} object as its data
111 * store. All changes of this configuration affect the given {@code Properties} object and vice versa. Note that while
112 * {@code Properties} actually implements {@code Map<Object, Object>}, we expect it to contain only string keys. Other
113 * key types will lead to {@code ClassCastException} exceptions on certain methods.
114 *
115 * @param props the {@code Properties} object defining the content of this configuration.
116 * @since 1.8
117 */
118 public MapConfiguration(final Properties props) {
119 map = toMap(Objects.requireNonNull(props));
120 }
121
122 @Override
123 protected void addPropertyDirect(final String key, final Object value) {
124 final Object previousValue = getProperty(key);
125
126 if (previousValue == null) {
127 map.put(key, value);
128 } else if (previousValue instanceof List) {
129 // the value is added to the existing list
130 // Note: This is problematic. See header comment!
131 ((List<Object>) previousValue).add(value);
132 } else {
133 // the previous value is replaced by a list containing the previous value and the new value
134 final List<Object> list = new ArrayList<>();
135 list.add(previousValue);
136 list.add(value);
137
138 map.put(key, list);
139 }
140 }
141
142 @Override
143 protected void clearPropertyDirect(final String key) {
144 map.remove(key);
145 }
146
147 /**
148 * Returns a copy of this object. The returned configuration will contain the same properties as the original. Event listeners are not cloned.
149 *
150 * @return the copy.
151 * @since 1.3
152 */
153 @Override
154 public Object clone() {
155 try {
156 final MapConfiguration copy = (MapConfiguration) super.clone();
157 copy.map = ConfigurationUtils.clone(map);
158 copy.cloneInterpolator(this);
159 return copy;
160 } catch (final CloneNotSupportedException cex) {
161 // cannot happen
162 throw new ConfigurationRuntimeException(cex);
163 }
164 }
165
166 @Override
167 protected boolean containsKeyInternal(final String key) {
168 return map.containsKey(key);
169 }
170
171 /**
172 * Tests whether this configuration contains one or more matches to this value. This operation stops at first match
173 * but may be more expensive than the containsKey method.
174 *
175 * @since 2.11.0
176 */
177 @Override
178 protected boolean containsValueInternal(final Object value) {
179 return value != null && map.containsValue(value);
180 }
181
182 @Override
183 protected Iterator<String> getKeysInternal() {
184 return map.keySet().iterator();
185 }
186
187 /**
188 * Gets the Map decorated by this configuration.
189 *
190 * @return the map this configuration is based onto.
191 */
192 public Map<String, Object> getMap() {
193 return map;
194 }
195
196 @Override
197 protected Object getPropertyInternal(final String key) {
198 final Object value = map.get(key);
199 if (value instanceof String) {
200 final Collection<String> list = getListDelimiterHandler().split((String) value, !isTrimmingDisabled());
201 return list.size() > 1 ? list : list.iterator().next();
202 }
203 return value;
204 }
205
206 @Override
207 protected boolean isEmptyInternal() {
208 return map.isEmpty();
209 }
210
211 /**
212 * Tests whether the flag whether trimming of property values is disabled.
213 *
214 * @return <strong>true</strong> if trimming of property values is disabled; <strong>false</strong> otherwise.
215 * @since 1.7
216 */
217 public boolean isTrimmingDisabled() {
218 return trimmingDisabled;
219 }
220
221 /**
222 * Sets a flag whether trimming of property values is disabled. This flag is only evaluated if list splitting is
223 * enabled. Refer to the header comment for more information about list splitting and trimming.
224 *
225 * @param trimmingDisabled a flag whether trimming of property values should be disabled.
226 * @since 1.7
227 */
228 public void setTrimmingDisabled(final boolean trimmingDisabled) {
229 this.trimmingDisabled = trimmingDisabled;
230 }
231
232 @Override
233 protected int sizeInternal() {
234 return map.size();
235 }
236
237 /**
238 * Converts this object to a String suitable for debugging and logging.
239 *
240 * @since 2.3
241 */
242 @Override
243 public String toString() {
244 return getClass().getSimpleName() + " [map=" + map + ", trimmingDisabled=" + trimmingDisabled + "]";
245 }
246 }