Create Wrapper for instances / revise MockUtils

- Add test resources folder
- Create basic test for Messages (todo: add concrete tests)
- Create WrapperMock
- Change UtilsTest (todo: make it work)
This commit is contained in:
ljacqu 2015-11-23 23:25:03 +01:00
parent 8719cce334
commit 6422f90114
11 changed files with 261 additions and 51 deletions

View File

@ -90,6 +90,11 @@
</includes> </includes>
</resource> </resource>
</resources> </resources>
<testResources>
<testResource>
<directory>src/test/resources</directory>
</testResource>
</testResources>
<plugins> <plugins>
<plugin> <plugin>

View File

@ -462,7 +462,7 @@ public class AuthMe extends JavaPlugin {
private void setupConsoleFilter() { private void setupConsoleFilter() {
if (Settings.removePassword) { if (Settings.removePassword) {
ConsoleFilter filter = new ConsoleFilter(); ConsoleFilter filter = new ConsoleFilter();
ConsoleLogger.getLogger().setFilter(filter); getLogger().setFilter(filter);
Bukkit.getLogger().setFilter(filter); Bukkit.getLogger().setFilter(filter);
Logger.getLogger("Minecraft").setFilter(filter); Logger.getLogger("Minecraft").setFilter(filter);
// Set Log4J Filter // Set Log4J Filter
@ -955,20 +955,10 @@ public class AuthMe extends JavaPlugin {
} }
/** /**
* Returns the management instance. * Return the management instance.
*/ */
public Management getManagement() { public Management getManagement() {
return management; return management;
} }
/**
* Returns the server instance running this plugin. Use this method in favor of {@link
* AuthMe#getServer()} for testability purposes.
*
* @return the server instance
*/
public Server getGameServer() {
return super.getServer();
}
} }

View File

@ -3,6 +3,7 @@ package fr.xephi.authme;
import com.google.common.base.Throwables; import com.google.common.base.Throwables;
import fr.xephi.authme.settings.Settings; import fr.xephi.authme.settings.Settings;
import fr.xephi.authme.util.StringUtils; import fr.xephi.authme.util.StringUtils;
import fr.xephi.authme.util.Wrapper;
import java.io.IOException; import java.io.IOException;
import java.nio.file.Files; import java.nio.file.Files;
@ -10,23 +11,17 @@ import java.nio.file.StandardOpenOption;
import java.text.DateFormat; import java.text.DateFormat;
import java.text.SimpleDateFormat; import java.text.SimpleDateFormat;
import java.util.Date; import java.util.Date;
import java.util.logging.Logger;
/** /**
* The plugin's static logger. * The plugin's static logger.
*/ */
public class ConsoleLogger { public final class ConsoleLogger {
private static final Logger log = AuthMe.getInstance().getLogger(); private static Wrapper wrapper = new Wrapper(AuthMe.getInstance());
private static final DateFormat df = new SimpleDateFormat("[MM-dd HH:mm:ss]"); private static final DateFormat df = new SimpleDateFormat("[MM-dd HH:mm:ss]");
/** private ConsoleLogger() {
* Returns the plugin's logger. // Service class
*
* @return Logger
*/
public static Logger getLogger() {
return log;
} }
/** /**
@ -35,7 +30,7 @@ public class ConsoleLogger {
* @param message String * @param message String
*/ */
public static void info(String message) { public static void info(String message) {
log.info(message); wrapper.getLogger().info(message);
if (!Settings.useLogging) { if (!Settings.useLogging) {
return; return;
} }
@ -48,7 +43,7 @@ public class ConsoleLogger {
* @param message String * @param message String
*/ */
public static void showError(String message) { public static void showError(String message) {
log.warning(message); wrapper.getLogger().warning(message);
if (!Settings.useLogging) { if (!Settings.useLogging) {
return; return;
} }

View File

@ -12,15 +12,15 @@ import java.util.zip.GZIPInputStream;
public class GeoLiteAPI { public class GeoLiteAPI {
private static final String GEOIP_URL = "http://geolite.maxmind.com/download/geoip/database/GeoLiteCountry" + private static final String GEOIP_URL = "http://geolite.maxmind.com/download/geoip/database/GeoLiteCountry"
"/GeoIP.dat.gz"; + "/GeoIP.dat.gz";
private static final AuthMe plugin = AuthMe.getInstance(); private static final Wrapper wrapper = new Wrapper(AuthMe.getInstance());
private static LookupService lookupService; private static LookupService lookupService;
/** /**
* Download (if absent) the GeoIpLite data file and then try to load it. * Download (if absent) the GeoIpLite data file and then try to load it.
* *
* @return Boolean True if the data is available, false if not. * @return True if the data is available, false otherwise.
*/ */
public static boolean isDataAvailable() { public static boolean isDataAvailable() {
if (lookupService != null) { if (lookupService != null) {
@ -30,15 +30,17 @@ public class GeoLiteAPI {
if (data.exists()) { if (data.exists()) {
try { try {
lookupService = new LookupService(data); lookupService = new LookupService(data);
plugin.getLogger().info("[LICENSE] This product uses data from the GeoLite API created by MaxMind, " + // TODO ljacqu 20151123: Should this not be output over the ConsoleLogger service?
wrapper.getLogger().info("[LICENSE] This product uses data from the GeoLite API created by MaxMind, " +
"available at http://www.maxmind.com"); "available at http://www.maxmind.com");
return true; return true;
} catch (IOException e) { } catch (IOException e) {
// TODO ljacqu 20151123: Log the exception instead of just swallowing it
return false; return false;
} }
} }
// Ok, let's try to download the data file! // Ok, let's try to download the data file!
plugin.getGameServer().getScheduler().runTaskAsynchronously(plugin, new Runnable() { wrapper.getServer().getScheduler().runTaskAsynchronously(wrapper.getAuthMe(), new Runnable() {
@Override @Override
public void run() { public void run() {
try { try {

View File

@ -31,7 +31,8 @@ import java.util.zip.GZIPInputStream;
*/ */
public final class Utils { public final class Utils {
public static AuthMe plugin; private static AuthMe plugin;
private static Wrapper wrapper;
private static boolean getOnlinePlayersIsCollection = false; private static boolean getOnlinePlayersIsCollection = false;
private static Method getOnlinePlayers; private static Method getOnlinePlayers;
@ -39,6 +40,7 @@ public final class Utils {
static { static {
plugin = AuthMe.getInstance(); plugin = AuthMe.getInstance();
wrapper = new Wrapper(plugin);
checkGeoIP(); checkGeoIP();
initializeOnlinePlayersIsCollectionField(); initializeOnlinePlayersIsCollectionField();
} }
@ -65,7 +67,7 @@ public final class Utils {
} }
} }
} }
plugin.getGameServer().getScheduler().runTaskAsynchronously(plugin, new Runnable() { wrapper.getServer().getScheduler().runTaskAsynchronously(wrapper.getAuthMe(), new Runnable() {
@Override @Override
public void run() { public void run() {
try { try {
@ -190,6 +192,7 @@ public final class Utils {
} }
if (!Settings.isForcedRegistrationEnabled) { if (!Settings.isForcedRegistrationEnabled) {
// TODO ljacqu 20151123: Use a setter to retrieve things from AuthMe
if (!plugin.database.isAuthAvailable(name)) { if (!plugin.database.isAuthAvailable(name)) {
return true; return true;
} }
@ -225,12 +228,12 @@ public final class Utils {
final World world = theWorld; final World world = theWorld;
final Location loc = new Location(world, x, y, z); final Location loc = new Location(world, x, y, z);
Bukkit.getScheduler().scheduleSyncDelayedTask(plugin, new Runnable() { Bukkit.getScheduler().scheduleSyncDelayedTask(wrapper.getAuthMe(), new Runnable() {
@Override @Override
public void run() { public void run() {
AuthMeTeleportEvent tpEvent = new AuthMeTeleportEvent(pl, loc); AuthMeTeleportEvent tpEvent = new AuthMeTeleportEvent(pl, loc);
plugin.getServer().getPluginManager().callEvent(tpEvent); wrapper.getServer().getPluginManager().callEvent(tpEvent);
if (!tpEvent.isCancelled()) { if (!tpEvent.isCancelled()) {
pl.teleport(tpEvent.getTo()); pl.teleport(tpEvent.getTo());
} }
@ -325,7 +328,7 @@ public final class Utils {
@SuppressWarnings("deprecation") @SuppressWarnings("deprecation")
public static Player getPlayer(String name) { public static Player getPlayer(String name) {
name = name.toLowerCase(); name = name.toLowerCase();
return plugin.getServer().getPlayer(name); return wrapper.getServer().getPlayer(name);
} }
public static boolean isNPC(final Entity player) { public static boolean isNPC(final Entity player) {
@ -333,6 +336,7 @@ public final class Utils {
if (player.hasMetadata("NPC")) { if (player.hasMetadata("NPC")) {
return true; return true;
} else if (plugin.combatTagPlus != null } else if (plugin.combatTagPlus != null
// TODO ljacqu 20151123: Use a getter for combatTagPlus in AuthMe instead of using direct field access
&& player instanceof Player && player instanceof Player
&& plugin.combatTagPlus.getNpcPlayerHelper().isNpc((Player) player)) { && plugin.combatTagPlus.getNpcPlayerHelper().isNpc((Player) player)) {
return true; return true;
@ -347,7 +351,7 @@ public final class Utils {
if (Settings.isTeleportToSpawnEnabled && !Settings.noTeleport) { if (Settings.isTeleportToSpawnEnabled && !Settings.noTeleport) {
Location spawn = plugin.getSpawnLocation(player); Location spawn = plugin.getSpawnLocation(player);
AuthMeTeleportEvent tpEvent = new AuthMeTeleportEvent(player, spawn); AuthMeTeleportEvent tpEvent = new AuthMeTeleportEvent(player, spawn);
plugin.getServer().getPluginManager().callEvent(tpEvent); wrapper.getServer().getPluginManager().callEvent(tpEvent);
if (!tpEvent.isCancelled()) { if (!tpEvent.isCancelled()) {
player.teleport(tpEvent.getTo()); player.teleport(tpEvent.getTo());
} }

View File

@ -0,0 +1,34 @@
package fr.xephi.authme.util;
import fr.xephi.authme.AuthMe;
import org.bukkit.Server;
import java.util.logging.Logger;
/**
* Wrapper for the retrieval of common singletons used throughout the application.
* This class simply delegates the calls.
*/
public class Wrapper {
private AuthMe authMe;
public Wrapper(AuthMe authMe) {
this.authMe = authMe;
}
public AuthMe getAuthMe() {
return authMe;
}
public Server getServer() {
return authMe.getServer();
}
public Logger getLogger() {
return authMe.getLogger();
}
}

View File

@ -1,6 +1,7 @@
package fr.xephi.authme; package fr.xephi.authme;
import fr.xephi.authme.settings.Messages; import fr.xephi.authme.settings.Messages;
import fr.xephi.authme.util.Wrapper;
import org.mockito.Mockito; import org.mockito.Mockito;
import java.lang.reflect.Field; import java.lang.reflect.Field;
@ -15,15 +16,18 @@ public final class AuthMeMockUtil {
} }
/** /**
* Sets the AuthMe plugin instance to a mock object. Use {@link AuthMe#getInstance()} to retrieve the mock. * Set the AuthMe plugin instance to a mock object. Use {@link AuthMe#getInstance()} to retrieve the mock.
*
* @return The generated mock for the AuthMe instance
*/ */
public static void mockAuthMeInstance() { public static AuthMe mockAuthMeInstance() {
AuthMe mock = Mockito.mock(AuthMe.class); AuthMe mock = Mockito.mock(AuthMe.class);
mockSingletonForClass(AuthMe.class, "plugin", mock); mockSingletonForClass(AuthMe.class, "plugin", mock);
return mock;
} }
/** /**
* Creates a mock Messages object for the instance returned from {@link Messages#getInstance()}. * Create a mock Messages object for the instance returned from {@link Messages#getInstance()}.
*/ */
public static void mockMessagesInstance() { public static void mockMessagesInstance() {
Messages mock = Mockito.mock(Messages.class); Messages mock = Mockito.mock(Messages.class);
@ -31,11 +35,39 @@ public final class AuthMeMockUtil {
} }
/** /**
* Sets a field of a class to the given mock. * Set the given class' {@link Wrapper} field to a mock implementation.
* *
* @param clazz the class to modify * @param clazz The class to modify
* @param fieldName the field name * @param fieldName The name of the field containing the Wrapper in the class
* @param mock the mock to set for the given field *
* @return The generated Wrapper mock
* @see WrapperMock
*/
public static Wrapper insertMockWrapperInstance(Class<?> clazz, String fieldName) {
Wrapper wrapperMock = new WrapperMock();
mockSingletonForClass(clazz, fieldName, wrapperMock);
return wrapperMock;
}
public static Wrapper insertMockWrapperInstance(Class<?> clazz, String fieldName, AuthMe authMe) {
Wrapper wrapperMock = new WrapperMock(authMe);
mockSingletonForClass(clazz, fieldName, wrapperMock);
return wrapperMock;
}
// TODO ljacqu 20151123: Find the use cases for the WrapperMock and remove any of these
// methods that will end up unused
public static Wrapper insertMockWrapperInstance(Class<?> clazz, String fieldName, WrapperMock wrapperMock) {
mockSingletonForClass(clazz, fieldName, wrapperMock);
return wrapperMock;
}
/**
* Set a field of a class to the given mock.
*
* @param clazz The class to modify
* @param fieldName The field name
* @param mock The mock to set for the given field
*/ */
private static void mockSingletonForClass(Class<?> clazz, String fieldName, Object mock) { private static void mockSingletonForClass(Class<?> clazz, String fieldName, Object mock) {
try { try {

View File

@ -0,0 +1,54 @@
package fr.xephi.authme;
import fr.xephi.authme.util.Wrapper;
import org.bukkit.Server;
import org.mockito.Mockito;
import java.util.HashMap;
import java.util.Map;
import java.util.logging.Logger;
/**
* Class returning mocks for all calls in {@link Wrapper}.
* This class keeps track of its mocks and will always return
* the same one for each type.
*/
public class WrapperMock extends Wrapper {
private static Map<Class<?>, Object> mocks = new HashMap<>();
public WrapperMock() {
this((AuthMe) getMock(AuthMe.class));
}
public WrapperMock(AuthMe authMe) {
super(authMe);
}
@Override
public Logger getLogger() {
return getMock(Logger.class);
}
@Override
public Server getServer() {
return getMock(Server.class);
}
@Override
public AuthMe getAuthMe() {
return getMock(AuthMe.class);
}
@SuppressWarnings("unchecked")
private static <T> T getMock(Class<?> clazz) {
Object o = mocks.get(clazz);
if (o == null) {
o = Mockito.mock(clazz);
mocks.put(clazz, o);
}
return (T) o;
}
}

View File

@ -0,0 +1,28 @@
package fr.xephi.authme.settings;
import fr.xephi.authme.AuthMe;
import fr.xephi.authme.AuthMeMockUtil;
import fr.xephi.authme.ConsoleLogger;
import org.junit.Before;
import org.junit.Test;
import java.io.File;
/**
* Test for {@link Messages}.
*/
public class MessagesTest {
@Before
public void setUpMessages() {
AuthMe authMe = AuthMeMockUtil.mockAuthMeInstance();
AuthMeMockUtil.insertMockWrapperInstance(ConsoleLogger.class, "wrapper", authMe);
File file = new File("messages_test.yml");
Messages messages = new Messages(file, "en");
}
@Test
public void shouldLoadMessages() {
}
}

View File

@ -2,6 +2,7 @@ package fr.xephi.authme.util;
import fr.xephi.authme.AuthMe; import fr.xephi.authme.AuthMe;
import fr.xephi.authme.AuthMeMockUtil; import fr.xephi.authme.AuthMeMockUtil;
import fr.xephi.authme.WrapperMock;
import fr.xephi.authme.permission.PermissionsManager; import fr.xephi.authme.permission.PermissionsManager;
import org.bukkit.Server; import org.bukkit.Server;
import org.bukkit.entity.Player; import org.bukkit.entity.Player;
@ -9,6 +10,7 @@ import org.bukkit.plugin.Plugin;
import org.bukkit.scheduler.BukkitScheduler; import org.bukkit.scheduler.BukkitScheduler;
import org.bukkit.scheduler.BukkitTask; import org.bukkit.scheduler.BukkitTask;
import org.junit.Before; import org.junit.Before;
import org.junit.Ignore;
import org.junit.Test; import org.junit.Test;
import java.lang.reflect.Field; import java.lang.reflect.Field;
@ -23,26 +25,32 @@ import static org.mockito.Mockito.when;
/** /**
* Test for the {@link Utils} class. * Test for the {@link Utils} class.
*/ */
@Ignore
// TODO ljacqu 20151123: Fix test setup
public class UtilsTest { public class UtilsTest {
private AuthMe authMeMock; private AuthMe authMeMock;
private PermissionsManager permissionsManagerMock; private PermissionsManager permissionsManagerMock;
private Wrapper wrapperMock;
@Before @Before
public void setUpMocks() { public void setUpMocks() {
AuthMeMockUtil.mockAuthMeInstance(); authMeMock = AuthMeMockUtil.mockAuthMeInstance();
authMeMock = AuthMe.getInstance();
permissionsManagerMock = mock(PermissionsManager.class); // We need to create the Wrapper mock before injecting it into Utils because it runs a lot of code in
when(authMeMock.getPermissionsManager()).thenReturn(permissionsManagerMock); // a static block which needs the proper mocks to be set up.
wrapperMock = new WrapperMock(authMeMock);
Server serverMock = mock(Server.class); Server serverMock = wrapperMock.getServer();
when(authMeMock.getGameServer()).thenReturn(serverMock);
BukkitScheduler schedulerMock = mock(BukkitScheduler.class); BukkitScheduler schedulerMock = mock(BukkitScheduler.class);
when(serverMock.getScheduler()).thenReturn(schedulerMock); when(serverMock.getScheduler()).thenReturn(schedulerMock);
when(schedulerMock.runTaskAsynchronously(any(Plugin.class), any(Runnable.class))) when(schedulerMock.runTaskAsynchronously(any(Plugin.class), any(Runnable.class)))
.thenReturn(mock(BukkitTask.class)); .thenReturn(mock(BukkitTask.class));
AuthMeMockUtil.insertMockWrapperInstance(Utils.class, "wrapper", (WrapperMock) wrapperMock);
permissionsManagerMock = mock(PermissionsManager.class);
when(authMeMock.getPermissionsManager()).thenReturn(permissionsManagerMock);
} }
// TODO ljacques 20151122: The tests for Utils.forceGM somehow can't be set up with the mocks correctly // TODO ljacques 20151122: The tests for Utils.forceGM somehow can't be set up with the mocks correctly

View File

@ -0,0 +1,58 @@
unknown_user: '&cCan''t find the requested user in the database!'
unsafe_spawn: '&cYour quit location was unsafe, you have been teleported to the world''s spawnpoint.'
not_logged_in: '&cYou''re not logged in!'
reg_voluntarily: 'You can register yourself to the server with the command "/register <password> <ConfirmPassword>"'
usage_log: '&cUsage: /login <password>'
wrong_pwd: '&cWrong password!'
unregistered: '&cSuccessfully unregistered!'
reg_disabled: '&cIn-game registration is disabled!'
valid_session: '&2Logged-in due to Session Reconnection.'
login: '&2Successful login!'
vb_nonActiv: '&cYour account isn''t activated yet, please check your emails!'
user_regged: '&cYou already have registered this username!'
usage_reg: '&cUsage: /register <password> <ConfirmPassword>'
max_reg: '&cYou have exceeded the maximum number of registrations for your connection!'
no_perm: '&4You don''t have the permission to perform this action!'
error: '&4An unexpected error occurred, please contact an Administrator!'
login_msg: '&cPlease, login with the command "/login <password>"'
reg_msg: '&3Please, register to the server with the command "/register <password> <ConfirmPassword>"'
reg_email_msg: '&3Please, register to the server with the command "/register <email> <confirmEmail>"'
usage_unreg: '&cUsage: /unregister <password>'
pwd_changed: '&2Password changed successfully!'
user_unknown: '&cThis user isn''t registered!'
password_error: '&cPasswords didn''t match, check them again!'
password_error_nick: '&cYou can''t use your name as password, please choose another one...'
password_error_unsafe: '&cThe chosen password isn''t safe, please choose another one...'
invalid_session: '&cYour IP has been changed and your session data has expired!'
reg_only: '&4Only registered users can join the server! Please visit http://example.com to register yourself!'
logged_in: '&cYou''re already logged in!'
logout: '&2Logged-out successfully!'
same_nick: '&4The same username is already playing on the server!'
registered: '&2Successfully registered!'
pass_len: '&cYour password is too short or too long! Please try with another one!'
reload: '&2Configuration and database have been reloaded correctly!'
timeout: '&4Login timeout exceeded, you have been kicked from the server, please try again!'
usage_changepassword: '&cUsage: /changepassword <oldPassword> <newPassword>'
name_len: '&4Your username is either too short or too long!'
regex: '&4Your username contains illegal characters. Allowed chars: REG_EX'
add_email: '&3Please add your email to your account with the command "/email add <yourEmail> <confirmEmail>"'
recovery_email: '&3Forgot your password? Please use the command "/email recovery <yourEmail>"'
usage_captcha: '&3To login you have to solve a captcha code, please use the command "/captcha <theCaptcha>"'
wrong_captcha: '&cWrong Captcha, please type "/captcha THE_CAPTCHA" into the chat!'
valid_captcha: '&2Captcha code solved correctly!'
kick_forvip: '&3A VIP Player has joined the server when it was full!'
kick_fullserver: '&4The server is full, try again later!'
usage_email_add: '&cUsage: /email add <email> <confirmEmail>'
usage_email_change: '&cUsage: /email change <oldEmail> <newEmail>'
usage_email_recovery: '&cUsage: /email recovery <Email>'
new_email_invalid: '&cInvalid New Email, try again!'
old_email_invalid: '&cInvalid Old Email, try again!'
email_invalid: '&cInvalid Email address, try again!'
email_added: '&2Email address successfully added to your account!'
email_confirm: '&cPlease confirm your email address!'
email_changed: '&2Email address changed correctly!'
email_send: '&2Recovery email sent correctly! Check your email inbox!'
email_exists: '&cA recovery email was already sent! You can discart it and send a new one using the command below:'
country_banned: '&4Your country is banned from this server!'
antibot_auto_enabled: '&4[AntiBotService] AntiBot enabled due to the huge number of connections!'
antibot_auto_disabled: '&2[AntiBotService] AntiBot disabled disabled after %m minutes!'