From 81533718cdac5f832606d10ce239395d9803cf7d Mon Sep 17 00:00:00 2001 From: Fuzzlemann Date: Sat, 3 Nov 2018 21:09:12 +0100 Subject: [PATCH] [Fix] Test & /raw/server/ improvements (#776) by Fuzzlemann * Optimizing the loading of the network page and raw server data page -> Loads the nicknames in bulk instead of one by one -> Performance gain with the raw data 45719ms vs 10054ms -> Scales up with more even more users * Adding UsersTable#getUUIDsAndNamesByID so when you want to get the UUID and the player name you do not have to execute UsersTable#getPlayerNames and UsersTable#getUUIDsByID -> Performance improvements * Fixes the test failures * Renamed method names so they correctly show the real DB used True fix of the tests (no fail after ~3500 passes) -> Implementation of TestRunnableFactory -> Removal of the timeout on CommonDBTest#testSaveCommandUse which is way too sensitive to the testing environment (disk speed, etc.) --- .../system/database/databases/Database.java | 2 +- .../system/database/databases/sql/H2DB.java | 4 +- .../database/databases/sql/MySQLDB.java | 3 +- .../system/database/databases/sql/SQLDB.java | 12 ++ .../database/databases/sql/SQLiteDB.java | 3 +- .../databases/sql/operation/SQLFetchOps.java | 7 +- .../databases/sql/tables/NicknamesTable.java | 36 ++++++ .../databases/sql/tables/UserInfoTable.java | 10 +- .../databases/sql/tables/UsersTable.java | 24 ++++ .../database/databases/CommonDBTest.java | 2 +- .../system/database/databases/SQLiteTest.java | 4 +- .../utilities/mocks/PlanBukkitMocker.java | 5 +- .../mocks/objects/TestRunnableFactory.java | 113 ++++++++++++++++++ 13 files changed, 209 insertions(+), 16 deletions(-) create mode 100644 Plan/src/test/java/utilities/mocks/objects/TestRunnableFactory.java diff --git a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/Database.java b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/Database.java index 40bcae431..ee08804fc 100644 --- a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/Database.java +++ b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/Database.java @@ -29,7 +29,7 @@ import com.djrapitops.plan.system.database.databases.operation.*; */ public abstract class Database { - protected boolean open = false; + protected volatile boolean open = false; public abstract void init() throws DBInitException; diff --git a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/H2DB.java b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/H2DB.java index 13121e89f..44891f394 100644 --- a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/H2DB.java +++ b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/H2DB.java @@ -145,13 +145,13 @@ public class H2DB extends SQLDB { @Override public void close() { + super.close(); stopConnectionPingTask(); + if (connection != null) { logger.debug("H2DB " + dbName + ": Closed Connection"); MiscUtils.close(connection); } - - super.close(); } @Override diff --git a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/MySQLDB.java b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/MySQLDB.java index 739e4f3e3..b87b86978 100644 --- a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/MySQLDB.java +++ b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/MySQLDB.java @@ -134,10 +134,11 @@ public class MySQLDB extends SQLDB { @Override public void close() { + super.close(); + if (dataSource instanceof HikariDataSource) { ((HikariDataSource) dataSource).close(); } - super.close(); } @Override diff --git a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/SQLDB.java b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/SQLDB.java index b275e937b..3206cf3fa 100644 --- a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/SQLDB.java +++ b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/SQLDB.java @@ -288,6 +288,10 @@ public abstract class SQLDB extends Database { public abstract void returnToPool(Connection connection); public boolean execute(ExecStatement statement) { + if (!isOpen()) { + throw new DBOpException("SQL Statement tried to execute while connection closed"); + } + Connection connection = null; try { connection = getConnection(); @@ -328,6 +332,10 @@ public abstract class SQLDB extends Database { } public void executeBatch(ExecStatement statement) { + if (!isOpen()) { + throw new DBOpException("SQL Batch tried to execute while connection closed"); + } + Connection connection = null; try { connection = getConnection(); @@ -346,6 +354,10 @@ public abstract class SQLDB extends Database { } public T query(QueryStatement statement) { + if (!isOpen()) { + throw new DBOpException("SQL Query tried to execute while connection closed"); + } + Connection connection = null; try { connection = getConnection(); diff --git a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/SQLiteDB.java b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/SQLiteDB.java index c5d678771..7b722c39f 100644 --- a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/SQLiteDB.java +++ b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/SQLiteDB.java @@ -137,12 +137,13 @@ public class SQLiteDB extends SQLDB { @Override public void close() { + super.close(); stopConnectionPingTask(); + if (connection != null) { logger.debug("SQLite " + dbName + ": Closed Connection"); MiscUtils.close(connection); } - super.close(); } @Override diff --git a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/operation/SQLFetchOps.java b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/operation/SQLFetchOps.java index ef1f614f8..071e22c85 100644 --- a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/operation/SQLFetchOps.java +++ b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/operation/SQLFetchOps.java @@ -24,6 +24,7 @@ import com.djrapitops.plan.data.store.mutators.PerServerMutator; import com.djrapitops.plan.data.store.mutators.PlayersMutator; import com.djrapitops.plan.data.store.mutators.SessionsMutator; import com.djrapitops.plan.data.store.objects.DateObj; +import com.djrapitops.plan.data.store.objects.Nickname; import com.djrapitops.plan.data.time.WorldTimes; import com.djrapitops.plan.system.cache.SessionCache; import com.djrapitops.plan.system.database.databases.operation.FetchOperations; @@ -123,6 +124,7 @@ public class SQLFetchOps extends SQLOps implements FetchOperations { Map timesKicked = usersTable.getAllTimesKicked(); Map> geoInfo = geoInfoTable.getAllGeoInfo(); Map> allPings = pingTable.getAllPings(); + Map> allNicknames = nicknamesTable.getAllNicknamesUnmapped(); Map> sessions = sessionsTable.getSessionInfoOfServer(serverUUID); Map>> map = new HashMap<>(); @@ -144,7 +146,7 @@ public class SQLFetchOps extends SQLOps implements FetchOperations { container.putRawData(PlayerKeys.KICK_COUNT, timesKicked.get(uuid)); container.putRawData(PlayerKeys.GEO_INFO, geoInfo.get(uuid)); container.putRawData(PlayerKeys.PING, allPings.get(uuid)); - container.putCachingSupplier(PlayerKeys.NICKNAMES, () -> nicknamesTable.getNicknameInformation(uuid)); + container.putRawData(PlayerKeys.NICKNAMES, allNicknames.get(uuid)); container.putRawData(PlayerKeys.PER_SERVER, perServerInfo.get(uuid)); container.putRawData(PlayerKeys.BANNED, userInfo.isBanned()); @@ -187,6 +189,7 @@ public class SQLFetchOps extends SQLOps implements FetchOperations { Map timesKicked = usersTable.getAllTimesKicked(); Map> geoInfo = geoInfoTable.getAllGeoInfo(); Map> allPings = pingTable.getAllPings(); + Map> allNicknames = nicknamesTable.getAllNicknamesUnmapped(); Map>> sessions = sessionsTable.getAllSessions(false); Map> allUserInfo = userInfoTable.getAllUserInfo(); @@ -202,7 +205,7 @@ public class SQLFetchOps extends SQLOps implements FetchOperations { container.putRawData(PlayerKeys.KICK_COUNT, timesKicked.get(uuid)); container.putRawData(PlayerKeys.GEO_INFO, geoInfo.get(uuid)); container.putRawData(PlayerKeys.PING, allPings.get(uuid)); - container.putCachingSupplier(PlayerKeys.NICKNAMES, () -> nicknamesTable.getNicknameInformation(uuid)); + container.putRawData(PlayerKeys.NICKNAMES, allNicknames.get(uuid)); container.putRawData(PlayerKeys.PER_SERVER, perServerInfo.get(uuid)); container.putCachingSupplier(PlayerKeys.SESSIONS, () -> { diff --git a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/NicknamesTable.java b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/NicknamesTable.java index fd18df308..ca8eafae3 100644 --- a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/NicknamesTable.java +++ b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/NicknamesTable.java @@ -162,6 +162,42 @@ public class NicknamesTable extends UserIDTable { }); } + /** + * Get nicknames of all users but doesn't map them by Server + * + * @return a {@code Map} with all nicknames of all users + * @see NicknamesTable#getAllNicknames(); + */ + public Map> getAllNicknamesUnmapped() { + String usersIDColumn = usersTable + "." + UsersTable.Col.ID; + String usersUUIDColumn = usersTable + "." + UsersTable.Col.UUID + " as uuid"; + String serverIDColumn = serverTable + "." + ServerTable.Col.SERVER_ID; + String serverUUIDColumn = serverTable + "." + ServerTable.Col.SERVER_UUID + " as s_uuid"; + String sql = "SELECT " + + Col.NICKNAME + ", " + + Col.LAST_USED + ", " + + usersUUIDColumn + ", " + + serverUUIDColumn + + " FROM " + tableName + + " INNER JOIN " + usersTable + " on " + usersIDColumn + "=" + Col.USER_ID + + " INNER JOIN " + serverTable + " on " + serverIDColumn + "=" + Col.SERVER_ID; + return query(new QueryAllStatement>>(sql, 5000) { + @Override + public Map> processResults(ResultSet set) throws SQLException { + Map> map = new HashMap<>(); + while (set.next()) { + UUID uuid = UUID.fromString(set.getString("uuid")); + UUID serverUUID = UUID.fromString(set.getString("s_uuid")); + List nicknames = map.computeIfAbsent(uuid, x -> new ArrayList<>()); + nicknames.add(new Nickname( + set.getString(Col.NICKNAME.get()), set.getLong(Col.LAST_USED.get()), serverUUID + )); + } + return map; + } + }); + } + public void saveUserName(UUID uuid, Nickname name) { List saved = getNicknameInformation(uuid); if (saved.contains(name)) { diff --git a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/UserInfoTable.java b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/UserInfoTable.java index f3c861c07..1caf8f7ae 100644 --- a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/UserInfoTable.java +++ b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/UserInfoTable.java @@ -187,8 +187,7 @@ public class UserInfoTable extends UserIDTable { return new ArrayList<>(); } - Map playerNames = usersTable.getPlayerNames(); - Map uuidsByID = usersTable.getUUIDsByID(); + Map> uuidsAndNamesByID = usersTable.getUUIDsAndNamesByID(); String sql = "SELECT * FROM " + tableName + " WHERE " + Col.SERVER_ID + "=?"; @@ -207,8 +206,11 @@ public class UserInfoTable extends UserIDTable { boolean op = set.getBoolean(Col.OP.get()); boolean banned = set.getBoolean(Col.BANNED.get()); int userId = set.getInt(Col.USER_ID.get()); - UUID uuid = uuidsByID.get(userId); - String name = playerNames.getOrDefault(uuid, "Unknown"); + + Map.Entry uuidNameEntry = uuidsAndNamesByID.get(userId); + UUID uuid = uuidNameEntry.getKey(); + String name = uuidNameEntry.getValue(); + UserInfo info = new UserInfo(uuid, name, registered, op, banned); if (!userInfo.contains(info)) { userInfo.add(info); diff --git a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/UsersTable.java b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/UsersTable.java index 26a7d15b4..367d307a6 100644 --- a/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/UsersTable.java +++ b/Plan/src/main/java/com/djrapitops/plan/system/database/databases/sql/tables/UsersTable.java @@ -416,6 +416,30 @@ public class UsersTable extends UserIDTable { }); } + /** + * Gets the {@code UUID} and the name of the player mapped to the user ID + * + * @return a {@code Map>} where the key is the user ID + * and the value is an {@code Map.Entry>} of the player's {@code UUID} and name + */ + public Map> getUUIDsAndNamesByID() { + String sql = Select.from(tableName, Col.ID, Col.UUID, Col.USER_NAME).toString(); + return query(new QueryAllStatement>>(sql, 20000) { + @Override + public Map> processResults(ResultSet set) throws SQLException { + Map> uuidsAndNamesByID = new TreeMap<>(); + while (set.next()) { + int id = set.getInt(Col.ID.get()); + UUID uuid = UUID.fromString(set.getString(Col.UUID.get())); + String name = set.getString(Col.USER_NAME.get()); + uuidsAndNamesByID.put(id, new AbstractMap.SimpleEntry<>(uuid, name)); + } + return uuidsAndNamesByID; + } + }); + } + + public DataContainer getUserInformation(UUID uuid) { Key user_data = new Key<>(DataContainer.class, "plan_users_data"); DataContainer returnValue = new DataContainer(); diff --git a/Plan/src/test/java/com/djrapitops/plan/system/database/databases/CommonDBTest.java b/Plan/src/test/java/com/djrapitops/plan/system/database/databases/CommonDBTest.java index 8334cc45a..af4634070 100644 --- a/Plan/src/test/java/com/djrapitops/plan/system/database/databases/CommonDBTest.java +++ b/Plan/src/test/java/com/djrapitops/plan/system/database/databases/CommonDBTest.java @@ -126,7 +126,7 @@ public abstract class CommonDBTest { db.commit(db.getConnection()); } - @Test(timeout = 3000) + @Test public void testSaveCommandUse() throws DBInitException { CommandUseTable commandUseTable = db.getCommandUseTable(); Map expected = new HashMap<>(); diff --git a/Plan/src/test/java/com/djrapitops/plan/system/database/databases/SQLiteTest.java b/Plan/src/test/java/com/djrapitops/plan/system/database/databases/SQLiteTest.java index 5ce09e60f..c4f0f2f81 100644 --- a/Plan/src/test/java/com/djrapitops/plan/system/database/databases/SQLiteTest.java +++ b/Plan/src/test/java/com/djrapitops/plan/system/database/databases/SQLiteTest.java @@ -42,12 +42,12 @@ public class SQLiteTest extends CommonDBTest { } @Test - public void testH2GetConfigName() { + public void testSQLiteGetConfigName() { assertEquals("sqlite", db.getType().getConfigName()); } @Test - public void testH2GetName() { + public void testSQLiteGetName() { assertEquals("SQLite", db.getType().getName()); } diff --git a/Plan/src/test/java/utilities/mocks/PlanBukkitMocker.java b/Plan/src/test/java/utilities/mocks/PlanBukkitMocker.java index 3fee8477e..bc6f964ba 100644 --- a/Plan/src/test/java/utilities/mocks/PlanBukkitMocker.java +++ b/Plan/src/test/java/utilities/mocks/PlanBukkitMocker.java @@ -14,7 +14,7 @@ import com.djrapitops.plugin.logging.debug.DebugLogger; import com.djrapitops.plugin.logging.debug.MemoryDebugLogger; import com.djrapitops.plugin.logging.error.ConsoleErrorLogger; import com.djrapitops.plugin.logging.error.ErrorHandler; -import com.djrapitops.plugin.task.thread.ThreadRunnableFactory; +import com.djrapitops.plugin.task.RunnableFactory; import org.bukkit.Server; import org.bukkit.command.ConsoleCommandSender; import org.bukkit.plugin.InvalidDescriptionException; @@ -23,6 +23,7 @@ import org.bukkit.scheduler.BukkitScheduler; import org.mockito.Mockito; import utilities.TestConstants; import utilities.mocks.objects.TestLogger; +import utilities.mocks.objects.TestRunnableFactory; import java.io.File; import java.io.FileInputStream; @@ -54,7 +55,7 @@ public class PlanBukkitMocker extends Mocker { doReturn("1.0.0").when(planMock).getVersion(); TestLogger testLogger = new TestLogger(); - ThreadRunnableFactory runnableFactory = new ThreadRunnableFactory(); + RunnableFactory runnableFactory = new TestRunnableFactory(); PluginLogger testPluginLogger = new TestPluginLogger(); DebugLogger debugLogger = new CombineDebugLogger(new MemoryDebugLogger()); ErrorHandler consoleErrorLogger = new ConsoleErrorLogger(testPluginLogger); diff --git a/Plan/src/test/java/utilities/mocks/objects/TestRunnableFactory.java b/Plan/src/test/java/utilities/mocks/objects/TestRunnableFactory.java new file mode 100644 index 000000000..512fab777 --- /dev/null +++ b/Plan/src/test/java/utilities/mocks/objects/TestRunnableFactory.java @@ -0,0 +1,113 @@ +/* + * License is provided in the jar as LICENSE also here: + * https://github.com/Rsl1122/Plan-PlayerAnalytics/blob/master/Plan/src/main/resources/LICENSE + */ +package utilities.mocks.objects; + +import com.djrapitops.plugin.api.TimeAmount; +import com.djrapitops.plugin.task.AbsRunnable; +import com.djrapitops.plugin.task.PluginRunnable; +import com.djrapitops.plugin.task.PluginTask; +import com.djrapitops.plugin.task.RunnableFactory; + +import java.util.concurrent.Executors; +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.TimeUnit; + +/** + * @author Fuzzlemann + * @since 4.5.1 + */ +public class TestRunnableFactory extends RunnableFactory { + + private final ScheduledExecutorService executorService = Executors.newScheduledThreadPool(1); + + @Override + protected PluginRunnable createNewRunnable(String name, AbsRunnable absRunnable, long l) { + return new PluginRunnable() { + @Override + public String getTaskName() { + return name; + } + + @Override + public void cancel() { + absRunnable.cancel(); + } + + @Override + public int getTaskId() { + return absRunnable.getTaskId(); + } + + @Override + public PluginTask runTask() { + absRunnable.run(); + return createPluginTask(getTaskId(), true, absRunnable::cancel); + } + + @Override + public PluginTask runTaskAsynchronously() { + executorService.submit(absRunnable); + return createPluginTask(getTaskId(), false, absRunnable::cancel); + } + + @Override + public PluginTask runTaskLater(long l) { + return runTaskLaterAsynchronously(l); + } + + @Override + public PluginTask runTaskLaterAsynchronously(long l) { + executorService.schedule(absRunnable, TimeAmount.ticksToMillis(l), TimeUnit.MILLISECONDS); + return createPluginTask(getTaskId(), false, absRunnable::cancel); + } + + @Override + public PluginTask runTaskTimer(long l, long l1) { + return runTaskLaterAsynchronously(l); + } + + @Override + public PluginTask runTaskTimerAsynchronously(long l, long l1) { + executorService.scheduleAtFixedRate(absRunnable, TimeAmount.ticksToMillis(l), TimeAmount.ticksToMillis(l1), TimeUnit.MILLISECONDS); + return createPluginTask(getTaskId(), false, absRunnable::cancel); + } + + @Override + public long getTime() { + return l; + } + }; + } + + @Override + public void cancelAllKnownTasks() { + executorService.shutdownNow(); + } + + private PluginTask createPluginTask(int taskID, boolean sync, ICloseTask closeTask) { + return new PluginTask() { + @Override + public int getTaskId() { + return taskID; + } + + @Override + public boolean isSync() { + return sync; + } + + @Override + public void cancel() { + if (closeTask != null) { + closeTask.close(); + } + } + }; + } + + private interface ICloseTask { + void close(); + } +}