de.xaniox.heavyspleef.persistence.handler.CachingReadWriteHandler.java Source code

Java tutorial

Introduction

Here is the source code for de.xaniox.heavyspleef.persistence.handler.CachingReadWriteHandler.java

Source

/*
 * This file is part of HeavySpleef.
 * Copyright (c) 2014-2016 Matthias Werning
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
 */
package de.xaniox.heavyspleef.persistence.handler;

import com.google.common.cache.*;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import de.xaniox.heavyspleef.core.HeavySpleef;
import de.xaniox.heavyspleef.core.floor.Floor;
import de.xaniox.heavyspleef.core.game.Game;
import de.xaniox.heavyspleef.core.persistence.ReadWriteHandler;
import de.xaniox.heavyspleef.core.stats.Statistic;
import de.xaniox.heavyspleef.core.uuid.GameProfile;
import de.xaniox.heavyspleef.core.uuid.UUIDManager;
import de.xaniox.heavyspleef.persistence.schematic.FloorAccessor;
import de.xaniox.heavyspleef.persistence.schematic.SchematicContext;
import de.xaniox.heavyspleef.persistence.sql.DatabaseUpgrader;
import de.xaniox.heavyspleef.persistence.sql.SQLDatabaseContext;
import de.xaniox.heavyspleef.persistence.sql.SQLDatabaseContext.SQLImplementation;
import de.xaniox.heavyspleef.persistence.sql.SQLQueryOptionsBuilder;
import de.xaniox.heavyspleef.persistence.sql.SQLQueryOptionsBuilder.ExpressionList;
import de.xaniox.heavyspleef.persistence.sql.StatisticAccessor;
import de.xaniox.heavyspleef.persistence.xml.GameAccessor;
import de.xaniox.heavyspleef.persistence.xml.XMLContext;
import org.bukkit.Bukkit;
import org.bukkit.OfflinePlayer;
import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;
import org.dom4j.io.OutputFormat;
import org.dom4j.io.SAXReader;
import org.dom4j.io.XMLWriter;

import java.io.File;
import java.io.FileOutputStream;
import java.io.FilenameFilter;
import java.io.IOException;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.*;
import java.util.Map.Entry;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.ReentrantLock;
import java.util.logging.Level;
import java.util.logging.Logger;

public class CachingReadWriteHandler implements ReadWriteHandler {

    private static final long STATISTIC_CACHE_EXPIRE = 10 * 60 * 1000L;
    private static final FilenameFilter FLOOR_SCHEMATIC_FILTER = new FilenameFilter() {

        @Override
        public boolean accept(File dir, String name) {
            return name.toLowerCase().endsWith(".floor");
        }
    };
    private static final FilenameFilter XML_GAME_FILTER = new FilenameFilter() {

        @Override
        public boolean accept(File dir, String name) {
            return name.toLowerCase().endsWith(".xml");
        }
    };

    private final File xmlFolder;
    private final File schematicFolder;

    private final Logger logger;
    private UUIDManager uuidManager;
    private final SAXReader saxReader = new SAXReader();
    private final OutputFormat xmlOutputFormat = OutputFormat.createPrettyPrint();

    private SQLDatabaseContext sqlContext;
    private SchematicContext schematicContext;
    private XMLContext xmlContext;

    private ReentrantLock rankLock = new ReentrantLock();
    private ReentrantLock renameLock = new ReentrantLock(true);

    private LoadingCache<UUID, Statistic> statisticCache;
    private final CacheLoader<UUID, Statistic> statisticCacheLoader = new CacheLoader<UUID, Statistic>() {

        @Override
        public Statistic load(UUID uuid) throws Exception {
            if (sqlContext == null) {
                throw new IllegalStateException("SQL database has not been initialized");
            }

            SQLQueryOptionsBuilder optionsBuilder = SQLQueryOptionsBuilder.newBuilder().limit(1).where()
                    .eq(StatisticAccessor.ColumnContract.UUID, uuid).back();

            List<Statistic> statisticResult = sqlContext.readSql(Statistic.class, optionsBuilder);
            return !statisticResult.isEmpty() ? statisticResult.get(0) : new Statistic(uuid);
        }
    };

