From d944c2bd89a89b903d25b5fa235bfba4e99ce6c2 Mon Sep 17 00:00:00 2001 From: Intelli Date: Mon, 25 Sep 2023 14:04:50 -0600 Subject: [PATCH] Fixed SQLFeatureNotSupportedException when logging entity data (fixes #455) --- .../net/coreprotect/database/Database.java | 11 ++++++- .../database/logger/EntityKillLogger.java | 21 ++++++++++--- .../database/logger/SkullBreakLogger.java | 18 ++++++++--- .../database/logger/SkullPlaceLogger.java | 18 ++++++++--- .../database/statement/EntityStatement.java | 13 ++++++-- .../database/statement/SkullStatement.java | 13 ++++++-- .../database/statement/UserStatement.java | 31 +++++++++++++++---- 7 files changed, 99 insertions(+), 26 deletions(-) diff --git a/src/main/java/net/coreprotect/database/Database.java b/src/main/java/net/coreprotect/database/Database.java index 95556b57..844c8f05 100755 --- a/src/main/java/net/coreprotect/database/Database.java +++ b/src/main/java/net/coreprotect/database/Database.java @@ -106,6 +106,10 @@ public static void setMultiInt(PreparedStatement statement, int value, int count } } + public static boolean hasReturningKeys() { + return (!Config.getGlobal().MYSQL && ConfigHandler.SERVER_VERSION >= 20); + } + public static void containerBreakCheck(String user, Material type, Object container, ItemStack[] contents, Location location) { if (BlockGroup.CONTAINERS.contains(type) && !BlockGroup.SHULKER_BOXES.contains(type)) { if (Config.getConfig(location.getWorld()).ITEM_TRANSACTIONS) { @@ -286,7 +290,12 @@ private static PreparedStatement prepareStatement(Connection connection, String PreparedStatement preparedStatement = null; try { if (keys) { - preparedStatement = connection.prepareStatement(query, Statement.RETURN_GENERATED_KEYS); + if (hasReturningKeys()) { + preparedStatement = connection.prepareStatement(query + " RETURNING rowid"); + } + else { + preparedStatement = connection.prepareStatement(query, Statement.RETURN_GENERATED_KEYS); + } } else { preparedStatement = connection.prepareStatement(query); diff --git a/src/main/java/net/coreprotect/database/logger/EntityKillLogger.java b/src/main/java/net/coreprotect/database/logger/EntityKillLogger.java index 526d0ebf..2fd3d9ce 100644 --- a/src/main/java/net/coreprotect/database/logger/EntityKillLogger.java +++ b/src/main/java/net/coreprotect/database/logger/EntityKillLogger.java @@ -10,6 +10,7 @@ import net.coreprotect.CoreProtect; import net.coreprotect.config.Config; import net.coreprotect.config.ConfigHandler; +import net.coreprotect.database.Database; import net.coreprotect.database.statement.BlockStatement; import net.coreprotect.database.statement.EntityStatement; import net.coreprotect.database.statement.UserStatement; @@ -43,11 +44,21 @@ public static void log(PreparedStatement preparedStmt, PreparedStatement prepare int x = block.getX(); int y = block.getY(); int z = block.getZ(); - EntityStatement.insert(preparedStmt2, time, data); - ResultSet keys = preparedStmt2.getGeneratedKeys(); - keys.next(); - int entity_key = keys.getInt(1); - keys.close(); + int entity_key = 0; + + ResultSet resultSet = EntityStatement.insert(preparedStmt2, time, data); + if (Database.hasReturningKeys()) { + resultSet.next(); + entity_key = resultSet.getInt(1); + resultSet.close(); + } + else { + ResultSet keys = preparedStmt2.getGeneratedKeys(); + keys.next(); + entity_key = keys.getInt(1); + keys.close(); + } + BlockStatement.insert(preparedStmt, batchCount, time, userId, wid, x, y, z, type, entity_key, null, null, 3, 0); } catch (Exception e) { diff --git a/src/main/java/net/coreprotect/database/logger/SkullBreakLogger.java b/src/main/java/net/coreprotect/database/logger/SkullBreakLogger.java index cf69194b..577f4b0c 100644 --- a/src/main/java/net/coreprotect/database/logger/SkullBreakLogger.java +++ b/src/main/java/net/coreprotect/database/logger/SkullBreakLogger.java @@ -8,6 +8,7 @@ import org.bukkit.block.Skull; import net.coreprotect.config.ConfigHandler; +import net.coreprotect.database.Database; import net.coreprotect.database.statement.SkullStatement; import net.coreprotect.utility.Util; @@ -29,11 +30,18 @@ public static void log(PreparedStatement preparedStmt, PreparedStatement prepare int skullKey = 0; if (skull.hasOwner()) { skullOwner = skull.getOwningPlayer().getUniqueId().toString(); - SkullStatement.insert(preparedStmt2, time, skullOwner); - ResultSet keys = preparedStmt2.getGeneratedKeys(); - keys.next(); - skullKey = keys.getInt(1); - keys.close(); + ResultSet resultSet = SkullStatement.insert(preparedStmt2, time, skullOwner); + if (Database.hasReturningKeys()) { + resultSet.next(); + skullKey = resultSet.getInt(1); + resultSet.close(); + } + else { + ResultSet keys = preparedStmt2.getGeneratedKeys(); + keys.next(); + skullKey = keys.getInt(1); + keys.close(); + } } BlockBreakLogger.log(preparedStmt, batchCount, user, block.getLocation(), type, skullKey, null, block.getBlockData().getAsString(), null); diff --git a/src/main/java/net/coreprotect/database/logger/SkullPlaceLogger.java b/src/main/java/net/coreprotect/database/logger/SkullPlaceLogger.java index ce39c5ff..e8bdfa59 100644 --- a/src/main/java/net/coreprotect/database/logger/SkullPlaceLogger.java +++ b/src/main/java/net/coreprotect/database/logger/SkullPlaceLogger.java @@ -9,6 +9,7 @@ import org.bukkit.block.Skull; import net.coreprotect.config.ConfigHandler; +import net.coreprotect.database.Database; import net.coreprotect.database.statement.SkullStatement; public class SkullPlaceLogger { @@ -31,11 +32,18 @@ public static void log(PreparedStatement preparedStmt, PreparedStatement prepare String skullOwner = ""; if (skull.hasOwner()) { skullOwner = skull.getOwningPlayer().getUniqueId().toString(); - SkullStatement.insert(preparedStmt2, time, skullOwner); - ResultSet keys = preparedStmt2.getGeneratedKeys(); - keys.next(); - skullKey = keys.getInt(1); - keys.close(); + ResultSet resultSet = SkullStatement.insert(preparedStmt2, time, skullOwner); + if (Database.hasReturningKeys()) { + resultSet.next(); + skullKey = resultSet.getInt(1); + resultSet.close(); + } + else { + ResultSet keys = preparedStmt2.getGeneratedKeys(); + keys.next(); + skullKey = keys.getInt(1); + keys.close(); + } } } diff --git a/src/main/java/net/coreprotect/database/statement/EntityStatement.java b/src/main/java/net/coreprotect/database/statement/EntityStatement.java index f5eea69d..517f9859 100644 --- a/src/main/java/net/coreprotect/database/statement/EntityStatement.java +++ b/src/main/java/net/coreprotect/database/statement/EntityStatement.java @@ -12,13 +12,15 @@ import org.bukkit.util.io.BukkitObjectInputStream; import org.bukkit.util.io.BukkitObjectOutputStream; +import net.coreprotect.database.Database; + public class EntityStatement { private EntityStatement() { throw new IllegalStateException("Database class"); } - public static void insert(PreparedStatement preparedStmt, int time, List data) { + public static ResultSet insert(PreparedStatement preparedStmt, int time, List data) { try { ByteArrayOutputStream bos = new ByteArrayOutputStream(); BukkitObjectOutputStream oos = new BukkitObjectOutputStream(bos); @@ -30,11 +32,18 @@ public static void insert(PreparedStatement preparedStmt, int time, List byte[] byte_data = bos.toByteArray(); preparedStmt.setInt(1, time); preparedStmt.setObject(2, byte_data); - preparedStmt.executeUpdate(); + if (Database.hasReturningKeys()) { + return preparedStmt.executeQuery(); + } + else { + preparedStmt.executeUpdate(); + } } catch (Exception e) { e.printStackTrace(); } + + return null; } public static List getData(Statement statement, BlockState block, String query) { diff --git a/src/main/java/net/coreprotect/database/statement/SkullStatement.java b/src/main/java/net/coreprotect/database/statement/SkullStatement.java index b1221791..1a1bd8b1 100644 --- a/src/main/java/net/coreprotect/database/statement/SkullStatement.java +++ b/src/main/java/net/coreprotect/database/statement/SkullStatement.java @@ -9,21 +9,30 @@ import org.bukkit.block.BlockState; import org.bukkit.block.Skull; +import net.coreprotect.database.Database; + public class SkullStatement { private SkullStatement() { throw new IllegalStateException("Database class"); } - public static void insert(PreparedStatement preparedStmt, int time, String owner) { + public static ResultSet insert(PreparedStatement preparedStmt, int time, String owner) { try { preparedStmt.setInt(1, time); preparedStmt.setString(2, owner); - preparedStmt.executeUpdate(); + if (Database.hasReturningKeys()) { + return preparedStmt.executeQuery(); + } + else { + preparedStmt.executeUpdate(); + } } catch (Exception e) { e.printStackTrace(); } + + return null; } public static void getData(Statement statement, BlockState block, String query) { diff --git a/src/main/java/net/coreprotect/database/statement/UserStatement.java b/src/main/java/net/coreprotect/database/statement/UserStatement.java index ff7ef4be..00df51a4 100644 --- a/src/main/java/net/coreprotect/database/statement/UserStatement.java +++ b/src/main/java/net/coreprotect/database/statement/UserStatement.java @@ -9,6 +9,7 @@ import net.coreprotect.config.Config; import net.coreprotect.config.ConfigHandler; +import net.coreprotect.database.Database; public class UserStatement { @@ -21,14 +22,32 @@ public static int insert(Connection connection, String user) { try { int unixtimestamp = (int) (System.currentTimeMillis() / 1000L); - PreparedStatement preparedStmt = connection.prepareStatement("INSERT INTO " + ConfigHandler.prefix + "user (time, user) VALUES (?, ?)", Statement.RETURN_GENERATED_KEYS); + + PreparedStatement preparedStmt = null; + if (Database.hasReturningKeys()) { + preparedStmt = connection.prepareStatement("INSERT INTO " + ConfigHandler.prefix + "user (time, user) VALUES (?, ?) RETURNING rowid"); + } + else { + preparedStmt = connection.prepareStatement("INSERT INTO " + ConfigHandler.prefix + "user (time, user) VALUES (?, ?)", Statement.RETURN_GENERATED_KEYS); + } + preparedStmt.setInt(1, unixtimestamp); preparedStmt.setString(2, user); - preparedStmt.executeUpdate(); - ResultSet keys = preparedStmt.getGeneratedKeys(); - keys.next(); - id = keys.getInt(1); - keys.close(); + + if (Database.hasReturningKeys()) { + ResultSet resultSet = preparedStmt.executeQuery(); + resultSet.next(); + id = resultSet.getInt(1); + resultSet.close(); + } + else { + preparedStmt.executeUpdate(); + ResultSet keys = preparedStmt.getGeneratedKeys(); + keys.next(); + id = keys.getInt(1); + keys.close(); + } + preparedStmt.close(); } catch (Exception e) {