123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545 |
- package com.gmail.nossr50.database;
- import com.gmail.nossr50.config.AdvancedConfig;
- import com.gmail.nossr50.config.Config;
- import com.gmail.nossr50.datatypes.database.DatabaseType;
- import com.gmail.nossr50.datatypes.database.PlayerStat;
- import com.gmail.nossr50.datatypes.database.UpgradeType;
- import com.gmail.nossr50.datatypes.player.*;
- import com.gmail.nossr50.datatypes.skills.PrimarySkillType;
- import com.gmail.nossr50.datatypes.skills.SuperAbilityType;
- import com.gmail.nossr50.mcMMO;
- import com.gmail.nossr50.runnables.database.UUIDUpdateAsyncTask;
- import com.gmail.nossr50.util.Misc;
- import com.gmail.nossr50.util.experience.MMOExperienceBarManager;
- import com.gmail.nossr50.util.skills.SkillUtils;
- import com.gmail.nossr50.util.text.StringUtils;
- import com.google.common.collect.ImmutableMap;
- import com.neetgames.mcmmo.MobHealthBarType;
- import com.neetgames.mcmmo.UniqueDataType;
- import com.neetgames.mcmmo.exceptions.ProfileRetrievalException;
- import com.neetgames.mcmmo.player.MMOPlayerData;
- import com.neetgames.mcmmo.skill.RootSkill;
- import com.neetgames.mcmmo.skill.SkillBossBarState;
- import org.apache.commons.lang.NullArgumentException;
- import org.bukkit.OfflinePlayer;
- import org.bukkit.entity.Player;
- import org.jetbrains.annotations.NotNull;
- import org.jetbrains.annotations.Nullable;
- import java.io.*;
- import java.util.*;
- public final class FlatFileDatabaseManager extends AbstractDatabaseManager {
- public static final String FLATFILE_SPLIT_CHARACTER_REGEX = ":";
- public static final String NULL_VALUE = NULL_VALUE;
- private final HashMap<RootSkill, List<PlayerStat>> playerStatHash = new HashMap<>();
- private final List<PlayerStat> powerLevels = new ArrayList<>();
- private long lastUpdate = 0;
- private final long UPDATE_WAIT_TIME = 600000L; // 10 minutes
- private final File usersFile;
- private static final Object fileWritingLock = new Object();
- protected FlatFileDatabaseManager() {
- usersFile = new File(mcMMO.getUsersFilePath());
- checkStructure();
- updateLeaderboards();
- if (mcMMO.getUpgradeManager().shouldUpgrade(UpgradeType.ADD_UUIDS)) {
- new UUIDUpdateAsyncTask(mcMMO.p, getStoredUsers()).start();
- }
- }
- public void purgePowerlessUsers() {
- int purgedUsers = 0;
- mcMMO.p.getLogger().info("Purging powerless users...");
- BufferedReader in = null;
- FileWriter out = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- // This code is O(n) instead of O(n²)
- synchronized (fileWritingLock) {
- try {
- in = new BufferedReader(new FileReader(usersFilePath));
- StringBuilder writer = new StringBuilder();
- String line;
- while ((line = in.readLine()) != null) {
- String[] character = line.split(":");
- Map<PrimarySkillType, Integer> skills = getSkillMapFromLine(character);
- boolean powerless = true;
- for (int skill : skills.values()) {
- if (skill != 0) {
- powerless = false;
- break;
- }
- }
- // If they're still around, rewrite them to the file.
- if (!powerless) {
- writer.append(line).append("\r\n");
- }
- else {
- purgedUsers++;
- }
- }
- // Write the new file
- out = new FileWriter(usersFilePath);
- out.write(writer.toString());
- }
- catch (IOException e) {
- mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
- }
- finally {
- if (in != null) {
- try {
- in.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- if (out != null) {
- try {
- out.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- mcMMO.p.getLogger().info("Purged " + purgedUsers + " users from the database.");
- }
- public void purgeOldUsers() {
- int removedPlayers = 0;
- long currentTime = System.currentTimeMillis();
- mcMMO.p.getLogger().info("Purging old users...");
- BufferedReader in = null;
- FileWriter out = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- // This code is O(n) instead of O(n²)
- synchronized (fileWritingLock) {
- try {
- in = new BufferedReader(new FileReader(usersFilePath));
- StringBuilder writer = new StringBuilder();
- String line;
- while ((line = in.readLine()) != null) {
- String[] character = line.split(":");
- String name = character[FlatFileMappings.USERNAME];
- long lastPlayed = 0;
- boolean rewrite = false;
- try {
- lastPlayed = Long.parseLong(character[37]) * Misc.TIME_CONVERSION_FACTOR;
- }
- catch (NumberFormatException e) {
- e.printStackTrace();
- }
- if (lastPlayed == 0) {
- OfflinePlayer player = mcMMO.p.getServer().getOfflinePlayer(name);
- lastPlayed = player.getLastPlayed();
- rewrite = true;
- }
- if (currentTime - lastPlayed > PURGE_TIME) {
- removedPlayers++;
- }
- else {
- if (rewrite) {
- // Rewrite their data with a valid time
- character[37] = Long.toString(lastPlayed);
- String newLine = org.apache.commons.lang.StringUtils.join(character, ":");
- writer.append(newLine).append("\r\n");
- }
- else {
- writer.append(line).append("\r\n");
- }
- }
- }
- // Write the new file
- out = new FileWriter(usersFilePath);
- out.write(writer.toString());
- }
- catch (IOException e) {
- mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
- }
- finally {
- if (in != null) {
- try {
- in.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- if (out != null) {
- try {
- out.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- mcMMO.p.getLogger().info("Purged " + removedPlayers + " users from the database.");
- }
- public boolean removeUser(@NotNull String playerName, @Nullable UUID uuid) {
- //NOTE: UUID is unused for FlatFile for this interface implementation
- boolean worked = false;
- BufferedReader in = null;
- FileWriter out = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- synchronized (fileWritingLock) {
- try {
- in = new BufferedReader(new FileReader(usersFilePath));
- StringBuilder writer = new StringBuilder();
- String line;
- while ((line = in.readLine()) != null) {
- // Write out the same file but when we get to the player we want to remove, we skip his line.
- if (!worked && line.split(":")[FlatFileMappings.USERNAME].equalsIgnoreCase(playerName)) {
- mcMMO.p.getLogger().info("User found, removing...");
- worked = true;
- continue; // Skip the player
- }
- writer.append(line).append("\r\n");
- }
- out = new FileWriter(usersFilePath); // Write out the new file
- out.write(writer.toString());
- }
- catch (Exception e) {
- mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
- }
- finally {
- if (in != null) {
- try {
- in.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- if (out != null) {
- try {
- out.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- Misc.profileCleanup(playerName);
- return worked;
- }
- public boolean saveUser(@NotNull MMODataSnapshot dataSnapshot) {
- String playerName = dataSnapshot.getPlayerName();
- UUID uuid = dataSnapshot.getPlayerUUID();
- BufferedReader in = null;
- FileWriter out = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- synchronized (fileWritingLock) {
- try {
- // Open the file
- in = new BufferedReader(new FileReader(usersFilePath));
- StringBuilder writer = new StringBuilder();
- String line;
- boolean wroteUser = false;
- // While not at the end of the file
- while ((line = in.readLine()) != null) {
- // Read the line in and copy it to the output if it's not the player we want to edit
- String[] character = line.split(":");
- if (!character[FlatFileMappings.UUID_INDEX].equalsIgnoreCase(uuid.toString()) && !character[FlatFileMappings.USERNAME].equalsIgnoreCase(playerName)) {
- writer.append(line).append("\r\n");
- }
- else {
- // Otherwise write the new player information
- writeUserToLine(dataSnapshot, playerName, uuid, writer);
- wroteUser = true;
- }
- }
- /*
- * If we couldn't find the user in the DB we need to add him
- */
- if(!wroteUser)
- {
- writeUserToLine(dataSnapshot, playerName, uuid, writer);
- }
- // Write the new file
- out = new FileWriter(usersFilePath);
- out.write(writer.toString());
- return true;
- }
- catch (Exception e) {
- e.printStackTrace();
- return false;
- }
- finally {
- if (in != null) {
- try {
- in.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- if (out != null) {
- try {
- out.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- }
- private void writeUserToLine(@NotNull MMODataSnapshot mmoDataSnapshot, @NotNull String playerName, @NotNull UUID uuid, @NotNull StringBuilder writer) {
- ImmutableMap<PrimarySkillType, Integer> primarySkillLevelMap = mmoDataSnapshot.getSkillLevelValues();
- ImmutableMap<PrimarySkillType, Float> primarySkillExperienceValueMap = mmoDataSnapshot.getSkillExperienceValues();
- writer.append(playerName).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.MINING)).append(":");
- writer.append(":");
- writer.append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.MINING)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.WOODCUTTING)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.WOODCUTTING)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.REPAIR)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.UNARMED)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.HERBALISM)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.EXCAVATION)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.ARCHERY)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.SWORDS)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.AXES)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.ACROBATICS)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.REPAIR)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.UNARMED)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.HERBALISM)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.EXCAVATION)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.ARCHERY)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.SWORDS)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.AXES)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.ACROBATICS)).append(":");
- writer.append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.TAMING)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.TAMING)).append(":");
- writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.BERSERK)).append(":");
- writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.GIGA_DRILL_BREAKER)).append(":");
- writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.TREE_FELLER)).append(":");
- writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.GREEN_TERRA)).append(":");
- writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.SERRATED_STRIKES)).append(":");
- writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.SKULL_SPLITTER)).append(":");
- writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.SUPER_BREAKER)).append(":");
- writer.append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.FISHING)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.FISHING)).append(":");
- writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.BLAST_MINING)).append(":");
- writer.append(System.currentTimeMillis() / Misc.TIME_CONVERSION_FACTOR).append(":");
- writer.append(NULL_VALUE).append(":"); //Mob Health Bars are no longer based on player data
- writer.append(primarySkillLevelMap.get(PrimarySkillType.ALCHEMY)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.ALCHEMY)).append(":");
- writer.append(uuid != null ? uuid.toString() : NULL_VALUE).append(":");
- writer.append(mmoDataSnapshot.getScoreboardTipsShown()).append(":");
- writer.append(mmoDataSnapshot.getUniqueData(UniqueDataType.CHIMAERA_WING_DATS)).append(":");
- /*
- public static int SKILLS_TRIDENTS = 44;
- public static int EXP_TRIDENTS = 45;
- public static int SKILLS_CROSSBOWS = 46;
- public static int EXP_CROSSBOWS = 47;
- public static int BARSTATE_ACROBATICS = 48;
- public static int BARSTATE_ALCHEMY = 49;
- public static int BARSTATE_ARCHERY = 50;
- public static int BARSTATE_AXES = 51;
- public static int BARSTATE_EXCAVATION = 52;
- public static int BARSTATE_FISHING = 53;
- public static int BARSTATE_HERBALISM = 54;
- public static int BARSTATE_MINING = 55;
- public static int BARSTATE_REPAIR = 56;
- public static int BARSTATE_SALVAGE = 57;
- public static int BARSTATE_SMELTING = 58;
- public static int BARSTATE_SWORDS = 59;
- public static int BARSTATE_TAMING = 60;
- public static int BARSTATE_UNARMED = 61;
- public static int BARSTATE_WOODCUTTING = 62;
- public static int BARSTATE_TRIDENTS = 63;
- public static int BARSTATE_CROSSBOWS = 64;
- */
- writer.append(primarySkillLevelMap.get(PrimarySkillType.TRIDENTS)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.TRIDENTS)).append(":");
- writer.append(primarySkillLevelMap.get(PrimarySkillType.CROSSBOWS)).append(":");
- writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.CROSSBOWS)).append(":");
- //XPBar States
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.ACROBATICS).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.ALCHEMY).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.ARCHERY).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.AXES).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.EXCAVATION).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.FISHING).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.HERBALISM).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.MINING).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.REPAIR).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.SALVAGE).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.SMELTING).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.SWORDS).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.TAMING).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.UNARMED).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.WOODCUTTING).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.TRIDENTS).toString()).append(":");
- writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.CROSSBOWS).toString()).append(":");
- writer.append(0).append(":"); //archery super 1 cd
- writer.append(0).append(":"); //xbow super 1 cd
- writer.append(0).append(":"); //tridents super 1 cd
- writer.append(0).append(":"); //chatspy toggle
- writer.append(0).append(":"); //leaderboard ignored
- writer.append("\r\n");
- }
- @Override
- public @NotNull List<PlayerStat> readLeaderboard(@NotNull RootSkill skill, int pageNumber, int statsPerPage) {
- updateLeaderboards();
- List<PlayerStat> statsList = skill == null ? powerLevels : playerStatHash.get(skill);
- int fromIndex = (Math.max(pageNumber, 1) - 1) * statsPerPage;
- return statsList.subList(Math.min(fromIndex, statsList.size()), Math.min(fromIndex + statsPerPage, statsList.size()));
- }
- @Override
- public @NotNull Map<PrimarySkillType, Integer> readRank(@NotNull String playerName) {
- updateLeaderboards();
- Map<PrimarySkillType, Integer> skills = new HashMap<>();
- for (RootSkill rootSkill : PrimarySkillType.getImmutableCoreRootSkillSet()) {
- if(PrimarySkillType.isChildSkill(rootSkill))
- continue;
- skills.put(rootSkill, getPlayerRank(playerName, playerStatHash.get(rootSkill)));
- }
- skills.put(null, getPlayerRank(playerName, powerLevels));
- return skills;
- }
- @Override
- public void newUser(@NotNull String playerName, @NotNull UUID uuid) {
- BufferedWriter out = null;
- synchronized (fileWritingLock) {
- try {
- // Open the file to write the player
- out = new BufferedWriter(new FileWriter(mcMMO.getUsersFilePath(), true));
- String startingLevel = AdvancedConfig.getInstance().getStartingLevel() + ":";
- // Add the player to the end
- out.append(playerName).append(":");
- out.append(startingLevel); // Mining
- out.append(":");
- out.append(":");
- out.append("0:"); // Xp
- out.append(startingLevel); // Woodcutting
- out.append("0:"); // WoodCuttingXp
- out.append(startingLevel); // Repair
- out.append(startingLevel); // Unarmed
- out.append(startingLevel); // Herbalism
- out.append(startingLevel); // Excavation
- out.append(startingLevel); // Archery
- out.append(startingLevel); // Swords
- out.append(startingLevel); // Axes
- out.append(startingLevel); // Acrobatics
- out.append("0:"); // RepairXp
- out.append("0:"); // UnarmedXp
- out.append("0:"); // HerbalismXp
- out.append("0:"); // ExcavationXp
- out.append("0:"); // ArcheryXp
- out.append("0:"); // SwordsXp
- out.append("0:"); // AxesXp
- out.append("0:"); // AcrobaticsXp
- out.append(":");
- out.append(startingLevel); // Taming
- out.append("0:"); // TamingXp
- out.append("0:"); // DATS
- out.append("0:"); // DATS
- out.append("0:"); // DATS
- out.append("0:"); // DATS
- out.append("0:"); // DATS
- out.append("0:"); // DATS
- out.append("0:"); // DATS
- out.append(":");
- out.append(startingLevel); // Fishing
- out.append("0:"); // FishingXp
- out.append("0:"); // Blast Mining
- out.append(String.valueOf(System.currentTimeMillis() / Misc.TIME_CONVERSION_FACTOR)).append(":"); // LastLogin
- out.append(Config.getInstance().getMobHealthbarDefault().toString()).append(":"); // Mob Healthbar HUD
- out.append(startingLevel); // Alchemy
- out.append("0:"); // AlchemyXp
- out.append(uuid != null ? uuid.toString() : NULL_VALUE).append(":"); // UUID
- out.append("0:"); // Scoreboard tips shown
- out.append("0:"); // Chimaera Wing Dats
- out.append("0:"); // Tridents Skill Level
- out.append("0:"); // Tridents XP
- out.append("0:"); // Crossbow Skill Level
- out.append("0:"); // Crossbow XP Level
- //Barstates for the 15 currently existing skills by ordinal value
- out.append("NORMAL:"); // Acrobatics
- out.append("NORMAL:"); // Alchemy
- out.append("NORMAL:"); // Archery
- out.append("NORMAL:"); // Axes
- out.append("NORMAL:"); // Excavation
- out.append("NORMAL:"); // Fishing
- out.append("NORMAL:"); // Herbalism
- out.append("NORMAL:"); // Mining
- out.append("NORMAL:"); // Repair
- out.append("DISABLED:"); // Salvage
- out.append("DISABLED:"); // Smelting
- out.append("NORMAL:"); // Swords
- out.append("NORMAL:"); // Taming
- out.append("NORMAL:"); // Unarmed
- out.append("NORMAL:"); // Woodcutting
- out.append("NORMAL:"); // Tridents
- out.append("NORMAL:"); // Crossbows
- //2.2.000+
- out.append("0:"); // arch super 1
- out.append("0:"); //xbow super 1
- out.append("0:"); //tridents super 1
- out.append("0:"); //chatspy toggle
- out.append("0:"); //leaderboard ignored toggle
- // Add more in the same format as the line above
- out.newLine();
- }
- catch (Exception e) {
- e.printStackTrace();
- }
- finally {
- if (out != null) {
- try {
- out.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- }
- @Override
- public @Nullable MMOPlayerData queryPlayerByName(@NotNull String playerName) throws ProfileRetrievalException {
- BufferedReader bufferedReader = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- //Retrieve player
- synchronized (fileWritingLock) {
- try {
- // Open the user file
- bufferedReader = new BufferedReader(new FileReader(usersFilePath));
- String currentLine;
- while ((currentLine = bufferedReader.readLine()) != null) {
- // Split the data which is stored as a string with : as break points
- String[] stringDataArray = currentLine.split(FLATFILE_SPLIT_CHARACTER_REGEX);
- //Search for matching name
- if (!stringDataArray[FlatFileMappings.USERNAME].equalsIgnoreCase(playerName)) {
- continue;
- }
- //We found our player, load the data
- return loadFromLine(stringDataArray);
- }
- throw new ProfileRetrievalException("Couldn't find a matching player in the database! Using name matching - " + playerName);
- }
- catch (Exception e) {
- e.printStackTrace();
- }
- //Cleanup resource leaks
- finally {
- if (bufferedReader != null) {
- try {
- bufferedReader.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- //Theoretically this statement should never be reached
- mcMMO.p.getLogger().severe("Critical failure in execution of loading player from DB, contact the devs!");
- return null;
- }
- public @Nullable MMOPlayerData queryPlayerDataByPlayer(@NotNull Player player) throws ProfileRetrievalException, NullArgumentException {
- return queryPlayerDataByUUID(player.getUniqueId(), player.getName());
- }
- /**
- * Queries by UUID will always have the current player name included as this method only gets executed when players join the server
- * The name will be used to update player names in the DB if the name has changed
- * There exists scenarios where players can share the same name in the DB, there is no code to account for this currently
- * @param uuid uuid to match
- * @param playerName used to overwrite playername values in the database if an existing value that is not equal to this one is found
- * @return the player profile if retrieved successfully, otherwise null
- * @throws ProfileRetrievalException
- * @throws NullArgumentException
- */
- public @Nullable MMOPlayerData queryPlayerDataByUUID(@NotNull UUID uuid, @NotNull String playerName) throws ProfileRetrievalException, NullArgumentException {
- BufferedReader bufferedReader = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- //Retrieve player
- synchronized (fileWritingLock) {
- try {
- // Open the user file
- bufferedReader = new BufferedReader(new FileReader(usersFilePath));
- String currentLine;
- while ((currentLine = bufferedReader.readLine()) != null) {
- // Split the data which is stored as a string with : as break points
- String[] stringDataArray = currentLine.split(FLATFILE_SPLIT_CHARACTER_REGEX);
- //Search for matching UUID
- if (!stringDataArray[FlatFileMappings.UUID_INDEX].equalsIgnoreCase(uuid.toString())) {
- continue;
- }
- //If the player has changed their name, we need to update it too
- if (!stringDataArray[FlatFileMappings.USERNAME].equalsIgnoreCase(playerName)) {
- mcMMO.p.getLogger().info("Name change detected: " + stringDataArray[FlatFileMappings.USERNAME] + " => " + playerName);
- stringDataArray[FlatFileMappings.USERNAME] = playerName;
- }
- //We found our player, load the data
- return loadFromLine(stringDataArray);
- }
- throw new ProfileRetrievalException("Couldn't find a matching player in the database! - "+playerName+", "+uuid.toString());
- }
- catch (Exception e) {
- e.printStackTrace();
- }
- //Cleanup resource leaks
- finally {
- if (bufferedReader != null) {
- try {
- bufferedReader.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- //Theoretically this statement should never be reached
- mcMMO.p.getLogger().severe("Critical failure in execution of loading player from DB, contact the devs!");
- return null;
- }
- public void convertUsers(@NotNull DatabaseManager destination) {
- BufferedReader in = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- int convertedUsers = 0;
- long startMillis = System.currentTimeMillis();
- synchronized (fileWritingLock) {
- try {
- // Open the user file
- in = new BufferedReader(new FileReader(usersFilePath));
- String line;
- while ((line = in.readLine()) != null) {
- String[] stringDataSplit = line.split(":");
- try {
- MMOPlayerData mmoPlayerData = loadFromLine(stringDataSplit);
- if(mmoPlayerData == null)
- continue;
- destination.saveUser(mcMMO.getUserManager().createPlayerDataSnapshot(mmoPlayerData));
- }
- catch (Exception e) {
- e.printStackTrace();
- }
- convertedUsers++;
- Misc.printProgress(convertedUsers, progressInterval, startMillis);
- }
- }
- catch (Exception e) {
- e.printStackTrace();
- }
- finally {
- if (in != null) {
- try {
- in.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- }
- public @NotNull List<String> getStoredUsers() {
- ArrayList<String> users = new ArrayList<>();
- BufferedReader in = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- synchronized (fileWritingLock) {
- try {
- // Open the user file
- in = new BufferedReader(new FileReader(usersFilePath));
- String line;
- while ((line = in.readLine()) != null) {
- String[] character = line.split(":");
- users.add(character[FlatFileMappings.USERNAME]);
- }
- }
- catch (Exception e) {
- e.printStackTrace();
- }
- finally {
- if (in != null) {
- try {
- in.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- return users;
- }
- /**
- * Update the leader boards.
- */
- private void updateLeaderboards() {
- // Only update FFS leaderboards every 10 minutes.. this puts a lot of strain on the server (depending on the size of the database) and should not be done frequently
- if (System.currentTimeMillis() < lastUpdate + UPDATE_WAIT_TIME) {
- return;
- }
- String usersFilePath = mcMMO.getUsersFilePath();
- lastUpdate = System.currentTimeMillis(); // Log when the last update was run
- powerLevels.clear(); // Clear old values from the power levels
- // Initialize lists
- List<PlayerStat> mining = new ArrayList<>();
- List<PlayerStat> woodcutting = new ArrayList<>();
- List<PlayerStat> herbalism = new ArrayList<>();
- List<PlayerStat> excavation = new ArrayList<>();
- List<PlayerStat> acrobatics = new ArrayList<>();
- List<PlayerStat> repair = new ArrayList<>();
- List<PlayerStat> swords = new ArrayList<>();
- List<PlayerStat> axes = new ArrayList<>();
- List<PlayerStat> archery = new ArrayList<>();
- List<PlayerStat> unarmed = new ArrayList<>();
- List<PlayerStat> taming = new ArrayList<>();
- List<PlayerStat> fishing = new ArrayList<>();
- List<PlayerStat> alchemy = new ArrayList<>();
- BufferedReader in = null;
- String playerName = null;
- // Read from the FlatFile database and fill our arrays with information
- synchronized (fileWritingLock) {
- try {
- in = new BufferedReader(new FileReader(usersFilePath));
- String line;
- while ((line = in.readLine()) != null) {
- String[] data = line.split(":");
- playerName = data[FlatFileMappings.USERNAME];
- int powerLevel = 0;
- Map<PrimarySkillType, Integer> skills = getSkillMapFromLine(data);
- powerLevel += putStat(acrobatics, playerName, skills.get(PrimarySkillType.ACROBATICS));
- powerLevel += putStat(alchemy, playerName, skills.get(PrimarySkillType.ALCHEMY));
- powerLevel += putStat(archery, playerName, skills.get(PrimarySkillType.ARCHERY));
- powerLevel += putStat(axes, playerName, skills.get(PrimarySkillType.AXES));
- powerLevel += putStat(excavation, playerName, skills.get(PrimarySkillType.EXCAVATION));
- powerLevel += putStat(fishing, playerName, skills.get(PrimarySkillType.FISHING));
- powerLevel += putStat(herbalism, playerName, skills.get(PrimarySkillType.HERBALISM));
- powerLevel += putStat(mining, playerName, skills.get(PrimarySkillType.MINING));
- powerLevel += putStat(repair, playerName, skills.get(PrimarySkillType.REPAIR));
- powerLevel += putStat(swords, playerName, skills.get(PrimarySkillType.SWORDS));
- powerLevel += putStat(taming, playerName, skills.get(PrimarySkillType.TAMING));
- powerLevel += putStat(unarmed, playerName, skills.get(PrimarySkillType.UNARMED));
- powerLevel += putStat(woodcutting, playerName, skills.get(PrimarySkillType.WOODCUTTING));
- powerLevel += putStat(woodcutting, playerName, skills.get(PrimarySkillType.CROSSBOWS));
- powerLevel += putStat(woodcutting, playerName, skills.get(PrimarySkillType.TRIDENTS));
- putStat(powerLevels, playerName, powerLevel);
- }
- }
- catch (Exception e) {
- mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " during user " + playerName + " (Are you sure you formatted it correctly?) " + e.toString());
- }
- finally {
- if (in != null) {
- try {
- in.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- SkillComparator c = new SkillComparator();
- mining.sort(c);
- woodcutting.sort(c);
- repair.sort(c);
- unarmed.sort(c);
- herbalism.sort(c);
- excavation.sort(c);
- archery.sort(c);
- swords.sort(c);
- axes.sort(c);
- acrobatics.sort(c);
- taming.sort(c);
- fishing.sort(c);
- alchemy.sort(c);
- powerLevels.sort(c);
- playerStatHash.put(PrimarySkillType.MINING, mining);
- playerStatHash.put(PrimarySkillType.WOODCUTTING, woodcutting);
- playerStatHash.put(PrimarySkillType.REPAIR, repair);
- playerStatHash.put(PrimarySkillType.UNARMED, unarmed);
- playerStatHash.put(PrimarySkillType.HERBALISM, herbalism);
- playerStatHash.put(PrimarySkillType.EXCAVATION, excavation);
- playerStatHash.put(PrimarySkillType.ARCHERY, archery);
- playerStatHash.put(PrimarySkillType.SWORDS, swords);
- playerStatHash.put(PrimarySkillType.AXES, axes);
- playerStatHash.put(PrimarySkillType.ACROBATICS, acrobatics);
- playerStatHash.put(PrimarySkillType.TAMING, taming);
- playerStatHash.put(PrimarySkillType.FISHING, fishing);
- playerStatHash.put(PrimarySkillType.ALCHEMY, alchemy);
- }
- /**
- * Checks that the file is present and valid
- */
- private void checkStructure() {
- if (usersFile.exists()) {
- BufferedReader in = null;
- FileWriter out = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- synchronized (fileWritingLock) {
- try {
- in = new BufferedReader(new FileReader(usersFilePath));
- StringBuilder writer = new StringBuilder();
- String line;
- HashSet<String> usernames = new HashSet<>();
- HashSet<String> players = new HashSet<>();
- while ((line = in.readLine()) != null) {
- // Remove empty lines from the file
- if (line.isEmpty()) {
- continue;
- }
- // Length checks depend on last stringDataArray being ':'
- if (line.charAt(line.length() - 1) != ':') {
- line = line.concat(":");
- }
- boolean updated = false;
- String[] stringDataArray = line.split(":");
- int originalLength = stringDataArray.length;
- // Prevent the same username from being present multiple times
- if (!usernames.add(stringDataArray[FlatFileMappings.USERNAME])) {
- stringDataArray[FlatFileMappings.USERNAME] = "_INVALID_OLD_USERNAME_'";
- updated = true;
- if (stringDataArray.length < FlatFileMappings.UUID_INDEX + 1 || stringDataArray[FlatFileMappings.UUID_INDEX].equals(NULL_VALUE)) {
- continue;
- }
- }
- if (stringDataArray.length < 33) {
- // Before Version 1.0 - Drop
- mcMMO.p.getLogger().warning("Dropping malformed or before version 1.0 line from database - " + line);
- continue;
- }
- String oldVersion = null;
- if (stringDataArray.length > 33 && !stringDataArray[33].isEmpty()) {
- // Removal of Spout Support
- // Version 1.4.07-dev2
- // commit 7bac0e2ca5143bce84dc160617fed97f0b1cb968
- stringDataArray[33] = "";
- oldVersion = "1.4.07";
- updated = true;
- }
- if (stringDataArray.length <= 33) {
- // Introduction of HUDType
- // Version 1.1.06
- // commit 78f79213cdd7190cd11ae54526f3b4ea42078e8a
- stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
- stringDataArray[stringDataArray.length - 1] = "";
- oldVersion = "1.1.06";
- updated = true;
- }
- if (stringDataArray.length <= 35) {
- // Introduction of Fishing
- // Version 1.2.00
- // commit a814b57311bc7734661109f0e77fc8bab3a0bd29
- stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 2);
- stringDataArray[stringDataArray.length - 1] = "0";
- stringDataArray[stringDataArray.length - 2] = "0";
- if (oldVersion == null) {
- oldVersion = "1.2.00";
- }
- updated = true;
- }
- if (stringDataArray.length <= 36) {
- // Introduction of Blast Mining cooldowns
- // Version 1.3.00-dev
- // commit fadbaf429d6b4764b8f1ad0efaa524a090e82ef5
- stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
- stringDataArray[stringDataArray.length - 1] = "0";
- if (oldVersion == null) {
- oldVersion = "1.3.00";
- }
- updated = true;
- }
- if (stringDataArray.length <= 37) {
- // Making old-purge work with flatfile
- // Version 1.4.00-dev
- // commmit 3f6c07ba6aaf44e388cc3b882cac3d8f51d0ac28
- // XXX Cannot create an OfflinePlayer at startup, use 0 and fix in purge
- stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
- stringDataArray[stringDataArray.length - 1] = "0";
- if (oldVersion == null) {
- oldVersion = "1.4.00";
- }
- updated = true;
- }
- if (stringDataArray.length <= 38) {
- // Addition of mob healthbars
- // Version 1.4.06
- // commit da29185b7dc7e0d992754bba555576d48fa08aa6
- stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
- stringDataArray[stringDataArray.length - 1] = Config.getInstance().getMobHealthbarDefault().toString();
- if (oldVersion == null) {
- oldVersion = "1.4.06";
- }
- updated = true;
- }
- if (stringDataArray.length <= 39) {
- // Addition of Alchemy
- // Version 1.4.08
- stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 2);
- stringDataArray[stringDataArray.length - 1] = "0";
- stringDataArray[stringDataArray.length - 2] = "0";
- if (oldVersion == null) {
- oldVersion = "1.4.08";
- }
- updated = true;
- }
- if (stringDataArray.length <= 41) {
- // Addition of UUIDs
- // Version 1.5.01
- // Add a value because otherwise it gets removed
- stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
- stringDataArray[stringDataArray.length - 1] = NULL_VALUE;
- if (oldVersion == null) {
- oldVersion = "1.5.01";
- }
- updated = true;
- }
- if (stringDataArray.length <= 42) {
- // Addition of scoreboard tips auto disable
- // Version 1.5.02
- stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
- stringDataArray[stringDataArray.length - 1] = "0";
- if (oldVersion == null) {
- oldVersion = "1.5.02";
- }
- updated = true;
- }
- if(stringDataArray.length <= 43) {
- // Addition of Chimaera wing DATS
- stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
- stringDataArray[stringDataArray.length - 1] = "0";
- if (oldVersion == null) {
- oldVersion = "2.1.133";
- }
- updated = true;
- }
- if(stringDataArray.length <= FlatFileMappings.LENGTH_OF_SPLIT_DATA_ARRAY) {
- if (oldVersion == null) {
- oldVersion = "2.1.134";
- }
- stringDataArray = Arrays.copyOf(stringDataArray, FlatFileMappings.LENGTH_OF_SPLIT_DATA_ARRAY); // new array size
- /*
- public static int SKILLS_TRIDENTS = 44;
- public static int EXP_TRIDENTS = 45;
- public static int SKILLS_CROSSBOWS = 46;
- public static int EXP_CROSSBOWS = 47;
- public static int BARSTATE_ACROBATICS = 48;
- public static int BARSTATE_ALCHEMY = 49;
- public static int BARSTATE_ARCHERY = 50;
- public static int BARSTATE_AXES = 51;
- public static int BARSTATE_EXCAVATION = 52;
- public static int BARSTATE_FISHING = 53;
- public static int BARSTATE_HERBALISM = 54;
- public static int BARSTATE_MINING = 55;
- public static int BARSTATE_REPAIR = 56;
- public static int BARSTATE_SALVAGE = 57;
- public static int BARSTATE_SMELTING = 58;
- public static int BARSTATE_SWORDS = 59;
- public static int BARSTATE_TAMING = 60;
- public static int BARSTATE_UNARMED = 61;
- public static int BARSTATE_WOODCUTTING = 62;
- public static int BARSTATE_TRIDENTS = 63;
- public static int BARSTATE_CROSSBOWS = 64;
- */
- stringDataArray[FlatFileMappings.SKILLS_TRIDENTS] = "0"; //trident skill lvl
- stringDataArray[FlatFileMappings.EXP_TRIDENTS] = "0"; //trident xp value
- stringDataArray[FlatFileMappings.SKILLS_CROSSBOWS] = "0"; //xbow skill lvl
- stringDataArray[FlatFileMappings.EXP_CROSSBOWS] = "0"; //xbow xp lvl
- //Barstates 48-64
- stringDataArray[FlatFileMappings.BARSTATE_ACROBATICS] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_ALCHEMY] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_ARCHERY] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_AXES] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_EXCAVATION] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_FISHING] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_HERBALISM] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_MINING] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_REPAIR] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_SALVAGE] = "DISABLED"; //Child skills
- stringDataArray[FlatFileMappings.BARSTATE_SMELTING] = "DISABLED"; //Child skills
- stringDataArray[FlatFileMappings.BARSTATE_SWORDS] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_TAMING] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_UNARMED] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_WOODCUTTING] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_TRIDENTS] = "NORMAL";
- stringDataArray[FlatFileMappings.BARSTATE_CROSSBOWS] = "NORMAL";
- stringDataArray[FlatFileMappings.COOLDOWN_ARCHERY_SUPER_1] = "0";
- stringDataArray[FlatFileMappings.COOLDOWN_CROSSBOWS_SUPER_1] = "0";
- stringDataArray[FlatFileMappings.COOLDOWN_TRIDENTS_SUPER_1] = "0";
- stringDataArray[FlatFileMappings.CHATSPY_TOGGLE] = "0";
- stringDataArray[FlatFileMappings.LEADERBOARD_IGNORED] = "0";
- //This part is a bit odd because lastlogin already has a place in the index but it was unused
- stringDataArray[FlatFileMappings.LAST_LOGIN] = "0";
- updated = true;
- }
- //TODO: If new skills are added this needs to be rewritten
- if (Config.getInstance().getTruncateSkills()) {
- for (RootSkill rootSkill : PrimarySkillType.getImmutableCoreRootSkillSet()) {
- if(PrimarySkillType.isChildSkill(rootSkill))
- continue;
- int index = getSkillIndex(rootSkill);
- if (index >= stringDataArray.length) {
- continue;
- }
- int cap = Config.getInstance().getLevelCap(rootSkill);
- if (Integer.parseInt(stringDataArray[index]) > cap) {
- mcMMO.p.getLogger().warning("Truncating " + rootSkill.getRawSkillName() + " to configured max level for player " + stringDataArray[FlatFileMappings.USERNAME]);
- stringDataArray[index] = cap + "";
- updated = true;
- }
- }
- }
- boolean corrupted = false;
- //TODO: Update this corruption code, its super out of date
- //TODO: Update this corruption code, its super out of date
- //TODO: Update this corruption code, its super out of date
- //TODO: Update this corruption code, its super out of date
- //TODO: Update this corruption code, its super out of date
- //TODO: Update this corruption code, its super out of date
- //TODO: Update this corruption code, its super out of date
- //TODO: Update this corruption code, its super out of date
- //TODO: Update this corruption code, its super out of dated
- for (int i = 0; i < stringDataArray.length; i++) {
- //Sigh... this code
- if (stringDataArray[i].isEmpty() && !(i == 2 || i == 3 || i == 23 || i == 33 || i == 41)) {
- mcMMO.p.getLogger().info("Player data at index "+i+" appears to be empty, possible corruption of data has occurred.");
- corrupted = true;
- if (i == 37) {
- stringDataArray[i] = String.valueOf(System.currentTimeMillis() / Misc.TIME_CONVERSION_FACTOR);
- }
- else if (i == 38) {
- stringDataArray[i] = Config.getInstance().getMobHealthbarDefault().toString();
- }
- else {
- stringDataArray[i] = "0";
- }
- }
- if (StringUtils.isInt(stringDataArray[i]) && i == 38) {
- corrupted = true;
- stringDataArray[i] = Config.getInstance().getMobHealthbarDefault().toString();
- }
- if (!StringUtils.isInt(stringDataArray[i]) && !(i == 0 || i == 2 || i == 3 || i == 23 || i == 33 || i == 38 || i == 41)) {
- corrupted = true;
- stringDataArray[i] = "0";
- }
- }
- if (corrupted) {
- mcMMO.p.getLogger().info("Updating corrupted database line for player " + stringDataArray[FlatFileMappings.USERNAME]);
- }
- if (oldVersion != null) {
- mcMMO.p.getLogger().info("Updating database line from before version " + oldVersion + " for player " + stringDataArray[FlatFileMappings.USERNAME]);
- }
- updated |= corrupted;
- updated |= oldVersion != null;
- if (Config.getInstance().getTruncateSkills()) {
- Map<PrimarySkillType, Integer> skillsMap = getSkillMapFromLine(stringDataArray);
- for (RootSkill rootSkill : PrimarySkillType.getNonChildSkills()) {
- int cap = Config.getInstance().getLevelCap(rootSkill);
- if (skillsMap.get(rootSkill) > cap) {
- updated = true;
- }
- }
- }
- if (updated) {
- line = org.apache.commons.lang.StringUtils.join(stringDataArray, ":") + ":";
- }
- // Prevent the same player from being present multiple times
- if (stringDataArray.length == originalLength //If the length changed then the schema was expanded
- && (!stringDataArray[FlatFileMappings.UUID_INDEX].isEmpty()
- && !stringDataArray[FlatFileMappings.UUID_INDEX].equals(NULL_VALUE)
- && !players.add(stringDataArray[FlatFileMappings.UUID_INDEX]))) {
- continue;
- }
- writer.append(line).append("\r\n");
- }
- // Write the new file
- out = new FileWriter(usersFilePath);
- out.write(writer.toString());
- }
- catch (IOException e) {
- mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
- }
- finally {
- if (in != null) {
- try {
- in.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- if (out != null) {
- try {
- out.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_FISHING);
- mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_BLAST_MINING_COOLDOWN);
- mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_SQL_INDEXES);
- mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_MOB_HEALTHBARS);
- // mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.DROP_SQL_PARTY_NAMES);
- mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.DROP_SPOUT);
- mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_ALCHEMY);
- return;
- }
- usersFile.getParentFile().mkdir();
- try {
- mcMMO.p.getLogger().info("Creating mcmmo.users file...");
- new File(mcMMO.getUsersFilePath()).createNewFile();
- }
- catch (IOException e) {
- e.printStackTrace();
- }
- }
- private Integer getPlayerRank(String playerName, List<PlayerStat> statsList) {
- if (statsList == null) {
- return null;
- }
- int currentPos = 1;
- for (PlayerStat stat : statsList) {
- if (stat.name.equalsIgnoreCase(playerName)) {
- return currentPos;
- }
- currentPos++;
- }
- return null;
- }
- private int putStat(List<PlayerStat> statList, String playerName, int statValue) {
- statList.add(new PlayerStat(playerName, statValue));
- return statValue;
- }
- private static class SkillComparator implements Comparator<PlayerStat> {
- @Override
- public int compare(PlayerStat o1, PlayerStat o2) {
- return (o2.statVal - o1.statVal);
- }
- }
- private @Nullable MMOPlayerData loadFromLine(@NotNull String[] dataStrSplit) {
- MMODataBuilder playerDataBuilder = new MMODataBuilder();
- String username = dataStrSplit[FlatFileMappings.USERNAME];
- Map<PrimarySkillType, Integer> skillLevelMap = getSkillMapFromLine(dataStrSplit); // Skill levels
- Map<PrimarySkillType, Float> skillExperienceValueMap = new HashMap<>(); // Skill & XP
- Map<SuperAbilityType, Integer> skillAbilityDeactivationTimeStamp = new HashMap<>(); // Ability & Cooldown
- Map<UniqueDataType, Integer> uniquePlayerDataMap = new EnumMap<UniqueDataType, Integer>(UniqueDataType.class);
- Map<PrimarySkillType, SkillBossBarState> xpBarStateMap = new HashMap<>();
- int scoreboardTipsShown;
- //XP Values
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.TAMING, FlatFileMappings.EXP_TAMING, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.MINING, FlatFileMappings.EXP_MINING, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.REPAIR, FlatFileMappings.EXP_REPAIR, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.WOODCUTTING, FlatFileMappings.EXP_WOODCUTTING, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.UNARMED, FlatFileMappings.EXP_UNARMED, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.HERBALISM, FlatFileMappings.EXP_HERBALISM, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.EXCAVATION, FlatFileMappings.EXP_EXCAVATION, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.ARCHERY, FlatFileMappings.EXP_ARCHERY, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.SWORDS, FlatFileMappings.EXP_SWORDS, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.AXES, FlatFileMappings.EXP_AXES, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.ACROBATICS, FlatFileMappings.EXP_ACROBATICS, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.FISHING, FlatFileMappings.EXP_FISHING, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.ALCHEMY, FlatFileMappings.EXP_ALCHEMY, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.TRIDENTS, FlatFileMappings.EXP_TRIDENTS, username);
- tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.CROSSBOWS, FlatFileMappings.EXP_CROSSBOWS, username);
- //Set Skill XP
- // Taming - Unused
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.SUPER_BREAKER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_SUPER_BREAKER]));
- // Repair - Unused
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.TREE_FELLER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_TREE_FELLER]));
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.BERSERK, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_BERSERK]));
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.GREEN_TERRA, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_GREEN_TERRA]));
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.GIGA_DRILL_BREAKER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_GIGA_DRILL_BREAKER]));
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.SERRATED_STRIKES, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_SERRATED_STRIKES]));
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.SKULL_SPLITTER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_SKULL_SPLITTER]));
- // Acrobatics - Unused
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.BLAST_MINING, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_BLAST_MINING]));
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.ARCHERY_SUPER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_ARCHERY_SUPER_1]));
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.SUPER_SHOTGUN, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_CROSSBOWS_SUPER_1]));
- skillAbilityDeactivationTimeStamp.put(SuperAbilityType.TRIDENT_SUPER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_TRIDENTS_SUPER_1]));
- /*
- * Mob Health Bars are no longer saved per player and are ignored from the data
- */
- //Sometimes players are retrieved by name
- UUID playerUUID;
- try {
- playerUUID = UUID.fromString(dataStrSplit[FlatFileMappings.UUID_INDEX]);
- }
- catch (Exception e) {
- mcMMO.p.getLogger().severe("UUID not found for data entry, skipping entry");
- return null;
- }
- try {
- scoreboardTipsShown = Integer.parseInt(dataStrSplit[FlatFileMappings.SCOREBOARD_TIPS]);
- }
- catch (Exception e) {
- scoreboardTipsShown = 0;
- }
- try {
- uniquePlayerDataMap.put(UniqueDataType.CHIMAERA_WING_DATS, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_CHIMAERA_WING]));
- }
- catch (Exception e) {
- uniquePlayerDataMap.put(UniqueDataType.CHIMAERA_WING_DATS, 0);
- }
- try {
- xpBarStateMap.put(PrimarySkillType.ACROBATICS, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_ACROBATICS]));
- xpBarStateMap.put(PrimarySkillType.ALCHEMY, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_ALCHEMY]));
- xpBarStateMap.put(PrimarySkillType.ARCHERY, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_ARCHERY]));
- xpBarStateMap.put(PrimarySkillType.AXES, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_AXES]));
- xpBarStateMap.put(PrimarySkillType.EXCAVATION, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_EXCAVATION]));
- xpBarStateMap.put(PrimarySkillType.FISHING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_FISHING]));
- xpBarStateMap.put(PrimarySkillType.HERBALISM, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_HERBALISM]));
- xpBarStateMap.put(PrimarySkillType.MINING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_MINING]));
- xpBarStateMap.put(PrimarySkillType.REPAIR, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_REPAIR]));
- xpBarStateMap.put(PrimarySkillType.SALVAGE, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_SALVAGE]));
- xpBarStateMap.put(PrimarySkillType.SMELTING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_SMELTING]));
- xpBarStateMap.put(PrimarySkillType.SWORDS, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_SWORDS]));
- xpBarStateMap.put(PrimarySkillType.TAMING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_TAMING]));
- xpBarStateMap.put(PrimarySkillType.UNARMED, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_UNARMED]));
- xpBarStateMap.put(PrimarySkillType.WOODCUTTING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_WOODCUTTING]));
- xpBarStateMap.put(PrimarySkillType.TRIDENTS, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_TRIDENTS]));
- xpBarStateMap.put(PrimarySkillType.CROSSBOWS, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_CROSSBOWS]));
- } catch (Exception e) {
- xpBarStateMap = MMOExperienceBarManager.generateDefaultBarStateMap();
- }
- MMOPlayerData mmoPlayerData;
- try {
- //Set Player Data
- playerDataBuilder.setSkillLevelValues(skillLevelMap)
- .setSkillExperienceValues(skillExperienceValueMap)
- .setAbilityDeactivationTimestamps(skillAbilityDeactivationTimeStamp)
- // .setMobHealthBarType(mobHealthbarType)
- .setPlayerUUID(playerUUID)
- .setScoreboardTipsShown(scoreboardTipsShown)
- .setUniquePlayerData(uniquePlayerDataMap)
- .setBarStateMap(xpBarStateMap);
- //Build Data
- return playerDataBuilder.build();
- } catch (Exception e) {
- mcMMO.p.getLogger().severe("Critical failure when trying to construct persistent player data!");
- e.printStackTrace();
- return null;
- }
- }
- private @NotNull Map<PrimarySkillType, Integer> getSkillMapFromLine(@NotNull String[] splitDataArray) {
- Map<PrimarySkillType, Integer> skills = new EnumMap<>(PrimarySkillType.class); // Skill & Level
- String username = splitDataArray[FlatFileMappings.USERNAME];
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.TAMING, FlatFileMappings.SKILLS_TAMING, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.MINING, FlatFileMappings.SKILLS_MINING, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.REPAIR, FlatFileMappings.SKILLS_REPAIR, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.WOODCUTTING, FlatFileMappings.SKILLS_WOODCUTTING, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.UNARMED, FlatFileMappings.SKILLS_UNARMED, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.HERBALISM, FlatFileMappings.SKILLS_HERBALISM, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.EXCAVATION, FlatFileMappings.SKILLS_EXCAVATION, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.ARCHERY, FlatFileMappings.SKILLS_ARCHERY, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.SWORDS, FlatFileMappings.SKILLS_SWORDS, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.AXES, FlatFileMappings.SKILLS_AXES, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.ACROBATICS, FlatFileMappings.SKILLS_ACROBATICS, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.FISHING, FlatFileMappings.SKILLS_FISHING, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.ALCHEMY, FlatFileMappings.SKILLS_ALCHEMY, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.TRIDENTS, FlatFileMappings.SKILLS_TRIDENTS, username);
- tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.CROSSBOWS, FlatFileMappings.SKILLS_CROSSBOWS, username);
- return skills;
- }
- private void tryLoadSkillCooldownFromRawData(@NotNull Map<SuperAbilityType, Integer> cooldownMap, @NotNull String[] character, @NotNull SuperAbilityType superAbilityType, int cooldownSuperBreaker, @NotNull String userName) {
- try {
- cooldownMap.put(superAbilityType, Integer.valueOf(character[cooldownSuperBreaker]));
- } catch (NumberFormatException e) {
- mcMMO.p.getLogger().severe("Data corruption when trying to load the value for skill "+superAbilityType.toString()+" for player named " + userName+ " setting value to zero");
- e.printStackTrace();
- }
- }
- private void tryLoadSkillFloatValuesFromRawData(@NotNull Map<PrimarySkillType, Float> skillMap, @NotNull String[] character, @NotNull PrimarySkillType primarySkillType, int index, @NotNull String userName) {
- try {
- float valueFromString = Integer.parseInt(character[index]);
- skillMap.put(primarySkillType, valueFromString);
- } catch (NumberFormatException e) {
- skillMap.put(primarySkillType, 0F);
- mcMMO.p.getLogger().severe("Data corruption when trying to load the value for skill "+primarySkillType.toString()+" for player named " + userName+ " setting value to zero");
- e.printStackTrace();
- }
- }
- private void tryLoadSkillIntValuesFromRawData(@NotNull Map<PrimarySkillType, Integer> skillMap, @NotNull String[] character, @NotNull PrimarySkillType primarySkillType, int index, @NotNull String userName) {
- try {
- int valueFromString = Integer.parseInt(character[index]);
- skillMap.put(primarySkillType, valueFromString);
- } catch (NumberFormatException e) {
- skillMap.put(primarySkillType, 0);
- mcMMO.p.getLogger().severe("Data corruption when trying to load the value for skill "+primarySkillType.toString()+" for player named " + userName+ " setting value to zero");
- e.printStackTrace();
- }
- }
- public @NotNull DatabaseType getDatabaseType() {
- return DatabaseType.FLATFILE;
- }
- private int getSkillIndex(@NotNull PrimarySkillType primarySkillType) {
- switch (primarySkillType) {
- case ACROBATICS:
- return FlatFileMappings.SKILLS_ACROBATICS;
- case ALCHEMY:
- return FlatFileMappings.SKILLS_ALCHEMY;
- case ARCHERY:
- return FlatFileMappings.SKILLS_ARCHERY;
- case AXES:
- return FlatFileMappings.SKILLS_AXES;
- case EXCAVATION:
- return FlatFileMappings.SKILLS_EXCAVATION;
- case FISHING:
- return FlatFileMappings.SKILLS_FISHING;
- case HERBALISM:
- return FlatFileMappings.SKILLS_HERBALISM;
- case MINING:
- return FlatFileMappings.SKILLS_MINING;
- case REPAIR:
- return FlatFileMappings.SKILLS_REPAIR;
- case SWORDS:
- return FlatFileMappings.SKILLS_SWORDS;
- case TAMING:
- return FlatFileMappings.SKILLS_TAMING;
- case UNARMED:
- return FlatFileMappings.SKILLS_UNARMED;
- case WOODCUTTING:
- return FlatFileMappings.SKILLS_WOODCUTTING;
- case TRIDENTS:
- return FlatFileMappings.SKILLS_TRIDENTS;
- case CROSSBOWS:
- return FlatFileMappings.SKILLS_CROSSBOWS;
- default:
- throw new RuntimeException("Primary Skills only");
- }
- }
- @Override
- public void onDisable() { }
- public void resetMobHealthSettings() {
- BufferedReader in = null;
- FileWriter out = null;
- String usersFilePath = mcMMO.getUsersFilePath();
- synchronized (fileWritingLock) {
- try {
- in = new BufferedReader(new FileReader(usersFilePath));
- StringBuilder writer = new StringBuilder();
- String line;
- while ((line = in.readLine()) != null) {
- // Remove empty lines from the file
- if (line.isEmpty()) {
- continue;
- }
- String[] character = line.split(":");
-
- character[FlatFileMappings.HEALTHBAR] = Config.getInstance().getMobHealthbarDefault().toString();
-
- line = org.apache.commons.lang.StringUtils.join(character, ":") + ":";
- writer.append(line).append("\r\n");
- }
- // Write the new file
- out = new FileWriter(usersFilePath);
- out.write(writer.toString());
- }
- catch (IOException e) {
- mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
- }
- finally {
- if (in != null) {
- try {
- in.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- if (out != null) {
- try {
- out.close();
- }
- catch (IOException e) {
- // Ignore
- }
- }
- }
- }
- }
- }
|