Skip to content

Commit

Permalink
Refactor common metadata code into base class. Fixes BUKKIT-3624
Browse files Browse the repository at this point in the history
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.
  • Loading branch information
crast authored and amaranth committed Apr 4, 2013
1 parent 6fb1647 commit 4d2e3d7
Show file tree
Hide file tree
Showing 3 changed files with 176 additions and 59 deletions.
61 changes: 2 additions & 59 deletions src/main/java/org/bukkit/metadata/LazyMetadataValue.java
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@

import org.apache.commons.lang.Validate;
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.
Expand All @@ -14,11 +13,10 @@
* or invalidated at the individual or plugin level. Once invalidated, the LazyMetadataValue will recompute its value
* when asked.
*/
public class LazyMetadataValue implements MetadataValue {
public class LazyMetadataValue extends MetadataValueAdapter implements MetadataValue {
private Callable<Object> lazyValue;
private CacheStrategy cacheStrategy;
private SoftReference<Object> internalValue = new SoftReference<Object>(null);
private Plugin owningPlugin;
private static final Object ACTUALLY_NULL = new Object();

/**
Expand All @@ -39,19 +37,14 @@ public LazyMetadataValue(Plugin owningPlugin, Callable<Object> lazyValue) {
* @param lazyValue the lazy value assigned to this metadata value.
*/
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(lazyValue, "lazyValue cannot be null");

this.lazyValue = lazyValue;
this.owningPlugin = owningPlugin;
this.cacheStrategy = cacheStrategy;
}

public Plugin getOwningPlugin() {
return owningPlugin;
}

public Object value() {
eval();
Object value = internalValue.get();
Expand All @@ -61,56 +54,6 @@ public Object 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.
*
Expand Down
77 changes: 77 additions & 0 deletions src/main/java/org/bukkit/metadata/MetadataValueAdapter.java
Original file line number Diff line number Diff line change
@@ -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();
}

}
97 changes: 97 additions & 0 deletions src/test/java/org/bukkit/metadata/MetadataValueAdapterTest.java
Original file line number Diff line number Diff line change
@@ -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;
}
}
}

0 comments on commit 4d2e3d7

Please sign in to comment.