    public CachingReadWriteHandler(HeavySpleef heavySpleef, Properties properties,
            Map<UUID, Statistic> initStatistics, UUIDManager uuidManager) throws IOException, Exception {
        this.logger = heavySpleef.getLogger();

        this.uuidManager = uuidManager != null ? uuidManager : new UUIDManager();
        this.xmlFolder = (File) properties.get("xml.dir");
        this.schematicFolder = (File) properties.get("schematic.dir");

        GameAccessor gameAccessor = new GameAccessor(heavySpleef);
        xmlContext = new XMLContext(gameAccessor);

        FloorAccessor floorAccessor = new FloorAccessor();
        schematicContext = new SchematicContext(floorAccessor);

        boolean statisticsEnabled = (boolean) properties.get("statistic.enabled");
        if (statisticsEnabled) {
            StatisticAccessor statisticAccessor = new StatisticAccessor();
            DatabaseUpgrader upgrader = new HeavySpleefDatabaseUpgrader();
            sqlContext = new SQLDatabaseContext(properties, upgrader, statisticAccessor);

            int maxCacheSize = (int) properties.get("statistic.max_cache_size");

            //Create a cache for fast data access
            statisticCache = CacheBuilder.newBuilder()
                    .expireAfterAccess(STATISTIC_CACHE_EXPIRE, TimeUnit.MILLISECONDS).maximumSize(maxCacheSize)
                    .removalListener(new RemovalListener<UUID, Statistic>() {

                        @Override
                        public void onRemoval(RemovalNotification<UUID, Statistic> notification) {
                            //Save the statistic on remove
                            Statistic statistic = notification.getValue();

                            if (!statistic.isEmpty()) {
                                try {
                                    saveStatistic(statistic);
                                } catch (SQLException e) {
                                    logger.log(Level.SEVERE, "Could not save statistic for player with uuid "
                                            + statistic.getUniqueIdentifier() + ": ", e);
                                }
                            }
                        }
                    }).build(statisticCacheLoader);
        }

        if (initStatistics != null && !initStatistics.isEmpty()) {
            for (Entry<UUID, Statistic> entry : initStatistics.entrySet()) {
                statisticCache.put(entry.getKey(), entry.getValue());
            }
        }
    }

    @Override
    public void saveGames(Iterable<Game> iterable) throws IOException {
        for (Game game : iterable) {
            saveGame(game);
        }
    }

    @Override
    public void saveGame(Game game) throws IOException {
        File gameFile = new File(xmlFolder, game.getName() + ".xml");
        if (!gameFile.exists()) {
            gameFile.createNewFile();
        }

        Document document = DocumentHelper.createDocument();
        Element rootElement = document.addElement("game");

        xmlContext.write(game, rootElement);

        File gameSchematicFolder = new File(schematicFolder, game.getName());
        if (!gameSchematicFolder.exists()) {
            gameSchematicFolder.mkdir();
        }

        XMLWriter writer = null;

        try {
            FileOutputStream out = new FileOutputStream(gameFile);

            writer = new XMLWriter(out, xmlOutputFormat);
            writer.write(document);
        } finally {
            if (writer != null) {
                writer.close();
            }
        }

        for (File file : gameSchematicFolder.listFiles(FLOOR_SCHEMATIC_FILTER)) {
            String floorName = file.getName().substring(2, file.getName().length() - 6);
            if (game.isFloorPresent(floorName)) {
                continue;
            }

            file.delete();
        }

        for (Floor floor : game.getFloors()) {
            File floorFile = new File(gameSchematicFolder, getFloorFileName(floor));
            if (!floorFile.exists()) {
                floorFile.createNewFile();
            }

            schematicContext.write(floorFile, floor);
        }
    }

    private String getFloorFileName(Floor floor) {
        return "r." + floor.getName() + ".floor";
    }

    @Override
    public Game getGame(String name) throws IOException, DocumentException {
        File gameFile = new File(xmlFolder, name + ".xml");
        if (!gameFile.exists()) {
            return null;
        }

        return getGame(gameFile);
    }

    private Game getGame(File file) throws IOException, DocumentException {
        Document document = saxReader.read(file);

        if (!document.hasContent()) {
            return null;
        }

        Element rootElement = document.getRootElement();

        Game game = xmlContext.read(rootElement, Game.class);

        File gameFloorFolder = new File(schematicFolder, game.getName());
        if (gameFloorFolder.exists()) {
            for (File floorSchematicFile : gameFloorFolder.listFiles(FLOOR_SCHEMATIC_FILTER)) {
                Floor floor = schematicContext.read(floorSchematicFile, Floor.class);

                game.addFloor(floor);
            }
        }

        return game;
    }

    @Override
    public List<Game> getGames() throws IOException, DocumentException {
        List<Game> result = Lists.newArrayList();

        for (File gameFile : xmlFolder.listFiles(XML_GAME_FILTER)) {
            Game game = getGame(gameFile);
            if (game == null) {
                return null;
            }

            result.add(game);
        }

        return result;
    }

