mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-24 09:11:26 +01:00
Refactor common metadata code into base class. Fixes BUKKIT-3624
Implementing the MetadataValue interface is significant work due to having to provide a large amount of conversion stub methods. This commit adds a new optional abstract base class to aid in implementation. By: crast <contact@jamescrasta.com>
This commit is contained in:
parent
0a419b9e4d
commit
79f657b1a7
@ -5,7 +5,6 @@ import java.util.concurrent.Callable;
|
|||||||
|
|
||||||
import org.apache.commons.lang.Validate;
|
import org.apache.commons.lang.Validate;
|
||||||
import org.bukkit.plugin.Plugin;
|
import org.bukkit.plugin.Plugin;
|
||||||
import org.bukkit.util.NumberConversions;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The LazyMetadataValue class implements a type of metadata that is not computed until another plugin asks for it.
|
* The LazyMetadataValue class implements a type of metadata that is not computed until another plugin asks for it.
|
||||||
@ -14,11 +13,10 @@ import org.bukkit.util.NumberConversions;
|
|||||||
* or invalidated at the individual or plugin level. Once invalidated, the LazyMetadataValue will recompute its value
|
* or invalidated at the individual or plugin level. Once invalidated, the LazyMetadataValue will recompute its value
|
||||||
* when asked.
|
* when asked.
|
||||||
*/
|
*/
|
||||||
public class LazyMetadataValue implements MetadataValue {
|
public class LazyMetadataValue extends MetadataValueAdapter implements MetadataValue {
|
||||||
private Callable<Object> lazyValue;
|
private Callable<Object> lazyValue;
|
||||||
private CacheStrategy cacheStrategy;
|
private CacheStrategy cacheStrategy;
|
||||||
private SoftReference<Object> internalValue = new SoftReference<Object>(null);
|
private SoftReference<Object> internalValue = new SoftReference<Object>(null);
|
||||||
private Plugin owningPlugin;
|
|
||||||
private static final Object ACTUALLY_NULL = new Object();
|
private static final Object ACTUALLY_NULL = new Object();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -39,19 +37,14 @@ public class LazyMetadataValue implements MetadataValue {
|
|||||||
* @param lazyValue the lazy value assigned to this metadata value.
|
* @param lazyValue the lazy value assigned to this metadata value.
|
||||||
*/
|
*/
|
||||||
public LazyMetadataValue(Plugin owningPlugin, CacheStrategy cacheStrategy, Callable<Object> lazyValue) {
|
public LazyMetadataValue(Plugin owningPlugin, CacheStrategy cacheStrategy, Callable<Object> lazyValue) {
|
||||||
Validate.notNull(owningPlugin, "owningPlugin cannot be null");
|
super(owningPlugin);
|
||||||
Validate.notNull(cacheStrategy, "cacheStrategy cannot be null");
|
Validate.notNull(cacheStrategy, "cacheStrategy cannot be null");
|
||||||
Validate.notNull(lazyValue, "lazyValue cannot be null");
|
Validate.notNull(lazyValue, "lazyValue cannot be null");
|
||||||
|
|
||||||
this.lazyValue = lazyValue;
|
this.lazyValue = lazyValue;
|
||||||
this.owningPlugin = owningPlugin;
|
|
||||||
this.cacheStrategy = cacheStrategy;
|
this.cacheStrategy = cacheStrategy;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Plugin getOwningPlugin() {
|
|
||||||
return owningPlugin;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Object value() {
|
public Object value() {
|
||||||
eval();
|
eval();
|
||||||
Object value = internalValue.get();
|
Object value = internalValue.get();
|
||||||
@ -61,56 +54,6 @@ public class LazyMetadataValue implements MetadataValue {
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public int asInt() {
|
|
||||||
return NumberConversions.toInt(value());
|
|
||||||
}
|
|
||||||
|
|
||||||
public float asFloat() {
|
|
||||||
return NumberConversions.toFloat(value());
|
|
||||||
}
|
|
||||||
|
|
||||||
public double asDouble() {
|
|
||||||
return NumberConversions.toDouble(value());
|
|
||||||
}
|
|
||||||
|
|
||||||
public long asLong() {
|
|
||||||
return NumberConversions.toLong(value());
|
|
||||||
}
|
|
||||||
|
|
||||||
public short asShort() {
|
|
||||||
return NumberConversions.toShort(value());
|
|
||||||
}
|
|
||||||
|
|
||||||
public byte asByte() {
|
|
||||||
return NumberConversions.toByte(value());
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean asBoolean() {
|
|
||||||
Object value = value();
|
|
||||||
if (value instanceof Boolean) {
|
|
||||||
return (Boolean) value;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (value instanceof Number) {
|
|
||||||
return ((Number) value).intValue() != 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (value instanceof String) {
|
|
||||||
return Boolean.parseBoolean((String) value);
|
|
||||||
}
|
|
||||||
|
|
||||||
return value != null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String asString() {
|
|
||||||
Object value = value();
|
|
||||||
|
|
||||||
if (value == null) {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
return value.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Lazily evaluates the value of this metadata item.
|
* Lazily evaluates the value of this metadata item.
|
||||||
*
|
*
|
||||||
|
@ -0,0 +1,77 @@
|
|||||||
|
package org.bukkit.metadata;
|
||||||
|
|
||||||
|
import org.apache.commons.lang.Validate;
|
||||||
|
import org.bukkit.plugin.Plugin;
|
||||||
|
import org.bukkit.util.NumberConversions;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optional base class for facilitating MetadataValue implementations.
|
||||||
|
*
|
||||||
|
* This provides all the conversion functions for MetadataValue
|
||||||
|
* so that writing an implementation of MetadataValue is as simple
|
||||||
|
* as implementing value() and invalidate().
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public abstract class MetadataValueAdapter implements MetadataValue {
|
||||||
|
protected final Plugin owningPlugin;
|
||||||
|
|
||||||
|
protected MetadataValueAdapter(Plugin owningPlugin) {
|
||||||
|
Validate.notNull(owningPlugin, "owningPlugin cannot be null");
|
||||||
|
this.owningPlugin = owningPlugin;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Plugin getOwningPlugin() {
|
||||||
|
return owningPlugin;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int asInt() {
|
||||||
|
return NumberConversions.toInt(value());
|
||||||
|
}
|
||||||
|
|
||||||
|
public float asFloat() {
|
||||||
|
return NumberConversions.toFloat(value());
|
||||||
|
}
|
||||||
|
|
||||||
|
public double asDouble() {
|
||||||
|
return NumberConversions.toDouble(value());
|
||||||
|
}
|
||||||
|
|
||||||
|
public long asLong() {
|
||||||
|
return NumberConversions.toLong(value());
|
||||||
|
}
|
||||||
|
|
||||||
|
public short asShort() {
|
||||||
|
return NumberConversions.toShort(value());
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte asByte() {
|
||||||
|
return NumberConversions.toByte(value());
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean asBoolean() {
|
||||||
|
Object value = value();
|
||||||
|
if (value instanceof Boolean) {
|
||||||
|
return (Boolean) value;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value instanceof Number) {
|
||||||
|
return ((Number) value).intValue() != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value instanceof String) {
|
||||||
|
return Boolean.parseBoolean((String) value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return value != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String asString() {
|
||||||
|
Object value = value();
|
||||||
|
|
||||||
|
if (value == null) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,97 @@
|
|||||||
|
package org.bukkit.metadata;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
import org.bukkit.plugin.Plugin;
|
||||||
|
import org.bukkit.plugin.TestPlugin;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class MetadataValueAdapterTest {
|
||||||
|
private TestPlugin plugin = new TestPlugin("x");
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAdapterBasics() {
|
||||||
|
IncrementingMetaValue mv = new IncrementingMetaValue(plugin);
|
||||||
|
// check getOwningPlugin
|
||||||
|
assertEquals(mv.getOwningPlugin(), this.plugin);
|
||||||
|
|
||||||
|
// Check value-getting and invalidation.
|
||||||
|
assertEquals(new Integer(1), mv.value());
|
||||||
|
assertEquals(new Integer(2), mv.value());
|
||||||
|
mv.invalidate();
|
||||||
|
assertEquals(new Integer(1), mv.value());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAdapterConversions() {
|
||||||
|
IncrementingMetaValue mv = new IncrementingMetaValue(plugin);
|
||||||
|
|
||||||
|
assertEquals(1, mv.asInt());
|
||||||
|
assertEquals(2L, mv.asLong());
|
||||||
|
assertEquals(3.0, mv.asFloat(), 0.001);
|
||||||
|
assertEquals(4, mv.asByte());
|
||||||
|
assertEquals(5.0, mv.asDouble(), 0.001);
|
||||||
|
assertEquals(6, mv.asShort());
|
||||||
|
assertEquals("7", mv.asString());
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Boolean conversion is non-trivial, we want to test it thoroughly. */
|
||||||
|
@Test
|
||||||
|
public void testBooleanConversion() {
|
||||||
|
// null is False.
|
||||||
|
assertEquals(false, simpleValue(null).asBoolean());
|
||||||
|
|
||||||
|
// String to boolean.
|
||||||
|
assertEquals(true, simpleValue("True").asBoolean());
|
||||||
|
assertEquals(true, simpleValue("TRUE").asBoolean());
|
||||||
|
assertEquals(false, simpleValue("false").asBoolean());
|
||||||
|
|
||||||
|
// Number to boolean.
|
||||||
|
assertEquals(true, simpleValue(1).asBoolean());
|
||||||
|
assertEquals(true, simpleValue(5.0).asBoolean());
|
||||||
|
assertEquals(false, simpleValue(0).asBoolean());
|
||||||
|
assertEquals(false, simpleValue(0.1).asBoolean());
|
||||||
|
|
||||||
|
// Boolean as boolean, of course.
|
||||||
|
assertEquals(true, simpleValue(Boolean.TRUE).asBoolean());
|
||||||
|
assertEquals(false, simpleValue(Boolean.FALSE).asBoolean());
|
||||||
|
|
||||||
|
// any object that is not null and not a Boolean, String, or Number is true.
|
||||||
|
assertEquals(true, simpleValue(new Object()).asBoolean());
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Test String conversions return an empty string when given null. */
|
||||||
|
@Test
|
||||||
|
public void testStringConversionNull() {
|
||||||
|
assertEquals("", simpleValue(null).asString());
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Get a fixed value MetadataValue. */
|
||||||
|
private MetadataValue simpleValue(Object value) {
|
||||||
|
return new FixedMetadataValue(plugin, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A sample non-trivial MetadataValueAdapter implementation.
|
||||||
|
*
|
||||||
|
* The rationale for implementing an incrementing value is to have a value
|
||||||
|
* which changes with every call to value(). This is important for testing
|
||||||
|
* because we want to make sure all the tested conversions are calling the
|
||||||
|
* value() method exactly once and no caching is going on.
|
||||||
|
*/
|
||||||
|
class IncrementingMetaValue extends MetadataValueAdapter {
|
||||||
|
private int internalValue = 0;
|
||||||
|
|
||||||
|
protected IncrementingMetaValue(Plugin owningPlugin) {
|
||||||
|
super(owningPlugin);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Object value() {
|
||||||
|
return ++internalValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void invalidate() {
|
||||||
|
internalValue = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user