Test PluginManager.removePermission

Static methods are death to testability.  However, irrelevant static
methods can be negotiated with until a later time in which they can be
removed.  When instantiating a new Permission object, static calls are
made to the Bukkit class during a recalculatePermissibles logic path.
This recalculatePermissibles call should probably be moved
appropriately, but until the time such testing can be accomplished
itself, these tests work around that situation by simply verifying the
static Bukkit server references are satisfied since what is called as
a result is irrelevant currently.

This commit also updates a few other tests for PluginManagerTest to
work towards the standard of using the Hamcrest unit testing library.

By: EdGruberman <ed@rjump.com>
This commit is contained in:
Bukkit/Spigot 2013-03-11 01:39:14 -07:00
parent d2a44f240c
commit fce9b199e3
2 changed files with 129 additions and 25 deletions

View file

@ -1,41 +1,93 @@
package org.bukkit; package org.bukkit;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationHandler; import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method; import java.lang.reflect.Method;
import java.lang.reflect.Proxy; import java.lang.reflect.Proxy;
import java.util.HashMap; import java.util.Map;
import java.util.logging.Logger;
import org.bukkit.command.SimpleCommandMap;
import org.bukkit.plugin.PluginManager;
import org.bukkit.plugin.SimplePluginManager;
import com.google.common.collect.ImmutableMap;
public class TestServer implements InvocationHandler { public class TestServer implements InvocationHandler {
private static interface MethodHandler { private static interface MethodHandler {
Object handle(TestServer server, Object[] args); Object handle(TestServer server, Object[] args);
} }
private static final Constructor<? extends Server> constructor;
private static final HashMap<Method, MethodHandler> methods = new HashMap<Method, MethodHandler>(); private static final Map<Method, MethodHandler> methods;
static { static {
try { try {
methods.put(Server.class.getMethod("isPrimaryThread"), ImmutableMap.Builder<Method, MethodHandler> methodMap = ImmutableMap.builder();
new MethodHandler() { methodMap.put(
public Object handle(TestServer server, Object[] args) { Server.class.getMethod("isPrimaryThread"),
return Thread.currentThread().equals(server.creatingThread); new MethodHandler() {
public Object handle(TestServer server, Object[] args) {
return Thread.currentThread().equals(server.creatingThread);
}
} }
}); );
constructor = Proxy.getProxyClass(Server.class.getClassLoader(), Server.class).asSubclass(Server.class).getConstructor(InvocationHandler.class); methodMap.put(
Server.class.getMethod("getPluginManager"),
new MethodHandler() {
public Object handle(TestServer server, Object[] args) {
return server.pluginManager;
}
}
);
methodMap.put(
Server.class.getMethod("getLogger"),
new MethodHandler() {
final Logger logger = Logger.getLogger(TestServer.class.getCanonicalName());
public Object handle(TestServer server, Object[] args) {
return logger;
}
}
);
methodMap.put(
Server.class.getMethod("getName"),
new MethodHandler() {
public Object handle(TestServer server, Object[] args) {
return TestServer.class.getSimpleName();
}
}
);
methodMap.put(
Server.class.getMethod("getVersion"),
new MethodHandler() {
public Object handle(TestServer server, Object[] args) {
return "Version_" + TestServer.class.getPackage().getImplementationVersion();
}
}
);
methodMap.put(
Server.class.getMethod("getBukkitVersion"),
new MethodHandler() {
public Object handle(TestServer server, Object[] args) {
return "BukkitVersion_" + TestServer.class.getPackage().getImplementationVersion();
}
}
);
methods = methodMap.build();
TestServer server = new TestServer();
Server instance = Proxy.getProxyClass(Server.class.getClassLoader(), Server.class).asSubclass(Server.class).getConstructor(InvocationHandler.class).newInstance(server);
Bukkit.setServer(instance);
server.pluginManager = new SimplePluginManager(instance, new SimpleCommandMap(instance));
} catch (Throwable t) { } catch (Throwable t) {
throw new Error(t); throw new Error(t);
} }
} }
private Thread creatingThread = Thread.currentThread(); private Thread creatingThread = Thread.currentThread();
private PluginManager pluginManager;
private TestServer() {}; private TestServer() {};
public static Server getInstance() { public static Server getInstance() {
try { return Bukkit.getServer();
return constructor.newInstance(new TestServer());
} catch (Throwable t) {
throw new RuntimeException(t);
}
} }
public Object invoke(Object proxy, Method method, Object[] args) { public Object invoke(Object proxy, Method method, Object[] args) {

View file

@ -1,12 +1,14 @@
package org.bukkit.plugin; package org.bukkit.plugin;
import static junit.framework.Assert.*; import static org.hamcrest.Matchers.*;
import static org.junit.Assert.*;
import org.bukkit.Server;
import org.bukkit.TestServer; import org.bukkit.TestServer;
import org.bukkit.command.SimpleCommandMap;
import org.bukkit.event.Event; import org.bukkit.event.Event;
import org.bukkit.event.TestEvent; import org.bukkit.event.TestEvent;
import org.bukkit.permissions.Permission;
import org.junit.After;
import org.junit.Test; import org.junit.Test;
public class PluginManagerTest { public class PluginManagerTest {
@ -14,10 +16,9 @@ public class PluginManagerTest {
volatile Object value = null; volatile Object value = null;
} }
final Server server = TestServer.getInstance(); private static final PluginManager pm = TestServer.getInstance().getPluginManager();
final SimpleCommandMap commandMap = new SimpleCommandMap(server);
final PluginManager pm = new SimplePluginManager(server, commandMap); private final MutableObject store = new MutableObject();
final MutableObject store = new MutableObject();
@Test @Test
public void testAsyncSameThread() { public void testAsyncSameThread() {
@ -25,7 +26,7 @@ public class PluginManagerTest {
try { try {
pm.callEvent(event); pm.callEvent(event);
} catch (IllegalStateException ex) { } catch (IllegalStateException ex) {
assertEquals(event.getEventName() + " cannot be triggered asynchronously from primary server thread.", ex.getMessage()); assertThat(event.getEventName() + " cannot be triggered asynchronously from primary server thread.", is(ex.getMessage()));
return; return;
} }
throw new IllegalStateException("No exception thrown"); throw new IllegalStateException("No exception thrown");
@ -53,8 +54,8 @@ public class PluginManagerTest {
}}); }});
secondThread.start(); secondThread.start();
secondThread.join(); secondThread.join();
assertTrue(store.value instanceof IllegalStateException); assertThat(store.value, is(instanceOf(IllegalStateException.class)));
assertEquals(event.getEventName() + " cannot be triggered asynchronously from inside synchronized code.", ((Throwable) store.value).getMessage()); assertThat(event.getEventName() + " cannot be triggered asynchronously from inside synchronized code.", is(((Throwable) store.value).getMessage()));
} }
@Test @Test
@ -115,4 +116,55 @@ public class PluginManagerTest {
throw new RuntimeException((Throwable) store.value); throw new RuntimeException((Throwable) store.value);
} }
} }
@Test
public void testRemovePermissionByNameLower() {
this.testRemovePermissionByName("lower");
}
@Test
public void testRemovePermissionByNameUpper() {
this.testRemovePermissionByName("UPPER");
}
@Test
public void testRemovePermissionByNameCamel() {
this.testRemovePermissionByName("CaMeL");
}
public void testRemovePermissionByPermissionLower() {
this.testRemovePermissionByPermission("lower");
}
@Test
public void testRemovePermissionByPermissionUpper() {
this.testRemovePermissionByPermission("UPPER");
}
@Test
public void testRemovePermissionByPermissionCamel() {
this.testRemovePermissionByPermission("CaMeL");
}
private void testRemovePermissionByName(final String name) {
final Permission perm = new Permission(name);
pm.addPermission(perm);
assertThat("Permission \"" + name + "\" was not added", pm.getPermission(name), is(perm));
pm.removePermission(name);
assertThat("Permission \"" + name + "\" was not removed", pm.getPermission(name), is(nullValue()));
}
private void testRemovePermissionByPermission(final String name) {
final Permission perm = new Permission(name);
pm.addPermission(perm);
assertThat("Permission \"" + name + "\" was not added", pm.getPermission(name), is(perm));
pm.removePermission(perm);
assertThat("Permission \"" + name + "\" was not removed", pm.getPermission(name), is(nullValue()));
}
@After
public void tearDown() {
pm.clearPlugins();
assertThat(pm.getPermissions(), is(empty()));
}
} }