    @Override
    public void renameGame(Game game, String from, String to) throws IOException {
        renameLock.lock();

        try {
            File xmlFile = new File(xmlFolder, from + ".xml");
            if (xmlFile.exists()) {
                xmlFile.delete();
            }

            File gameSchematicFolder = new File(schematicFolder, from);
            File newGameSchematicFolder = new File(schematicFolder, to);

            if (gameSchematicFolder.exists()) {
                gameSchematicFolder.renameTo(newGameSchematicFolder);
            }

            saveGame(game);
        } finally {
            renameLock.unlock();
        }
    }

    @Override
    public void deleteGame(Game game) {
        File gameFile = new File(xmlFolder, game.getName() + ".xml");
        if (gameFile.exists()) {
            gameFile.delete();
        }

        File floorDir = new File(schematicFolder, game.getName());
        for (File schematicFile : floorDir.listFiles(FLOOR_SCHEMATIC_FILTER)) {
            if (schematicFile.isFile()) {
                schematicFile.delete();
            }
        }

        if (floorDir.listFiles().length == 0) {
            floorDir.delete();
        }
    }

    @Override
    public void saveStatistics(Iterable<Statistic> iterable) throws SQLException {
        validateSqlDatabaseSetup();

        for (Statistic statistic : iterable) {
            saveStatistic(statistic);
        }
    }

    @Override
    public void saveStatistic(Statistic statistic) throws SQLException {
        validateSqlDatabaseSetup();

        sqlContext.writeObject(statistic);
    }

    @SuppressWarnings("deprecation")
    @Override
    public Statistic getStatistic(String playerName) throws Exception {
        GameProfile profile;

        try {
            profile = uuidManager.getProfile(playerName);
        } catch (ExecutionException e) {
            logger.log(Level.SEVERE,
                    "Could not retrieve player uuid from mojang api, using OfflinePlayer#getUniqueId()", e);
            OfflinePlayer player = Bukkit.getOfflinePlayer(playerName);
            profile = new GameProfile(player.getUniqueId(), player.getName());
        }

        Statistic statistic = getStatistic(profile.getUniqueIdentifier());
        statistic.setLastName(playerName);
        return statistic;
    }

    @Override
    public Statistic getStatistic(UUID uuid) throws Exception {
        validateSqlDatabaseSetup();

        Statistic statistic = statisticCache.get(uuid);
        return statistic;
    }

    @SuppressWarnings("deprecation")
    @Override
    public Integer getStatisticRank(String playerName) throws Exception {
        GameProfile profile;

        try {
            profile = uuidManager.getProfile(playerName);
        } catch (ExecutionException e) {
            logger.log(Level.SEVERE,
                    "Could not receive player uuid from mojang api, using OfflinePlayer#getUniqueId()", e);
            OfflinePlayer player = Bukkit.getOfflinePlayer(playerName);
            profile = new GameProfile(player.getUniqueId(), player.getName());
        }

        return getStatisticRank(profile.getUniqueIdentifier());
    }

    @Override
    public Integer getStatisticRank(UUID uuid) throws Exception {
        validateSqlDatabaseSetup();

        rankLock.lock();
        Connection connection = null;

        int resultRank = 0;

        try {
            connection = sqlContext.getConnectionFromPool();

            final String uuidColumn = StatisticAccessor.ColumnContract.UUID;
            final String ratingColumn = StatisticAccessor.ColumnContract.RATING;
            final String tableName = StatisticAccessor.ColumnContract.TABLE_NAME;
            final String rankColumn = "rank";

            PreparedStatement rankStatement;
            SQLImplementation implementation = sqlContext.getImplementationType();

            if (implementation == SQLImplementation.MYSQL) {
                rankStatement = connection
                        .prepareStatement("SELECT " + rankColumn + " FROM " + "(SELECT @rn:=@rn+1 AS " + rankColumn
                                + ", " + uuidColumn + " FROM " + tableName + " ORDER BY " + ratingColumn + " DESC"
                                + ") AS t1, " + "(SELECT @rn:=0) t2 " + "WHERE " + uuidColumn + " = ?");
            } else if (implementation == SQLImplementation.SQLITE) {
                rankStatement = connection.prepareStatement("SELECT " + "(SELECT COUNT() + 1 FROM"
                        + "(SELECT DISTINCT " + ratingColumn + " FROM " + tableName + " AS t WHERE " + ratingColumn
                        + " > " + tableName + "." + ratingColumn + ")" + ") " + "AS " + rankColumn + " " + "FROM "
                        + tableName + " " + "WHERE " + uuidColumn + "=?");
            } else {
                throw new IllegalStateException(
                        "Unknown sql implementation " + (implementation == null ? null : implementation.name()));
            }

            rankStatement.setString(1, uuid.toString());
            ResultSet result = rankStatement.executeQuery();

            if (result.next()) {
                resultRank = result.getInt(rankColumn);
            }
        } finally {
            rankLock.unlock();

            if (connection != null) {
                connection.close();
            }
        }

        return resultRank;
    }

