mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-23 16:51:35 +01:00
Configurations now properly support lists of serializable objects, and ItemStack is properly serializable. Big thanks to GICodeWarrior for the PR. This fixes BUKKIT-425
By: Nathan Adams <dinnerbone@dinnerbone.com>
This commit is contained in:
parent
768732a914
commit
cff0c0ecc5
@ -189,7 +189,7 @@ public class MemorySection implements ConfigurationSection {
|
|||||||
if (value == null) {
|
if (value == null) {
|
||||||
map.remove(key);
|
map.remove(key);
|
||||||
} else {
|
} else {
|
||||||
map.put(key, prepForStorage(value));
|
map.put(key, value);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
section.set(key, value);
|
section.set(key, value);
|
||||||
@ -905,20 +905,6 @@ public class MemorySection implements ConfigurationSection {
|
|||||||
return val instanceof ConfigurationSection;
|
return val instanceof ConfigurationSection;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Object prepForStorage(Object input) {
|
|
||||||
if (input == null) {
|
|
||||||
throw new IllegalArgumentException("Cannot store null");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isPrimitiveWrapper(input) || isNaturallyStorable(input)) {
|
|
||||||
return input;
|
|
||||||
} else if (input instanceof ConfigurationSerializable) {
|
|
||||||
return input;
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new IllegalArgumentException("Cannot store " + input + " into " + this + ", unsupported class");
|
|
||||||
}
|
|
||||||
|
|
||||||
protected boolean isPrimitiveWrapper(Object input) {
|
protected boolean isPrimitiveWrapper(Object input) {
|
||||||
return input instanceof Integer || input instanceof Boolean ||
|
return input instanceof Integer || input instanceof Boolean ||
|
||||||
input instanceof Character || input instanceof Byte ||
|
input instanceof Character || input instanceof Byte ||
|
||||||
@ -926,12 +912,6 @@ public class MemorySection implements ConfigurationSection {
|
|||||||
input instanceof Long || input instanceof Float;
|
input instanceof Long || input instanceof Float;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected boolean isNaturallyStorable(Object input) {
|
|
||||||
return input instanceof List || input instanceof Iterable ||
|
|
||||||
input instanceof String || input instanceof File ||
|
|
||||||
input instanceof Enum;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected Object getDefault(String path) {
|
protected Object getDefault(String path) {
|
||||||
if (path == null) {
|
if (path == null) {
|
||||||
throw new IllegalArgumentException("Path cannot be null");
|
throw new IllegalArgumentException("Path cannot be null");
|
||||||
|
@ -27,8 +27,8 @@ public class YamlConfiguration extends FileConfiguration {
|
|||||||
protected static final String COMMENT_PREFIX = "# ";
|
protected static final String COMMENT_PREFIX = "# ";
|
||||||
protected static final String BLANK_CONFIG = "{}\n";
|
protected static final String BLANK_CONFIG = "{}\n";
|
||||||
private final DumperOptions yamlOptions = new DumperOptions();
|
private final DumperOptions yamlOptions = new DumperOptions();
|
||||||
private final Representer yamlRepresenter = new Representer();
|
private final Representer yamlRepresenter = new YamlRepresenter();
|
||||||
private final Yaml yaml = new Yaml(new SafeConstructor(), yamlRepresenter, yamlOptions);
|
private final Yaml yaml = new Yaml(new YamlConstructor(), yamlRepresenter, yamlOptions);
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String saveToString() {
|
public String saveToString() {
|
||||||
@ -38,10 +38,8 @@ public class YamlConfiguration extends FileConfiguration {
|
|||||||
yamlOptions.setDefaultFlowStyle(DumperOptions.FlowStyle.BLOCK);
|
yamlOptions.setDefaultFlowStyle(DumperOptions.FlowStyle.BLOCK);
|
||||||
yamlRepresenter.setDefaultFlowStyle(DumperOptions.FlowStyle.BLOCK);
|
yamlRepresenter.setDefaultFlowStyle(DumperOptions.FlowStyle.BLOCK);
|
||||||
|
|
||||||
serializeValues(output, getValues(false));
|
|
||||||
|
|
||||||
String header = buildHeader();
|
String header = buildHeader();
|
||||||
String dump = yaml.dump(output);
|
String dump = yaml.dump(getValues(false));
|
||||||
|
|
||||||
if (dump.equals(BLANK_CONFIG)) {
|
if (dump.equals(BLANK_CONFIG)) {
|
||||||
dump = "";
|
dump = "";
|
||||||
@ -56,90 +54,34 @@ public class YamlConfiguration extends FileConfiguration {
|
|||||||
throw new IllegalArgumentException("Contents cannot be null");
|
throw new IllegalArgumentException("Contents cannot be null");
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
Map<Object, Object> input;
|
||||||
Map<Object, Object> input = (Map<Object, Object>) yaml.load(contents);
|
try {
|
||||||
int size = (input == null) ? 0 : input.size();
|
input = (Map<Object, Object>) yaml.load(contents);
|
||||||
Map<String, Object> result = new LinkedHashMap<String, Object>(size);
|
} catch (YAMLException e) {
|
||||||
|
throw new InvalidConfigurationException(e);
|
||||||
if (size > 0) {
|
} catch (ClassCastException e) {
|
||||||
for (Map.Entry<Object, Object> entry : input.entrySet()) {
|
throw new InvalidConfigurationException("Top level is not a Map.");
|
||||||
result.put(entry.getKey().toString(), entry.getValue());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String header = parseHeader(contents);
|
String header = parseHeader(contents);
|
||||||
|
|
||||||
if (header.length() > 0) {
|
if (header.length() > 0) {
|
||||||
options().header(header);
|
options().header(header);
|
||||||
}
|
}
|
||||||
|
|
||||||
deserializeValues(result, this);
|
if (input != null) {
|
||||||
|
convertMapsToSections(input, this);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void deserializeValues(Map<String, Object> input, ConfigurationSection section) throws InvalidConfigurationException {
|
protected void convertMapsToSections(Map<Object, Object> input, ConfigurationSection section) {
|
||||||
if (input == null) {
|
for (Map.Entry<Object, Object> entry : input.entrySet()) {
|
||||||
return;
|
String key = entry.getKey().toString();
|
||||||
}
|
|
||||||
|
|
||||||
for (Map.Entry<String, Object> entry : input.entrySet()) {
|
|
||||||
Object value = entry.getValue();
|
Object value = entry.getValue();
|
||||||
|
|
||||||
if (value instanceof Map) {
|
if (value instanceof Map<?, ?>) {
|
||||||
@SuppressWarnings("unchecked")
|
convertMapsToSections((Map<Object, Object>) value, section.createSection(key));
|
||||||
Map<Object, Object> subinput = (Map<Object, Object>) value;
|
|
||||||
int size = (subinput == null) ? 0 : subinput.size();
|
|
||||||
Map<String, Object> subvalues = new LinkedHashMap<String, Object>(size);
|
|
||||||
|
|
||||||
if (size > 0) {
|
|
||||||
for (Map.Entry<Object, Object> subentry : subinput.entrySet()) {
|
|
||||||
subvalues.put(subentry.getKey().toString(), subentry.getValue());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (subvalues.containsKey(ConfigurationSerialization.SERIALIZED_TYPE_KEY)) {
|
|
||||||
try {
|
|
||||||
ConfigurationSerializable serializable = ConfigurationSerialization.deserializeObject(subvalues);
|
|
||||||
section.set(entry.getKey(), serializable);
|
|
||||||
} catch (IllegalArgumentException ex) {
|
|
||||||
throw new InvalidConfigurationException("Could not deserialize object", ex);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
ConfigurationSection subsection = section.createSection(entry.getKey());
|
|
||||||
deserializeValues(subvalues, subsection);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
section.set(entry.getKey(), entry.getValue());
|
section.set(key, value);
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected void serializeValues(Map<String, Object> output, Map<String, Object> input) {
|
|
||||||
if (input == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (Map.Entry<String, Object> entry : input.entrySet()) {
|
|
||||||
Object value = entry.getValue();
|
|
||||||
|
|
||||||
if (value instanceof ConfigurationSection) {
|
|
||||||
ConfigurationSection subsection = (ConfigurationSection) entry.getValue();
|
|
||||||
Map<String, Object> subvalues = new LinkedHashMap<String, Object>();
|
|
||||||
|
|
||||||
serializeValues(subvalues, subsection.getValues(false));
|
|
||||||
value = subvalues;
|
|
||||||
} else if (value instanceof ConfigurationSerializable) {
|
|
||||||
ConfigurationSerializable serializable = (ConfigurationSerializable) value;
|
|
||||||
Map<String, Object> subvalues = new LinkedHashMap<String, Object>();
|
|
||||||
subvalues.put(ConfigurationSerialization.SERIALIZED_TYPE_KEY, ConfigurationSerialization.getAlias(serializable.getClass()));
|
|
||||||
|
|
||||||
serializeValues(subvalues, serializable.serialize());
|
|
||||||
value = subvalues;
|
|
||||||
} else if ((!isPrimitiveWrapper(value)) && (!isNaturallyStorable(value))) {
|
|
||||||
throw new IllegalStateException("Configuration contains non-serializable values, cannot process");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (value != null) {
|
|
||||||
output.put(entry.getKey(), value);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,47 @@
|
|||||||
|
package org.bukkit.configuration.file;
|
||||||
|
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.yaml.snakeyaml.nodes.Node;
|
||||||
|
import org.yaml.snakeyaml.constructor.SafeConstructor;
|
||||||
|
import org.yaml.snakeyaml.error.YAMLException;
|
||||||
|
import org.yaml.snakeyaml.nodes.Tag;
|
||||||
|
|
||||||
|
import org.bukkit.configuration.serialization.ConfigurationSerialization;
|
||||||
|
|
||||||
|
public class YamlConstructor extends SafeConstructor {
|
||||||
|
|
||||||
|
public YamlConstructor() {
|
||||||
|
this.yamlConstructors.put(Tag.MAP, new ConstructCustomObject());
|
||||||
|
}
|
||||||
|
|
||||||
|
private class ConstructCustomObject extends ConstructYamlMap {
|
||||||
|
public Object construct(Node node) {
|
||||||
|
if (node.isTwoStepsConstruction()) {
|
||||||
|
throw new YAMLException("Unexpected referential mapping structure. Node: " + node);
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<Object, Object> raw = (Map<Object, Object>) super.construct(node);
|
||||||
|
|
||||||
|
if (raw.containsKey(ConfigurationSerialization.SERIALIZED_TYPE_KEY)) {
|
||||||
|
Map<String, Object> typed = new LinkedHashMap<String, Object>(raw.size());
|
||||||
|
for (Map.Entry<Object, Object> entry : raw.entrySet()) {
|
||||||
|
typed.put(entry.getKey().toString(), entry.getValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
return ConfigurationSerialization.deserializeObject(typed);
|
||||||
|
} catch (IllegalArgumentException ex) {
|
||||||
|
throw new YAMLException("Could not deserialize object", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return raw;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void construct2ndStep(Node node, Object object) {
|
||||||
|
throw new YAMLException("Unexpected referential mapping structure. Node: " + node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,40 @@
|
|||||||
|
|
||||||
|
package org.bukkit.configuration.file;
|
||||||
|
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.bukkit.configuration.ConfigurationSection;
|
||||||
|
import org.bukkit.configuration.serialization.ConfigurationSerializable;
|
||||||
|
import org.bukkit.configuration.serialization.ConfigurationSerialization;
|
||||||
|
|
||||||
|
import org.yaml.snakeyaml.nodes.Node;
|
||||||
|
import org.yaml.snakeyaml.representer.Represent;
|
||||||
|
import org.yaml.snakeyaml.representer.Representer;
|
||||||
|
|
||||||
|
public class YamlRepresenter extends Representer {
|
||||||
|
|
||||||
|
public YamlRepresenter() {
|
||||||
|
this.multiRepresenters.put(ConfigurationSection.class, new RepresentConfigurationSection());
|
||||||
|
this.multiRepresenters.put(ConfigurationSerializable.class, new RepresentConfigurationSerializable());
|
||||||
|
}
|
||||||
|
|
||||||
|
private class RepresentConfigurationSection extends RepresentMap {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
public Node representData(Object data) {
|
||||||
|
return super.representData(((ConfigurationSection) data).getValues(false));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class RepresentConfigurationSerializable extends RepresentMap {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
public Node representData(Object data) {
|
||||||
|
ConfigurationSerializable serializable = (ConfigurationSerializable) data;
|
||||||
|
Map<String, Object> values = new LinkedHashMap<String, Object>();
|
||||||
|
values.put(ConfigurationSerialization.SERIALIZED_TYPE_KEY, ConfigurationSerialization.getAlias(serializable.getClass()));
|
||||||
|
values.putAll(serializable.serialize());
|
||||||
|
|
||||||
|
return super.representData(values);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -7,6 +7,7 @@ import java.util.HashMap;
|
|||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.logging.Level;
|
import java.util.logging.Level;
|
||||||
import java.util.logging.Logger;
|
import java.util.logging.Logger;
|
||||||
|
import org.bukkit.inventory.ItemStack;
|
||||||
import org.bukkit.util.BlockVector;
|
import org.bukkit.util.BlockVector;
|
||||||
import org.bukkit.util.Vector;
|
import org.bukkit.util.Vector;
|
||||||
|
|
||||||
@ -21,6 +22,7 @@ public class ConfigurationSerialization {
|
|||||||
static {
|
static {
|
||||||
registerClass(Vector.class);
|
registerClass(Vector.class);
|
||||||
registerClass(BlockVector.class);
|
registerClass(BlockVector.class);
|
||||||
|
registerClass(ItemStack.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected ConfigurationSerialization(Class<? extends ConfigurationSerializable> clazz) {
|
protected ConfigurationSerialization(Class<? extends ConfigurationSerializable> clazz) {
|
||||||
@ -250,4 +252,4 @@ public class ConfigurationSerialization {
|
|||||||
|
|
||||||
return clazz.getName();
|
return clazz.getName();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -329,7 +329,7 @@ public class ItemStack implements ConfigurationSerializable {
|
|||||||
public Map<String, Object> serialize() {
|
public Map<String, Object> serialize() {
|
||||||
Map<String, Object> result = new LinkedHashMap<String, Object>();
|
Map<String, Object> result = new LinkedHashMap<String, Object>();
|
||||||
|
|
||||||
result.put("type", getType());
|
result.put("type", getType().name());
|
||||||
|
|
||||||
if (durability != 0) {
|
if (durability != 0) {
|
||||||
result.put("damage", durability);
|
result.put("damage", durability);
|
||||||
@ -356,18 +356,18 @@ public class ItemStack implements ConfigurationSerializable {
|
|||||||
|
|
||||||
public static ItemStack deserialize(Map<String, Object> args) {
|
public static ItemStack deserialize(Map<String, Object> args) {
|
||||||
Material type = Material.getMaterial((String) args.get("type"));
|
Material type = Material.getMaterial((String) args.get("type"));
|
||||||
short damage = 0;
|
int damage = 0;
|
||||||
int amount = 1;
|
int amount = 1;
|
||||||
|
|
||||||
if (args.containsKey("damage")) {
|
if (args.containsKey("damage")) {
|
||||||
damage = (Short) args.get("damage");
|
damage = (Integer) args.get("damage");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (args.containsKey("amount")) {
|
if (args.containsKey("amount")) {
|
||||||
amount = (Integer) args.get("amount");
|
amount = (Integer) args.get("amount");
|
||||||
}
|
}
|
||||||
|
|
||||||
ItemStack result = new ItemStack(type, amount, damage);
|
ItemStack result = new ItemStack(type, amount, (short) damage);
|
||||||
|
|
||||||
if (args.containsKey("enchantments")) {
|
if (args.containsKey("enchantments")) {
|
||||||
Object raw = args.get("enchantments");
|
Object raw = args.get("enchantments");
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
package org.bukkit.configuration.file;
|
package org.bukkit.configuration.file;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.bukkit.configuration.InvalidConfigurationException;
|
||||||
|
import org.bukkit.configuration.serialization.ConfigurationSerialization;
|
||||||
|
import org.bukkit.inventory.ItemStack;
|
||||||
|
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
@ -52,4 +59,26 @@ public class YamlConfigurationTest extends FileConfigurationTest {
|
|||||||
|
|
||||||
assertEquals(expected, result);
|
assertEquals(expected, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSaveRestoreCompositeList() throws InvalidConfigurationException {
|
||||||
|
YamlConfiguration out = getConfig();
|
||||||
|
|
||||||
|
List<ItemStack> stacks = new ArrayList<ItemStack>();
|
||||||
|
stacks.add(new ItemStack(1));
|
||||||
|
stacks.add(new ItemStack(2));
|
||||||
|
stacks.add(new ItemStack(3));
|
||||||
|
|
||||||
|
out.set("composite-list.abc.def", stacks);
|
||||||
|
String yaml = out.saveToString();
|
||||||
|
|
||||||
|
YamlConfiguration in = new YamlConfiguration();
|
||||||
|
in.loadFromString(yaml);
|
||||||
|
List<Object> raw = in.getList("composite-list.abc.def");
|
||||||
|
|
||||||
|
assertEquals(stacks.size(), raw.size());
|
||||||
|
assertEquals(stacks.get(0), raw.get(0));
|
||||||
|
assertEquals(stacks.get(1), raw.get(1));
|
||||||
|
assertEquals(stacks.get(2), raw.get(2));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user