PropertiesConverter.java
/*
* Copyright (C) 2004, 2005 Joe Walnes.
* Copyright (C) 2006, 2007, 2008, 2009, 2013, 2014, 2015, 2020 XStream Committers.
* All rights reserved.
*
* The software in this package is published under the terms of the BSD
* style license a copy of which has been included with this distribution in
* the LICENSE.txt file.
*
* Created on 23. February 2004 by Joe Walnes
*/
package com.thoughtworks.xstream.converters.collections;
import java.lang.reflect.Field;
import java.util.Map;
import java.util.Properties;
import java.util.TreeMap;
import com.thoughtworks.xstream.converters.Converter;
import com.thoughtworks.xstream.converters.MarshallingContext;
import com.thoughtworks.xstream.converters.UnmarshallingContext;
import com.thoughtworks.xstream.core.util.Fields;
import com.thoughtworks.xstream.io.HierarchicalStreamReader;
import com.thoughtworks.xstream.io.HierarchicalStreamWriter;
/**
* Special converter for {@link Properties} that stores properties in a more compact form than java.util.Map.
* <p>
* Because all entries of a Properties instance are Strings, a single element is used for each property with two
* attributes; one for key and one for value.
* </p>
* <p>
* Additionally, default properties are also serialized, if they are present or if a SecurityManager is set, and it has
* permissions for SecurityManager.checkPackageAccess, SecurityManager.checkMemberAccess(this, EnumSet.MEMBER) and
* ReflectPermission("suppressAccessChecks").
* </p>
*
* @author Joe Walnes
* @author Kevin Ring
*/
public class PropertiesConverter implements Converter {
private final boolean sort;
public PropertiesConverter() {
this(false);
}
public PropertiesConverter(final boolean sort) {
this.sort = sort;
}
@Override
public boolean canConvert(final Class<?> type) {
return Properties.class == type;
}
@Override
public void marshal(final Object source, final HierarchicalStreamWriter writer, final MarshallingContext context) {
final Properties properties = (Properties)source;
final Map<Object, Object> map = sort ? new TreeMap<>(properties) : properties;
for (final Map.Entry<Object, Object> entry : map.entrySet()) {
writer.startNode("property");
writer.addAttribute("name", entry.getKey().toString());
writer.addAttribute("value", entry.getValue().toString());
writer.endNode();
}
if (Reflections.defaultsField != null) {
final Properties defaults = (Properties)Fields.read(Reflections.defaultsField, properties);
if (defaults != null) {
writer.startNode("defaults");
marshal(defaults, writer, context);
writer.endNode();
}
}
}
@Override
public Object unmarshal(final HierarchicalStreamReader reader, final UnmarshallingContext context) {
final Properties properties = new Properties();
Properties defaults = null;
while (reader.hasMoreChildren()) {
reader.moveDown();
if (reader.getNodeName().equals("defaults")) {
defaults = (Properties)unmarshal(reader, context);
} else {
final String name = reader.getAttribute("name");
final String value = reader.getAttribute("value");
properties.setProperty(name, value);
}
reader.moveUp();
}
if (defaults == null) {
return properties;
} else {
final Properties propertiesWithDefaults = new Properties(defaults);
propertiesWithDefaults.putAll(properties);
return propertiesWithDefaults;
}
}
private static class Reflections {
private final static Field defaultsField = Fields.locate(Properties.class, Properties.class, false);
}
}