    @Override
    public Map<String, Statistic> getStatistics(String[] players) throws Exception {
        validateSqlDatabaseSetup();

        List<GameProfile> profiles = uuidManager.getProfiles(players);

        if (sqlContext == null) {
            throw new IllegalStateException("SQL database has not been initialized");
        }

        SQLQueryOptionsBuilder optionsBuilder = SQLQueryOptionsBuilder.newBuilder();
        ExpressionList where = optionsBuilder.where();

        for (int i = 0; i < profiles.size(); i++) {
            GameProfile profile = profiles.get(i);
            where.eq(StatisticAccessor.ColumnContract.UUID, profile.getUniqueIdentifier());

            if (i + 1 < profiles.size()) {
                where.or();
            }
        }

        List<Statistic> statisticResult = sqlContext.readSql(Statistic.class, optionsBuilder);
        Map<String, Statistic> statisticsMap = Maps.newHashMap();

        for (GameProfile profile : profiles) {
            Statistic found = null;

            for (Statistic statistic : statisticResult) {
                if (profile.getUniqueIdentifier().equals(statistic.getUniqueIdentifier())) {
                    found = statistic;
                }
            }

            if (found == null) {
                found = new Statistic(profile.getUniqueIdentifier());
            }

            found.setLastName(profile.getName());

            String name = null;
            for (String playerName : players) {
                if (playerName.equalsIgnoreCase(profile.getName())) {
                    name = playerName;
                }
            }

            if (name == null) {
                throw new IllegalStateException(
                        "Could not find name for " + profile.getName() + ": " + profile.getUniqueIdentifier());
            }

            statisticsMap.put(name, found);
            statisticCache.put(profile.getUniqueIdentifier(), found);
        }

        return statisticsMap;
    }

    @Override
    public Map<String, Statistic> getTopStatistics(int offset, int limit) throws SQLException, ExecutionException {
        validateSqlDatabaseSetup();

        String limitStr = offset == 0 ? String.valueOf(limit) : offset + "," + limit;

        SQLQueryOptionsBuilder optionsBuilder = SQLQueryOptionsBuilder.newBuilder().limit(limitStr)
                .sortBy(StatisticAccessor.ColumnContract.RATING + " DESC");
        List<Statistic> result = sqlContext.readSql(Statistic.class, optionsBuilder);

        Map<String, Statistic> statisticMapping = Maps.newLinkedHashMap();
        for (Statistic statistic : result) {
            UUID uuid = statistic.getUniqueIdentifier();
            GameProfile profile;

            profile = uuidManager.getProfile(uuid);
            String name = profile.getName();
            if (name == null) {
                name = statistic.getLastName();
            }

            statisticMapping.put(name, statistic);
        }

        return statisticMapping;
    }

    @Override
    public void clearCache() {
        if (statisticCache != null) {
            statisticCache.asMap().clear();
        }
    }

    @Override
    public void forceCacheSave() throws SQLException {
        Collection<Statistic> cachedStatistics = statisticCache.asMap().values();
        saveStatistics(cachedStatistics);
    }

    @Override
    public void shutdownGracefully() {
        Collection<Statistic> statistics = statisticCache.asMap().values();

        try {
            saveStatistics(statistics);
        } catch (SQLException e) {
            throw new RuntimeException("Could not save cached statistics on reload: ", e);
        }

        release();
    }

    @Override
    public void release() {
        if (sqlContext != null) {
            sqlContext.release();
        }

        uuidManager = null;
        statisticCache = null;
    }

    private void validateSqlDatabaseSetup() {
        if (sqlContext == null) {
            throw new IllegalStateException("No statistic-database has been setup");
        }
    }

    public Map<UUID, Statistic> getCachedStatistics() {
        return ImmutableMap.copyOf(statisticCache.asMap());
    }

    public UUIDManager getUUIDManager() {
        return uuidManager;
    }

}