FlatFileDatabaseManager.java 75 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545
  1. package com.gmail.nossr50.database;
  2. import com.gmail.nossr50.config.AdvancedConfig;
  3. import com.gmail.nossr50.config.Config;
  4. import com.gmail.nossr50.datatypes.database.DatabaseType;
  5. import com.gmail.nossr50.datatypes.database.PlayerStat;
  6. import com.gmail.nossr50.datatypes.database.UpgradeType;
  7. import com.gmail.nossr50.datatypes.player.*;
  8. import com.gmail.nossr50.datatypes.skills.PrimarySkillType;
  9. import com.gmail.nossr50.datatypes.skills.SuperAbilityType;
  10. import com.gmail.nossr50.mcMMO;
  11. import com.gmail.nossr50.runnables.database.UUIDUpdateAsyncTask;
  12. import com.gmail.nossr50.util.Misc;
  13. import com.gmail.nossr50.util.experience.MMOExperienceBarManager;
  14. import com.gmail.nossr50.util.skills.SkillUtils;
  15. import com.gmail.nossr50.util.text.StringUtils;
  16. import com.google.common.collect.ImmutableMap;
  17. import com.neetgames.mcmmo.MobHealthBarType;
  18. import com.neetgames.mcmmo.UniqueDataType;
  19. import com.neetgames.mcmmo.exceptions.ProfileRetrievalException;
  20. import com.neetgames.mcmmo.player.MMOPlayerData;
  21. import com.neetgames.mcmmo.skill.RootSkill;
  22. import com.neetgames.mcmmo.skill.SkillBossBarState;
  23. import org.apache.commons.lang.NullArgumentException;
  24. import org.bukkit.OfflinePlayer;
  25. import org.bukkit.entity.Player;
  26. import org.jetbrains.annotations.NotNull;
  27. import org.jetbrains.annotations.Nullable;
  28. import java.io.*;
  29. import java.util.*;
  30. public final class FlatFileDatabaseManager extends AbstractDatabaseManager {
  31. public static final String FLATFILE_SPLIT_CHARACTER_REGEX = ":";
  32. public static final String NULL_VALUE = NULL_VALUE;
  33. private final HashMap<RootSkill, List<PlayerStat>> playerStatHash = new HashMap<>();
  34. private final List<PlayerStat> powerLevels = new ArrayList<>();
  35. private long lastUpdate = 0;
  36. private final long UPDATE_WAIT_TIME = 600000L; // 10 minutes
  37. private final File usersFile;
  38. private static final Object fileWritingLock = new Object();
  39. protected FlatFileDatabaseManager() {
  40. usersFile = new File(mcMMO.getUsersFilePath());
  41. checkStructure();
  42. updateLeaderboards();
  43. if (mcMMO.getUpgradeManager().shouldUpgrade(UpgradeType.ADD_UUIDS)) {
  44. new UUIDUpdateAsyncTask(mcMMO.p, getStoredUsers()).start();
  45. }
  46. }
  47. public void purgePowerlessUsers() {
  48. int purgedUsers = 0;
  49. mcMMO.p.getLogger().info("Purging powerless users...");
  50. BufferedReader in = null;
  51. FileWriter out = null;
  52. String usersFilePath = mcMMO.getUsersFilePath();
  53. // This code is O(n) instead of O(n²)
  54. synchronized (fileWritingLock) {
  55. try {
  56. in = new BufferedReader(new FileReader(usersFilePath));
  57. StringBuilder writer = new StringBuilder();
  58. String line;
  59. while ((line = in.readLine()) != null) {
  60. String[] character = line.split(":");
  61. Map<PrimarySkillType, Integer> skills = getSkillMapFromLine(character);
  62. boolean powerless = true;
  63. for (int skill : skills.values()) {
  64. if (skill != 0) {
  65. powerless = false;
  66. break;
  67. }
  68. }
  69. // If they're still around, rewrite them to the file.
  70. if (!powerless) {
  71. writer.append(line).append("\r\n");
  72. }
  73. else {
  74. purgedUsers++;
  75. }
  76. }
  77. // Write the new file
  78. out = new FileWriter(usersFilePath);
  79. out.write(writer.toString());
  80. }
  81. catch (IOException e) {
  82. mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
  83. }
  84. finally {
  85. if (in != null) {
  86. try {
  87. in.close();
  88. }
  89. catch (IOException e) {
  90. // Ignore
  91. }
  92. }
  93. if (out != null) {
  94. try {
  95. out.close();
  96. }
  97. catch (IOException e) {
  98. // Ignore
  99. }
  100. }
  101. }
  102. }
  103. mcMMO.p.getLogger().info("Purged " + purgedUsers + " users from the database.");
  104. }
  105. public void purgeOldUsers() {
  106. int removedPlayers = 0;
  107. long currentTime = System.currentTimeMillis();
  108. mcMMO.p.getLogger().info("Purging old users...");
  109. BufferedReader in = null;
  110. FileWriter out = null;
  111. String usersFilePath = mcMMO.getUsersFilePath();
  112. // This code is O(n) instead of O(n²)
  113. synchronized (fileWritingLock) {
  114. try {
  115. in = new BufferedReader(new FileReader(usersFilePath));
  116. StringBuilder writer = new StringBuilder();
  117. String line;
  118. while ((line = in.readLine()) != null) {
  119. String[] character = line.split(":");
  120. String name = character[FlatFileMappings.USERNAME];
  121. long lastPlayed = 0;
  122. boolean rewrite = false;
  123. try {
  124. lastPlayed = Long.parseLong(character[37]) * Misc.TIME_CONVERSION_FACTOR;
  125. }
  126. catch (NumberFormatException e) {
  127. e.printStackTrace();
  128. }
  129. if (lastPlayed == 0) {
  130. OfflinePlayer player = mcMMO.p.getServer().getOfflinePlayer(name);
  131. lastPlayed = player.getLastPlayed();
  132. rewrite = true;
  133. }
  134. if (currentTime - lastPlayed > PURGE_TIME) {
  135. removedPlayers++;
  136. }
  137. else {
  138. if (rewrite) {
  139. // Rewrite their data with a valid time
  140. character[37] = Long.toString(lastPlayed);
  141. String newLine = org.apache.commons.lang.StringUtils.join(character, ":");
  142. writer.append(newLine).append("\r\n");
  143. }
  144. else {
  145. writer.append(line).append("\r\n");
  146. }
  147. }
  148. }
  149. // Write the new file
  150. out = new FileWriter(usersFilePath);
  151. out.write(writer.toString());
  152. }
  153. catch (IOException e) {
  154. mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
  155. }
  156. finally {
  157. if (in != null) {
  158. try {
  159. in.close();
  160. }
  161. catch (IOException e) {
  162. // Ignore
  163. }
  164. }
  165. if (out != null) {
  166. try {
  167. out.close();
  168. }
  169. catch (IOException e) {
  170. // Ignore
  171. }
  172. }
  173. }
  174. }
  175. mcMMO.p.getLogger().info("Purged " + removedPlayers + " users from the database.");
  176. }
  177. public boolean removeUser(@NotNull String playerName, @Nullable UUID uuid) {
  178. //NOTE: UUID is unused for FlatFile for this interface implementation
  179. boolean worked = false;
  180. BufferedReader in = null;
  181. FileWriter out = null;
  182. String usersFilePath = mcMMO.getUsersFilePath();
  183. synchronized (fileWritingLock) {
  184. try {
  185. in = new BufferedReader(new FileReader(usersFilePath));
  186. StringBuilder writer = new StringBuilder();
  187. String line;
  188. while ((line = in.readLine()) != null) {
  189. // Write out the same file but when we get to the player we want to remove, we skip his line.
  190. if (!worked && line.split(":")[FlatFileMappings.USERNAME].equalsIgnoreCase(playerName)) {
  191. mcMMO.p.getLogger().info("User found, removing...");
  192. worked = true;
  193. continue; // Skip the player
  194. }
  195. writer.append(line).append("\r\n");
  196. }
  197. out = new FileWriter(usersFilePath); // Write out the new file
  198. out.write(writer.toString());
  199. }
  200. catch (Exception e) {
  201. mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
  202. }
  203. finally {
  204. if (in != null) {
  205. try {
  206. in.close();
  207. }
  208. catch (IOException e) {
  209. // Ignore
  210. }
  211. }
  212. if (out != null) {
  213. try {
  214. out.close();
  215. }
  216. catch (IOException e) {
  217. // Ignore
  218. }
  219. }
  220. }
  221. }
  222. Misc.profileCleanup(playerName);
  223. return worked;
  224. }
  225. public boolean saveUser(@NotNull MMODataSnapshot dataSnapshot) {
  226. String playerName = dataSnapshot.getPlayerName();
  227. UUID uuid = dataSnapshot.getPlayerUUID();
  228. BufferedReader in = null;
  229. FileWriter out = null;
  230. String usersFilePath = mcMMO.getUsersFilePath();
  231. synchronized (fileWritingLock) {
  232. try {
  233. // Open the file
  234. in = new BufferedReader(new FileReader(usersFilePath));
  235. StringBuilder writer = new StringBuilder();
  236. String line;
  237. boolean wroteUser = false;
  238. // While not at the end of the file
  239. while ((line = in.readLine()) != null) {
  240. // Read the line in and copy it to the output if it's not the player we want to edit
  241. String[] character = line.split(":");
  242. if (!character[FlatFileMappings.UUID_INDEX].equalsIgnoreCase(uuid.toString()) && !character[FlatFileMappings.USERNAME].equalsIgnoreCase(playerName)) {
  243. writer.append(line).append("\r\n");
  244. }
  245. else {
  246. // Otherwise write the new player information
  247. writeUserToLine(dataSnapshot, playerName, uuid, writer);
  248. wroteUser = true;
  249. }
  250. }
  251. /*
  252. * If we couldn't find the user in the DB we need to add him
  253. */
  254. if(!wroteUser)
  255. {
  256. writeUserToLine(dataSnapshot, playerName, uuid, writer);
  257. }
  258. // Write the new file
  259. out = new FileWriter(usersFilePath);
  260. out.write(writer.toString());
  261. return true;
  262. }
  263. catch (Exception e) {
  264. e.printStackTrace();
  265. return false;
  266. }
  267. finally {
  268. if (in != null) {
  269. try {
  270. in.close();
  271. }
  272. catch (IOException e) {
  273. // Ignore
  274. }
  275. }
  276. if (out != null) {
  277. try {
  278. out.close();
  279. }
  280. catch (IOException e) {
  281. // Ignore
  282. }
  283. }
  284. }
  285. }
  286. }
  287. private void writeUserToLine(@NotNull MMODataSnapshot mmoDataSnapshot, @NotNull String playerName, @NotNull UUID uuid, @NotNull StringBuilder writer) {
  288. ImmutableMap<PrimarySkillType, Integer> primarySkillLevelMap = mmoDataSnapshot.getSkillLevelValues();
  289. ImmutableMap<PrimarySkillType, Float> primarySkillExperienceValueMap = mmoDataSnapshot.getSkillExperienceValues();
  290. writer.append(playerName).append(":");
  291. writer.append(primarySkillLevelMap.get(PrimarySkillType.MINING)).append(":");
  292. writer.append(":");
  293. writer.append(":");
  294. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.MINING)).append(":");
  295. writer.append(primarySkillLevelMap.get(PrimarySkillType.WOODCUTTING)).append(":");
  296. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.WOODCUTTING)).append(":");
  297. writer.append(primarySkillLevelMap.get(PrimarySkillType.REPAIR)).append(":");
  298. writer.append(primarySkillLevelMap.get(PrimarySkillType.UNARMED)).append(":");
  299. writer.append(primarySkillLevelMap.get(PrimarySkillType.HERBALISM)).append(":");
  300. writer.append(primarySkillLevelMap.get(PrimarySkillType.EXCAVATION)).append(":");
  301. writer.append(primarySkillLevelMap.get(PrimarySkillType.ARCHERY)).append(":");
  302. writer.append(primarySkillLevelMap.get(PrimarySkillType.SWORDS)).append(":");
  303. writer.append(primarySkillLevelMap.get(PrimarySkillType.AXES)).append(":");
  304. writer.append(primarySkillLevelMap.get(PrimarySkillType.ACROBATICS)).append(":");
  305. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.REPAIR)).append(":");
  306. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.UNARMED)).append(":");
  307. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.HERBALISM)).append(":");
  308. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.EXCAVATION)).append(":");
  309. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.ARCHERY)).append(":");
  310. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.SWORDS)).append(":");
  311. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.AXES)).append(":");
  312. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.ACROBATICS)).append(":");
  313. writer.append(":");
  314. writer.append(primarySkillLevelMap.get(PrimarySkillType.TAMING)).append(":");
  315. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.TAMING)).append(":");
  316. writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.BERSERK)).append(":");
  317. writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.GIGA_DRILL_BREAKER)).append(":");
  318. writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.TREE_FELLER)).append(":");
  319. writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.GREEN_TERRA)).append(":");
  320. writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.SERRATED_STRIKES)).append(":");
  321. writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.SKULL_SPLITTER)).append(":");
  322. writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.SUPER_BREAKER)).append(":");
  323. writer.append(":");
  324. writer.append(primarySkillLevelMap.get(PrimarySkillType.FISHING)).append(":");
  325. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.FISHING)).append(":");
  326. writer.append((int) mmoDataSnapshot.getAbilityDATS(SuperAbilityType.BLAST_MINING)).append(":");
  327. writer.append(System.currentTimeMillis() / Misc.TIME_CONVERSION_FACTOR).append(":");
  328. writer.append(NULL_VALUE).append(":"); //Mob Health Bars are no longer based on player data
  329. writer.append(primarySkillLevelMap.get(PrimarySkillType.ALCHEMY)).append(":");
  330. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.ALCHEMY)).append(":");
  331. writer.append(uuid != null ? uuid.toString() : NULL_VALUE).append(":");
  332. writer.append(mmoDataSnapshot.getScoreboardTipsShown()).append(":");
  333. writer.append(mmoDataSnapshot.getUniqueData(UniqueDataType.CHIMAERA_WING_DATS)).append(":");
  334. /*
  335. public static int SKILLS_TRIDENTS = 44;
  336. public static int EXP_TRIDENTS = 45;
  337. public static int SKILLS_CROSSBOWS = 46;
  338. public static int EXP_CROSSBOWS = 47;
  339. public static int BARSTATE_ACROBATICS = 48;
  340. public static int BARSTATE_ALCHEMY = 49;
  341. public static int BARSTATE_ARCHERY = 50;
  342. public static int BARSTATE_AXES = 51;
  343. public static int BARSTATE_EXCAVATION = 52;
  344. public static int BARSTATE_FISHING = 53;
  345. public static int BARSTATE_HERBALISM = 54;
  346. public static int BARSTATE_MINING = 55;
  347. public static int BARSTATE_REPAIR = 56;
  348. public static int BARSTATE_SALVAGE = 57;
  349. public static int BARSTATE_SMELTING = 58;
  350. public static int BARSTATE_SWORDS = 59;
  351. public static int BARSTATE_TAMING = 60;
  352. public static int BARSTATE_UNARMED = 61;
  353. public static int BARSTATE_WOODCUTTING = 62;
  354. public static int BARSTATE_TRIDENTS = 63;
  355. public static int BARSTATE_CROSSBOWS = 64;
  356. */
  357. writer.append(primarySkillLevelMap.get(PrimarySkillType.TRIDENTS)).append(":");
  358. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.TRIDENTS)).append(":");
  359. writer.append(primarySkillLevelMap.get(PrimarySkillType.CROSSBOWS)).append(":");
  360. writer.append(primarySkillExperienceValueMap.get(PrimarySkillType.CROSSBOWS)).append(":");
  361. //XPBar States
  362. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.ACROBATICS).toString()).append(":");
  363. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.ALCHEMY).toString()).append(":");
  364. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.ARCHERY).toString()).append(":");
  365. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.AXES).toString()).append(":");
  366. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.EXCAVATION).toString()).append(":");
  367. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.FISHING).toString()).append(":");
  368. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.HERBALISM).toString()).append(":");
  369. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.MINING).toString()).append(":");
  370. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.REPAIR).toString()).append(":");
  371. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.SALVAGE).toString()).append(":");
  372. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.SMELTING).toString()).append(":");
  373. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.SWORDS).toString()).append(":");
  374. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.TAMING).toString()).append(":");
  375. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.UNARMED).toString()).append(":");
  376. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.WOODCUTTING).toString()).append(":");
  377. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.TRIDENTS).toString()).append(":");
  378. writer.append(mmoDataSnapshot.getBarStateMap().get(PrimarySkillType.CROSSBOWS).toString()).append(":");
  379. writer.append(0).append(":"); //archery super 1 cd
  380. writer.append(0).append(":"); //xbow super 1 cd
  381. writer.append(0).append(":"); //tridents super 1 cd
  382. writer.append(0).append(":"); //chatspy toggle
  383. writer.append(0).append(":"); //leaderboard ignored
  384. writer.append("\r\n");
  385. }
  386. @Override
  387. public @NotNull List<PlayerStat> readLeaderboard(@NotNull RootSkill skill, int pageNumber, int statsPerPage) {
  388. updateLeaderboards();
  389. List<PlayerStat> statsList = skill == null ? powerLevels : playerStatHash.get(skill);
  390. int fromIndex = (Math.max(pageNumber, 1) - 1) * statsPerPage;
  391. return statsList.subList(Math.min(fromIndex, statsList.size()), Math.min(fromIndex + statsPerPage, statsList.size()));
  392. }
  393. @Override
  394. public @NotNull Map<PrimarySkillType, Integer> readRank(@NotNull String playerName) {
  395. updateLeaderboards();
  396. Map<PrimarySkillType, Integer> skills = new HashMap<>();
  397. for (RootSkill rootSkill : PrimarySkillType.getImmutableCoreRootSkillSet()) {
  398. if(PrimarySkillType.isChildSkill(rootSkill))
  399. continue;
  400. skills.put(rootSkill, getPlayerRank(playerName, playerStatHash.get(rootSkill)));
  401. }
  402. skills.put(null, getPlayerRank(playerName, powerLevels));
  403. return skills;
  404. }
  405. @Override
  406. public void newUser(@NotNull String playerName, @NotNull UUID uuid) {
  407. BufferedWriter out = null;
  408. synchronized (fileWritingLock) {
  409. try {
  410. // Open the file to write the player
  411. out = new BufferedWriter(new FileWriter(mcMMO.getUsersFilePath(), true));
  412. String startingLevel = AdvancedConfig.getInstance().getStartingLevel() + ":";
  413. // Add the player to the end
  414. out.append(playerName).append(":");
  415. out.append(startingLevel); // Mining
  416. out.append(":");
  417. out.append(":");
  418. out.append("0:"); // Xp
  419. out.append(startingLevel); // Woodcutting
  420. out.append("0:"); // WoodCuttingXp
  421. out.append(startingLevel); // Repair
  422. out.append(startingLevel); // Unarmed
  423. out.append(startingLevel); // Herbalism
  424. out.append(startingLevel); // Excavation
  425. out.append(startingLevel); // Archery
  426. out.append(startingLevel); // Swords
  427. out.append(startingLevel); // Axes
  428. out.append(startingLevel); // Acrobatics
  429. out.append("0:"); // RepairXp
  430. out.append("0:"); // UnarmedXp
  431. out.append("0:"); // HerbalismXp
  432. out.append("0:"); // ExcavationXp
  433. out.append("0:"); // ArcheryXp
  434. out.append("0:"); // SwordsXp
  435. out.append("0:"); // AxesXp
  436. out.append("0:"); // AcrobaticsXp
  437. out.append(":");
  438. out.append(startingLevel); // Taming
  439. out.append("0:"); // TamingXp
  440. out.append("0:"); // DATS
  441. out.append("0:"); // DATS
  442. out.append("0:"); // DATS
  443. out.append("0:"); // DATS
  444. out.append("0:"); // DATS
  445. out.append("0:"); // DATS
  446. out.append("0:"); // DATS
  447. out.append(":");
  448. out.append(startingLevel); // Fishing
  449. out.append("0:"); // FishingXp
  450. out.append("0:"); // Blast Mining
  451. out.append(String.valueOf(System.currentTimeMillis() / Misc.TIME_CONVERSION_FACTOR)).append(":"); // LastLogin
  452. out.append(Config.getInstance().getMobHealthbarDefault().toString()).append(":"); // Mob Healthbar HUD
  453. out.append(startingLevel); // Alchemy
  454. out.append("0:"); // AlchemyXp
  455. out.append(uuid != null ? uuid.toString() : NULL_VALUE).append(":"); // UUID
  456. out.append("0:"); // Scoreboard tips shown
  457. out.append("0:"); // Chimaera Wing Dats
  458. out.append("0:"); // Tridents Skill Level
  459. out.append("0:"); // Tridents XP
  460. out.append("0:"); // Crossbow Skill Level
  461. out.append("0:"); // Crossbow XP Level
  462. //Barstates for the 15 currently existing skills by ordinal value
  463. out.append("NORMAL:"); // Acrobatics
  464. out.append("NORMAL:"); // Alchemy
  465. out.append("NORMAL:"); // Archery
  466. out.append("NORMAL:"); // Axes
  467. out.append("NORMAL:"); // Excavation
  468. out.append("NORMAL:"); // Fishing
  469. out.append("NORMAL:"); // Herbalism
  470. out.append("NORMAL:"); // Mining
  471. out.append("NORMAL:"); // Repair
  472. out.append("DISABLED:"); // Salvage
  473. out.append("DISABLED:"); // Smelting
  474. out.append("NORMAL:"); // Swords
  475. out.append("NORMAL:"); // Taming
  476. out.append("NORMAL:"); // Unarmed
  477. out.append("NORMAL:"); // Woodcutting
  478. out.append("NORMAL:"); // Tridents
  479. out.append("NORMAL:"); // Crossbows
  480. //2.2.000+
  481. out.append("0:"); // arch super 1
  482. out.append("0:"); //xbow super 1
  483. out.append("0:"); //tridents super 1
  484. out.append("0:"); //chatspy toggle
  485. out.append("0:"); //leaderboard ignored toggle
  486. // Add more in the same format as the line above
  487. out.newLine();
  488. }
  489. catch (Exception e) {
  490. e.printStackTrace();
  491. }
  492. finally {
  493. if (out != null) {
  494. try {
  495. out.close();
  496. }
  497. catch (IOException e) {
  498. // Ignore
  499. }
  500. }
  501. }
  502. }
  503. }
  504. @Override
  505. public @Nullable MMOPlayerData queryPlayerByName(@NotNull String playerName) throws ProfileRetrievalException {
  506. BufferedReader bufferedReader = null;
  507. String usersFilePath = mcMMO.getUsersFilePath();
  508. //Retrieve player
  509. synchronized (fileWritingLock) {
  510. try {
  511. // Open the user file
  512. bufferedReader = new BufferedReader(new FileReader(usersFilePath));
  513. String currentLine;
  514. while ((currentLine = bufferedReader.readLine()) != null) {
  515. // Split the data which is stored as a string with : as break points
  516. String[] stringDataArray = currentLine.split(FLATFILE_SPLIT_CHARACTER_REGEX);
  517. //Search for matching name
  518. if (!stringDataArray[FlatFileMappings.USERNAME].equalsIgnoreCase(playerName)) {
  519. continue;
  520. }
  521. //We found our player, load the data
  522. return loadFromLine(stringDataArray);
  523. }
  524. throw new ProfileRetrievalException("Couldn't find a matching player in the database! Using name matching - " + playerName);
  525. }
  526. catch (Exception e) {
  527. e.printStackTrace();
  528. }
  529. //Cleanup resource leaks
  530. finally {
  531. if (bufferedReader != null) {
  532. try {
  533. bufferedReader.close();
  534. }
  535. catch (IOException e) {
  536. // Ignore
  537. }
  538. }
  539. }
  540. }
  541. //Theoretically this statement should never be reached
  542. mcMMO.p.getLogger().severe("Critical failure in execution of loading player from DB, contact the devs!");
  543. return null;
  544. }
  545. public @Nullable MMOPlayerData queryPlayerDataByPlayer(@NotNull Player player) throws ProfileRetrievalException, NullArgumentException {
  546. return queryPlayerDataByUUID(player.getUniqueId(), player.getName());
  547. }
  548. /**
  549. * Queries by UUID will always have the current player name included as this method only gets executed when players join the server
  550. * The name will be used to update player names in the DB if the name has changed
  551. * There exists scenarios where players can share the same name in the DB, there is no code to account for this currently
  552. * @param uuid uuid to match
  553. * @param playerName used to overwrite playername values in the database if an existing value that is not equal to this one is found
  554. * @return the player profile if retrieved successfully, otherwise null
  555. * @throws ProfileRetrievalException
  556. * @throws NullArgumentException
  557. */
  558. public @Nullable MMOPlayerData queryPlayerDataByUUID(@NotNull UUID uuid, @NotNull String playerName) throws ProfileRetrievalException, NullArgumentException {
  559. BufferedReader bufferedReader = null;
  560. String usersFilePath = mcMMO.getUsersFilePath();
  561. //Retrieve player
  562. synchronized (fileWritingLock) {
  563. try {
  564. // Open the user file
  565. bufferedReader = new BufferedReader(new FileReader(usersFilePath));
  566. String currentLine;
  567. while ((currentLine = bufferedReader.readLine()) != null) {
  568. // Split the data which is stored as a string with : as break points
  569. String[] stringDataArray = currentLine.split(FLATFILE_SPLIT_CHARACTER_REGEX);
  570. //Search for matching UUID
  571. if (!stringDataArray[FlatFileMappings.UUID_INDEX].equalsIgnoreCase(uuid.toString())) {
  572. continue;
  573. }
  574. //If the player has changed their name, we need to update it too
  575. if (!stringDataArray[FlatFileMappings.USERNAME].equalsIgnoreCase(playerName)) {
  576. mcMMO.p.getLogger().info("Name change detected: " + stringDataArray[FlatFileMappings.USERNAME] + " => " + playerName);
  577. stringDataArray[FlatFileMappings.USERNAME] = playerName;
  578. }
  579. //We found our player, load the data
  580. return loadFromLine(stringDataArray);
  581. }
  582. throw new ProfileRetrievalException("Couldn't find a matching player in the database! - "+playerName+", "+uuid.toString());
  583. }
  584. catch (Exception e) {
  585. e.printStackTrace();
  586. }
  587. //Cleanup resource leaks
  588. finally {
  589. if (bufferedReader != null) {
  590. try {
  591. bufferedReader.close();
  592. }
  593. catch (IOException e) {
  594. // Ignore
  595. }
  596. }
  597. }
  598. }
  599. //Theoretically this statement should never be reached
  600. mcMMO.p.getLogger().severe("Critical failure in execution of loading player from DB, contact the devs!");
  601. return null;
  602. }
  603. public void convertUsers(@NotNull DatabaseManager destination) {
  604. BufferedReader in = null;
  605. String usersFilePath = mcMMO.getUsersFilePath();
  606. int convertedUsers = 0;
  607. long startMillis = System.currentTimeMillis();
  608. synchronized (fileWritingLock) {
  609. try {
  610. // Open the user file
  611. in = new BufferedReader(new FileReader(usersFilePath));
  612. String line;
  613. while ((line = in.readLine()) != null) {
  614. String[] stringDataSplit = line.split(":");
  615. try {
  616. MMOPlayerData mmoPlayerData = loadFromLine(stringDataSplit);
  617. if(mmoPlayerData == null)
  618. continue;
  619. destination.saveUser(mcMMO.getUserManager().createPlayerDataSnapshot(mmoPlayerData));
  620. }
  621. catch (Exception e) {
  622. e.printStackTrace();
  623. }
  624. convertedUsers++;
  625. Misc.printProgress(convertedUsers, progressInterval, startMillis);
  626. }
  627. }
  628. catch (Exception e) {
  629. e.printStackTrace();
  630. }
  631. finally {
  632. if (in != null) {
  633. try {
  634. in.close();
  635. }
  636. catch (IOException e) {
  637. // Ignore
  638. }
  639. }
  640. }
  641. }
  642. }
  643. public @NotNull List<String> getStoredUsers() {
  644. ArrayList<String> users = new ArrayList<>();
  645. BufferedReader in = null;
  646. String usersFilePath = mcMMO.getUsersFilePath();
  647. synchronized (fileWritingLock) {
  648. try {
  649. // Open the user file
  650. in = new BufferedReader(new FileReader(usersFilePath));
  651. String line;
  652. while ((line = in.readLine()) != null) {
  653. String[] character = line.split(":");
  654. users.add(character[FlatFileMappings.USERNAME]);
  655. }
  656. }
  657. catch (Exception e) {
  658. e.printStackTrace();
  659. }
  660. finally {
  661. if (in != null) {
  662. try {
  663. in.close();
  664. }
  665. catch (IOException e) {
  666. // Ignore
  667. }
  668. }
  669. }
  670. }
  671. return users;
  672. }
  673. /**
  674. * Update the leader boards.
  675. */
  676. private void updateLeaderboards() {
  677. // 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
  678. if (System.currentTimeMillis() < lastUpdate + UPDATE_WAIT_TIME) {
  679. return;
  680. }
  681. String usersFilePath = mcMMO.getUsersFilePath();
  682. lastUpdate = System.currentTimeMillis(); // Log when the last update was run
  683. powerLevels.clear(); // Clear old values from the power levels
  684. // Initialize lists
  685. List<PlayerStat> mining = new ArrayList<>();
  686. List<PlayerStat> woodcutting = new ArrayList<>();
  687. List<PlayerStat> herbalism = new ArrayList<>();
  688. List<PlayerStat> excavation = new ArrayList<>();
  689. List<PlayerStat> acrobatics = new ArrayList<>();
  690. List<PlayerStat> repair = new ArrayList<>();
  691. List<PlayerStat> swords = new ArrayList<>();
  692. List<PlayerStat> axes = new ArrayList<>();
  693. List<PlayerStat> archery = new ArrayList<>();
  694. List<PlayerStat> unarmed = new ArrayList<>();
  695. List<PlayerStat> taming = new ArrayList<>();
  696. List<PlayerStat> fishing = new ArrayList<>();
  697. List<PlayerStat> alchemy = new ArrayList<>();
  698. BufferedReader in = null;
  699. String playerName = null;
  700. // Read from the FlatFile database and fill our arrays with information
  701. synchronized (fileWritingLock) {
  702. try {
  703. in = new BufferedReader(new FileReader(usersFilePath));
  704. String line;
  705. while ((line = in.readLine()) != null) {
  706. String[] data = line.split(":");
  707. playerName = data[FlatFileMappings.USERNAME];
  708. int powerLevel = 0;
  709. Map<PrimarySkillType, Integer> skills = getSkillMapFromLine(data);
  710. powerLevel += putStat(acrobatics, playerName, skills.get(PrimarySkillType.ACROBATICS));
  711. powerLevel += putStat(alchemy, playerName, skills.get(PrimarySkillType.ALCHEMY));
  712. powerLevel += putStat(archery, playerName, skills.get(PrimarySkillType.ARCHERY));
  713. powerLevel += putStat(axes, playerName, skills.get(PrimarySkillType.AXES));
  714. powerLevel += putStat(excavation, playerName, skills.get(PrimarySkillType.EXCAVATION));
  715. powerLevel += putStat(fishing, playerName, skills.get(PrimarySkillType.FISHING));
  716. powerLevel += putStat(herbalism, playerName, skills.get(PrimarySkillType.HERBALISM));
  717. powerLevel += putStat(mining, playerName, skills.get(PrimarySkillType.MINING));
  718. powerLevel += putStat(repair, playerName, skills.get(PrimarySkillType.REPAIR));
  719. powerLevel += putStat(swords, playerName, skills.get(PrimarySkillType.SWORDS));
  720. powerLevel += putStat(taming, playerName, skills.get(PrimarySkillType.TAMING));
  721. powerLevel += putStat(unarmed, playerName, skills.get(PrimarySkillType.UNARMED));
  722. powerLevel += putStat(woodcutting, playerName, skills.get(PrimarySkillType.WOODCUTTING));
  723. powerLevel += putStat(woodcutting, playerName, skills.get(PrimarySkillType.CROSSBOWS));
  724. powerLevel += putStat(woodcutting, playerName, skills.get(PrimarySkillType.TRIDENTS));
  725. putStat(powerLevels, playerName, powerLevel);
  726. }
  727. }
  728. catch (Exception e) {
  729. mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " during user " + playerName + " (Are you sure you formatted it correctly?) " + e.toString());
  730. }
  731. finally {
  732. if (in != null) {
  733. try {
  734. in.close();
  735. }
  736. catch (IOException e) {
  737. // Ignore
  738. }
  739. }
  740. }
  741. }
  742. SkillComparator c = new SkillComparator();
  743. mining.sort(c);
  744. woodcutting.sort(c);
  745. repair.sort(c);
  746. unarmed.sort(c);
  747. herbalism.sort(c);
  748. excavation.sort(c);
  749. archery.sort(c);
  750. swords.sort(c);
  751. axes.sort(c);
  752. acrobatics.sort(c);
  753. taming.sort(c);
  754. fishing.sort(c);
  755. alchemy.sort(c);
  756. powerLevels.sort(c);
  757. playerStatHash.put(PrimarySkillType.MINING, mining);
  758. playerStatHash.put(PrimarySkillType.WOODCUTTING, woodcutting);
  759. playerStatHash.put(PrimarySkillType.REPAIR, repair);
  760. playerStatHash.put(PrimarySkillType.UNARMED, unarmed);
  761. playerStatHash.put(PrimarySkillType.HERBALISM, herbalism);
  762. playerStatHash.put(PrimarySkillType.EXCAVATION, excavation);
  763. playerStatHash.put(PrimarySkillType.ARCHERY, archery);
  764. playerStatHash.put(PrimarySkillType.SWORDS, swords);
  765. playerStatHash.put(PrimarySkillType.AXES, axes);
  766. playerStatHash.put(PrimarySkillType.ACROBATICS, acrobatics);
  767. playerStatHash.put(PrimarySkillType.TAMING, taming);
  768. playerStatHash.put(PrimarySkillType.FISHING, fishing);
  769. playerStatHash.put(PrimarySkillType.ALCHEMY, alchemy);
  770. }
  771. /**
  772. * Checks that the file is present and valid
  773. */
  774. private void checkStructure() {
  775. if (usersFile.exists()) {
  776. BufferedReader in = null;
  777. FileWriter out = null;
  778. String usersFilePath = mcMMO.getUsersFilePath();
  779. synchronized (fileWritingLock) {
  780. try {
  781. in = new BufferedReader(new FileReader(usersFilePath));
  782. StringBuilder writer = new StringBuilder();
  783. String line;
  784. HashSet<String> usernames = new HashSet<>();
  785. HashSet<String> players = new HashSet<>();
  786. while ((line = in.readLine()) != null) {
  787. // Remove empty lines from the file
  788. if (line.isEmpty()) {
  789. continue;
  790. }
  791. // Length checks depend on last stringDataArray being ':'
  792. if (line.charAt(line.length() - 1) != ':') {
  793. line = line.concat(":");
  794. }
  795. boolean updated = false;
  796. String[] stringDataArray = line.split(":");
  797. int originalLength = stringDataArray.length;
  798. // Prevent the same username from being present multiple times
  799. if (!usernames.add(stringDataArray[FlatFileMappings.USERNAME])) {
  800. stringDataArray[FlatFileMappings.USERNAME] = "_INVALID_OLD_USERNAME_'";
  801. updated = true;
  802. if (stringDataArray.length < FlatFileMappings.UUID_INDEX + 1 || stringDataArray[FlatFileMappings.UUID_INDEX].equals(NULL_VALUE)) {
  803. continue;
  804. }
  805. }
  806. if (stringDataArray.length < 33) {
  807. // Before Version 1.0 - Drop
  808. mcMMO.p.getLogger().warning("Dropping malformed or before version 1.0 line from database - " + line);
  809. continue;
  810. }
  811. String oldVersion = null;
  812. if (stringDataArray.length > 33 && !stringDataArray[33].isEmpty()) {
  813. // Removal of Spout Support
  814. // Version 1.4.07-dev2
  815. // commit 7bac0e2ca5143bce84dc160617fed97f0b1cb968
  816. stringDataArray[33] = "";
  817. oldVersion = "1.4.07";
  818. updated = true;
  819. }
  820. if (stringDataArray.length <= 33) {
  821. // Introduction of HUDType
  822. // Version 1.1.06
  823. // commit 78f79213cdd7190cd11ae54526f3b4ea42078e8a
  824. stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
  825. stringDataArray[stringDataArray.length - 1] = "";
  826. oldVersion = "1.1.06";
  827. updated = true;
  828. }
  829. if (stringDataArray.length <= 35) {
  830. // Introduction of Fishing
  831. // Version 1.2.00
  832. // commit a814b57311bc7734661109f0e77fc8bab3a0bd29
  833. stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 2);
  834. stringDataArray[stringDataArray.length - 1] = "0";
  835. stringDataArray[stringDataArray.length - 2] = "0";
  836. if (oldVersion == null) {
  837. oldVersion = "1.2.00";
  838. }
  839. updated = true;
  840. }
  841. if (stringDataArray.length <= 36) {
  842. // Introduction of Blast Mining cooldowns
  843. // Version 1.3.00-dev
  844. // commit fadbaf429d6b4764b8f1ad0efaa524a090e82ef5
  845. stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
  846. stringDataArray[stringDataArray.length - 1] = "0";
  847. if (oldVersion == null) {
  848. oldVersion = "1.3.00";
  849. }
  850. updated = true;
  851. }
  852. if (stringDataArray.length <= 37) {
  853. // Making old-purge work with flatfile
  854. // Version 1.4.00-dev
  855. // commmit 3f6c07ba6aaf44e388cc3b882cac3d8f51d0ac28
  856. // XXX Cannot create an OfflinePlayer at startup, use 0 and fix in purge
  857. stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
  858. stringDataArray[stringDataArray.length - 1] = "0";
  859. if (oldVersion == null) {
  860. oldVersion = "1.4.00";
  861. }
  862. updated = true;
  863. }
  864. if (stringDataArray.length <= 38) {
  865. // Addition of mob healthbars
  866. // Version 1.4.06
  867. // commit da29185b7dc7e0d992754bba555576d48fa08aa6
  868. stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
  869. stringDataArray[stringDataArray.length - 1] = Config.getInstance().getMobHealthbarDefault().toString();
  870. if (oldVersion == null) {
  871. oldVersion = "1.4.06";
  872. }
  873. updated = true;
  874. }
  875. if (stringDataArray.length <= 39) {
  876. // Addition of Alchemy
  877. // Version 1.4.08
  878. stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 2);
  879. stringDataArray[stringDataArray.length - 1] = "0";
  880. stringDataArray[stringDataArray.length - 2] = "0";
  881. if (oldVersion == null) {
  882. oldVersion = "1.4.08";
  883. }
  884. updated = true;
  885. }
  886. if (stringDataArray.length <= 41) {
  887. // Addition of UUIDs
  888. // Version 1.5.01
  889. // Add a value because otherwise it gets removed
  890. stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
  891. stringDataArray[stringDataArray.length - 1] = NULL_VALUE;
  892. if (oldVersion == null) {
  893. oldVersion = "1.5.01";
  894. }
  895. updated = true;
  896. }
  897. if (stringDataArray.length <= 42) {
  898. // Addition of scoreboard tips auto disable
  899. // Version 1.5.02
  900. stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
  901. stringDataArray[stringDataArray.length - 1] = "0";
  902. if (oldVersion == null) {
  903. oldVersion = "1.5.02";
  904. }
  905. updated = true;
  906. }
  907. if(stringDataArray.length <= 43) {
  908. // Addition of Chimaera wing DATS
  909. stringDataArray = Arrays.copyOf(stringDataArray, stringDataArray.length + 1);
  910. stringDataArray[stringDataArray.length - 1] = "0";
  911. if (oldVersion == null) {
  912. oldVersion = "2.1.133";
  913. }
  914. updated = true;
  915. }
  916. if(stringDataArray.length <= FlatFileMappings.LENGTH_OF_SPLIT_DATA_ARRAY) {
  917. if (oldVersion == null) {
  918. oldVersion = "2.1.134";
  919. }
  920. stringDataArray = Arrays.copyOf(stringDataArray, FlatFileMappings.LENGTH_OF_SPLIT_DATA_ARRAY); // new array size
  921. /*
  922. public static int SKILLS_TRIDENTS = 44;
  923. public static int EXP_TRIDENTS = 45;
  924. public static int SKILLS_CROSSBOWS = 46;
  925. public static int EXP_CROSSBOWS = 47;
  926. public static int BARSTATE_ACROBATICS = 48;
  927. public static int BARSTATE_ALCHEMY = 49;
  928. public static int BARSTATE_ARCHERY = 50;
  929. public static int BARSTATE_AXES = 51;
  930. public static int BARSTATE_EXCAVATION = 52;
  931. public static int BARSTATE_FISHING = 53;
  932. public static int BARSTATE_HERBALISM = 54;
  933. public static int BARSTATE_MINING = 55;
  934. public static int BARSTATE_REPAIR = 56;
  935. public static int BARSTATE_SALVAGE = 57;
  936. public static int BARSTATE_SMELTING = 58;
  937. public static int BARSTATE_SWORDS = 59;
  938. public static int BARSTATE_TAMING = 60;
  939. public static int BARSTATE_UNARMED = 61;
  940. public static int BARSTATE_WOODCUTTING = 62;
  941. public static int BARSTATE_TRIDENTS = 63;
  942. public static int BARSTATE_CROSSBOWS = 64;
  943. */
  944. stringDataArray[FlatFileMappings.SKILLS_TRIDENTS] = "0"; //trident skill lvl
  945. stringDataArray[FlatFileMappings.EXP_TRIDENTS] = "0"; //trident xp value
  946. stringDataArray[FlatFileMappings.SKILLS_CROSSBOWS] = "0"; //xbow skill lvl
  947. stringDataArray[FlatFileMappings.EXP_CROSSBOWS] = "0"; //xbow xp lvl
  948. //Barstates 48-64
  949. stringDataArray[FlatFileMappings.BARSTATE_ACROBATICS] = "NORMAL";
  950. stringDataArray[FlatFileMappings.BARSTATE_ALCHEMY] = "NORMAL";
  951. stringDataArray[FlatFileMappings.BARSTATE_ARCHERY] = "NORMAL";
  952. stringDataArray[FlatFileMappings.BARSTATE_AXES] = "NORMAL";
  953. stringDataArray[FlatFileMappings.BARSTATE_EXCAVATION] = "NORMAL";
  954. stringDataArray[FlatFileMappings.BARSTATE_FISHING] = "NORMAL";
  955. stringDataArray[FlatFileMappings.BARSTATE_HERBALISM] = "NORMAL";
  956. stringDataArray[FlatFileMappings.BARSTATE_MINING] = "NORMAL";
  957. stringDataArray[FlatFileMappings.BARSTATE_REPAIR] = "NORMAL";
  958. stringDataArray[FlatFileMappings.BARSTATE_SALVAGE] = "DISABLED"; //Child skills
  959. stringDataArray[FlatFileMappings.BARSTATE_SMELTING] = "DISABLED"; //Child skills
  960. stringDataArray[FlatFileMappings.BARSTATE_SWORDS] = "NORMAL";
  961. stringDataArray[FlatFileMappings.BARSTATE_TAMING] = "NORMAL";
  962. stringDataArray[FlatFileMappings.BARSTATE_UNARMED] = "NORMAL";
  963. stringDataArray[FlatFileMappings.BARSTATE_WOODCUTTING] = "NORMAL";
  964. stringDataArray[FlatFileMappings.BARSTATE_TRIDENTS] = "NORMAL";
  965. stringDataArray[FlatFileMappings.BARSTATE_CROSSBOWS] = "NORMAL";
  966. stringDataArray[FlatFileMappings.COOLDOWN_ARCHERY_SUPER_1] = "0";
  967. stringDataArray[FlatFileMappings.COOLDOWN_CROSSBOWS_SUPER_1] = "0";
  968. stringDataArray[FlatFileMappings.COOLDOWN_TRIDENTS_SUPER_1] = "0";
  969. stringDataArray[FlatFileMappings.CHATSPY_TOGGLE] = "0";
  970. stringDataArray[FlatFileMappings.LEADERBOARD_IGNORED] = "0";
  971. //This part is a bit odd because lastlogin already has a place in the index but it was unused
  972. stringDataArray[FlatFileMappings.LAST_LOGIN] = "0";
  973. updated = true;
  974. }
  975. //TODO: If new skills are added this needs to be rewritten
  976. if (Config.getInstance().getTruncateSkills()) {
  977. for (RootSkill rootSkill : PrimarySkillType.getImmutableCoreRootSkillSet()) {
  978. if(PrimarySkillType.isChildSkill(rootSkill))
  979. continue;
  980. int index = getSkillIndex(rootSkill);
  981. if (index >= stringDataArray.length) {
  982. continue;
  983. }
  984. int cap = Config.getInstance().getLevelCap(rootSkill);
  985. if (Integer.parseInt(stringDataArray[index]) > cap) {
  986. mcMMO.p.getLogger().warning("Truncating " + rootSkill.getRawSkillName() + " to configured max level for player " + stringDataArray[FlatFileMappings.USERNAME]);
  987. stringDataArray[index] = cap + "";
  988. updated = true;
  989. }
  990. }
  991. }
  992. boolean corrupted = false;
  993. //TODO: Update this corruption code, its super out of date
  994. //TODO: Update this corruption code, its super out of date
  995. //TODO: Update this corruption code, its super out of date
  996. //TODO: Update this corruption code, its super out of date
  997. //TODO: Update this corruption code, its super out of date
  998. //TODO: Update this corruption code, its super out of date
  999. //TODO: Update this corruption code, its super out of date
  1000. //TODO: Update this corruption code, its super out of date
  1001. //TODO: Update this corruption code, its super out of dated
  1002. for (int i = 0; i < stringDataArray.length; i++) {
  1003. //Sigh... this code
  1004. if (stringDataArray[i].isEmpty() && !(i == 2 || i == 3 || i == 23 || i == 33 || i == 41)) {
  1005. mcMMO.p.getLogger().info("Player data at index "+i+" appears to be empty, possible corruption of data has occurred.");
  1006. corrupted = true;
  1007. if (i == 37) {
  1008. stringDataArray[i] = String.valueOf(System.currentTimeMillis() / Misc.TIME_CONVERSION_FACTOR);
  1009. }
  1010. else if (i == 38) {
  1011. stringDataArray[i] = Config.getInstance().getMobHealthbarDefault().toString();
  1012. }
  1013. else {
  1014. stringDataArray[i] = "0";
  1015. }
  1016. }
  1017. if (StringUtils.isInt(stringDataArray[i]) && i == 38) {
  1018. corrupted = true;
  1019. stringDataArray[i] = Config.getInstance().getMobHealthbarDefault().toString();
  1020. }
  1021. if (!StringUtils.isInt(stringDataArray[i]) && !(i == 0 || i == 2 || i == 3 || i == 23 || i == 33 || i == 38 || i == 41)) {
  1022. corrupted = true;
  1023. stringDataArray[i] = "0";
  1024. }
  1025. }
  1026. if (corrupted) {
  1027. mcMMO.p.getLogger().info("Updating corrupted database line for player " + stringDataArray[FlatFileMappings.USERNAME]);
  1028. }
  1029. if (oldVersion != null) {
  1030. mcMMO.p.getLogger().info("Updating database line from before version " + oldVersion + " for player " + stringDataArray[FlatFileMappings.USERNAME]);
  1031. }
  1032. updated |= corrupted;
  1033. updated |= oldVersion != null;
  1034. if (Config.getInstance().getTruncateSkills()) {
  1035. Map<PrimarySkillType, Integer> skillsMap = getSkillMapFromLine(stringDataArray);
  1036. for (RootSkill rootSkill : PrimarySkillType.getNonChildSkills()) {
  1037. int cap = Config.getInstance().getLevelCap(rootSkill);
  1038. if (skillsMap.get(rootSkill) > cap) {
  1039. updated = true;
  1040. }
  1041. }
  1042. }
  1043. if (updated) {
  1044. line = org.apache.commons.lang.StringUtils.join(stringDataArray, ":") + ":";
  1045. }
  1046. // Prevent the same player from being present multiple times
  1047. if (stringDataArray.length == originalLength //If the length changed then the schema was expanded
  1048. && (!stringDataArray[FlatFileMappings.UUID_INDEX].isEmpty()
  1049. && !stringDataArray[FlatFileMappings.UUID_INDEX].equals(NULL_VALUE)
  1050. && !players.add(stringDataArray[FlatFileMappings.UUID_INDEX]))) {
  1051. continue;
  1052. }
  1053. writer.append(line).append("\r\n");
  1054. }
  1055. // Write the new file
  1056. out = new FileWriter(usersFilePath);
  1057. out.write(writer.toString());
  1058. }
  1059. catch (IOException e) {
  1060. mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
  1061. }
  1062. finally {
  1063. if (in != null) {
  1064. try {
  1065. in.close();
  1066. }
  1067. catch (IOException e) {
  1068. // Ignore
  1069. }
  1070. }
  1071. if (out != null) {
  1072. try {
  1073. out.close();
  1074. }
  1075. catch (IOException e) {
  1076. // Ignore
  1077. }
  1078. }
  1079. }
  1080. }
  1081. mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_FISHING);
  1082. mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_BLAST_MINING_COOLDOWN);
  1083. mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_SQL_INDEXES);
  1084. mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_MOB_HEALTHBARS);
  1085. // mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.DROP_SQL_PARTY_NAMES);
  1086. mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.DROP_SPOUT);
  1087. mcMMO.getUpgradeManager().setUpgradeCompleted(UpgradeType.ADD_ALCHEMY);
  1088. return;
  1089. }
  1090. usersFile.getParentFile().mkdir();
  1091. try {
  1092. mcMMO.p.getLogger().info("Creating mcmmo.users file...");
  1093. new File(mcMMO.getUsersFilePath()).createNewFile();
  1094. }
  1095. catch (IOException e) {
  1096. e.printStackTrace();
  1097. }
  1098. }
  1099. private Integer getPlayerRank(String playerName, List<PlayerStat> statsList) {
  1100. if (statsList == null) {
  1101. return null;
  1102. }
  1103. int currentPos = 1;
  1104. for (PlayerStat stat : statsList) {
  1105. if (stat.name.equalsIgnoreCase(playerName)) {
  1106. return currentPos;
  1107. }
  1108. currentPos++;
  1109. }
  1110. return null;
  1111. }
  1112. private int putStat(List<PlayerStat> statList, String playerName, int statValue) {
  1113. statList.add(new PlayerStat(playerName, statValue));
  1114. return statValue;
  1115. }
  1116. private static class SkillComparator implements Comparator<PlayerStat> {
  1117. @Override
  1118. public int compare(PlayerStat o1, PlayerStat o2) {
  1119. return (o2.statVal - o1.statVal);
  1120. }
  1121. }
  1122. private @Nullable MMOPlayerData loadFromLine(@NotNull String[] dataStrSplit) {
  1123. MMODataBuilder playerDataBuilder = new MMODataBuilder();
  1124. String username = dataStrSplit[FlatFileMappings.USERNAME];
  1125. Map<PrimarySkillType, Integer> skillLevelMap = getSkillMapFromLine(dataStrSplit); // Skill levels
  1126. Map<PrimarySkillType, Float> skillExperienceValueMap = new HashMap<>(); // Skill & XP
  1127. Map<SuperAbilityType, Integer> skillAbilityDeactivationTimeStamp = new HashMap<>(); // Ability & Cooldown
  1128. Map<UniqueDataType, Integer> uniquePlayerDataMap = new EnumMap<UniqueDataType, Integer>(UniqueDataType.class);
  1129. Map<PrimarySkillType, SkillBossBarState> xpBarStateMap = new HashMap<>();
  1130. int scoreboardTipsShown;
  1131. //XP Values
  1132. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.TAMING, FlatFileMappings.EXP_TAMING, username);
  1133. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.MINING, FlatFileMappings.EXP_MINING, username);
  1134. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.REPAIR, FlatFileMappings.EXP_REPAIR, username);
  1135. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.WOODCUTTING, FlatFileMappings.EXP_WOODCUTTING, username);
  1136. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.UNARMED, FlatFileMappings.EXP_UNARMED, username);
  1137. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.HERBALISM, FlatFileMappings.EXP_HERBALISM, username);
  1138. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.EXCAVATION, FlatFileMappings.EXP_EXCAVATION, username);
  1139. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.ARCHERY, FlatFileMappings.EXP_ARCHERY, username);
  1140. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.SWORDS, FlatFileMappings.EXP_SWORDS, username);
  1141. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.AXES, FlatFileMappings.EXP_AXES, username);
  1142. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.ACROBATICS, FlatFileMappings.EXP_ACROBATICS, username);
  1143. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.FISHING, FlatFileMappings.EXP_FISHING, username);
  1144. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.ALCHEMY, FlatFileMappings.EXP_ALCHEMY, username);
  1145. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.TRIDENTS, FlatFileMappings.EXP_TRIDENTS, username);
  1146. tryLoadSkillFloatValuesFromRawData(skillExperienceValueMap, dataStrSplit, PrimarySkillType.CROSSBOWS, FlatFileMappings.EXP_CROSSBOWS, username);
  1147. //Set Skill XP
  1148. // Taming - Unused
  1149. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.SUPER_BREAKER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_SUPER_BREAKER]));
  1150. // Repair - Unused
  1151. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.TREE_FELLER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_TREE_FELLER]));
  1152. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.BERSERK, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_BERSERK]));
  1153. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.GREEN_TERRA, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_GREEN_TERRA]));
  1154. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.GIGA_DRILL_BREAKER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_GIGA_DRILL_BREAKER]));
  1155. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.SERRATED_STRIKES, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_SERRATED_STRIKES]));
  1156. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.SKULL_SPLITTER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_SKULL_SPLITTER]));
  1157. // Acrobatics - Unused
  1158. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.BLAST_MINING, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_BLAST_MINING]));
  1159. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.ARCHERY_SUPER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_ARCHERY_SUPER_1]));
  1160. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.SUPER_SHOTGUN, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_CROSSBOWS_SUPER_1]));
  1161. skillAbilityDeactivationTimeStamp.put(SuperAbilityType.TRIDENT_SUPER, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_TRIDENTS_SUPER_1]));
  1162. /*
  1163. * Mob Health Bars are no longer saved per player and are ignored from the data
  1164. */
  1165. //Sometimes players are retrieved by name
  1166. UUID playerUUID;
  1167. try {
  1168. playerUUID = UUID.fromString(dataStrSplit[FlatFileMappings.UUID_INDEX]);
  1169. }
  1170. catch (Exception e) {
  1171. mcMMO.p.getLogger().severe("UUID not found for data entry, skipping entry");
  1172. return null;
  1173. }
  1174. try {
  1175. scoreboardTipsShown = Integer.parseInt(dataStrSplit[FlatFileMappings.SCOREBOARD_TIPS]);
  1176. }
  1177. catch (Exception e) {
  1178. scoreboardTipsShown = 0;
  1179. }
  1180. try {
  1181. uniquePlayerDataMap.put(UniqueDataType.CHIMAERA_WING_DATS, Integer.valueOf(dataStrSplit[FlatFileMappings.COOLDOWN_CHIMAERA_WING]));
  1182. }
  1183. catch (Exception e) {
  1184. uniquePlayerDataMap.put(UniqueDataType.CHIMAERA_WING_DATS, 0);
  1185. }
  1186. try {
  1187. xpBarStateMap.put(PrimarySkillType.ACROBATICS, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_ACROBATICS]));
  1188. xpBarStateMap.put(PrimarySkillType.ALCHEMY, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_ALCHEMY]));
  1189. xpBarStateMap.put(PrimarySkillType.ARCHERY, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_ARCHERY]));
  1190. xpBarStateMap.put(PrimarySkillType.AXES, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_AXES]));
  1191. xpBarStateMap.put(PrimarySkillType.EXCAVATION, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_EXCAVATION]));
  1192. xpBarStateMap.put(PrimarySkillType.FISHING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_FISHING]));
  1193. xpBarStateMap.put(PrimarySkillType.HERBALISM, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_HERBALISM]));
  1194. xpBarStateMap.put(PrimarySkillType.MINING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_MINING]));
  1195. xpBarStateMap.put(PrimarySkillType.REPAIR, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_REPAIR]));
  1196. xpBarStateMap.put(PrimarySkillType.SALVAGE, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_SALVAGE]));
  1197. xpBarStateMap.put(PrimarySkillType.SMELTING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_SMELTING]));
  1198. xpBarStateMap.put(PrimarySkillType.SWORDS, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_SWORDS]));
  1199. xpBarStateMap.put(PrimarySkillType.TAMING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_TAMING]));
  1200. xpBarStateMap.put(PrimarySkillType.UNARMED, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_UNARMED]));
  1201. xpBarStateMap.put(PrimarySkillType.WOODCUTTING, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_WOODCUTTING]));
  1202. xpBarStateMap.put(PrimarySkillType.TRIDENTS, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_TRIDENTS]));
  1203. xpBarStateMap.put(PrimarySkillType.CROSSBOWS, SkillUtils.asBarState(dataStrSplit[FlatFileMappings.BARSTATE_CROSSBOWS]));
  1204. } catch (Exception e) {
  1205. xpBarStateMap = MMOExperienceBarManager.generateDefaultBarStateMap();
  1206. }
  1207. MMOPlayerData mmoPlayerData;
  1208. try {
  1209. //Set Player Data
  1210. playerDataBuilder.setSkillLevelValues(skillLevelMap)
  1211. .setSkillExperienceValues(skillExperienceValueMap)
  1212. .setAbilityDeactivationTimestamps(skillAbilityDeactivationTimeStamp)
  1213. // .setMobHealthBarType(mobHealthbarType)
  1214. .setPlayerUUID(playerUUID)
  1215. .setScoreboardTipsShown(scoreboardTipsShown)
  1216. .setUniquePlayerData(uniquePlayerDataMap)
  1217. .setBarStateMap(xpBarStateMap);
  1218. //Build Data
  1219. return playerDataBuilder.build();
  1220. } catch (Exception e) {
  1221. mcMMO.p.getLogger().severe("Critical failure when trying to construct persistent player data!");
  1222. e.printStackTrace();
  1223. return null;
  1224. }
  1225. }
  1226. private @NotNull Map<PrimarySkillType, Integer> getSkillMapFromLine(@NotNull String[] splitDataArray) {
  1227. Map<PrimarySkillType, Integer> skills = new EnumMap<>(PrimarySkillType.class); // Skill & Level
  1228. String username = splitDataArray[FlatFileMappings.USERNAME];
  1229. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.TAMING, FlatFileMappings.SKILLS_TAMING, username);
  1230. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.MINING, FlatFileMappings.SKILLS_MINING, username);
  1231. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.REPAIR, FlatFileMappings.SKILLS_REPAIR, username);
  1232. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.WOODCUTTING, FlatFileMappings.SKILLS_WOODCUTTING, username);
  1233. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.UNARMED, FlatFileMappings.SKILLS_UNARMED, username);
  1234. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.HERBALISM, FlatFileMappings.SKILLS_HERBALISM, username);
  1235. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.EXCAVATION, FlatFileMappings.SKILLS_EXCAVATION, username);
  1236. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.ARCHERY, FlatFileMappings.SKILLS_ARCHERY, username);
  1237. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.SWORDS, FlatFileMappings.SKILLS_SWORDS, username);
  1238. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.AXES, FlatFileMappings.SKILLS_AXES, username);
  1239. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.ACROBATICS, FlatFileMappings.SKILLS_ACROBATICS, username);
  1240. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.FISHING, FlatFileMappings.SKILLS_FISHING, username);
  1241. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.ALCHEMY, FlatFileMappings.SKILLS_ALCHEMY, username);
  1242. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.TRIDENTS, FlatFileMappings.SKILLS_TRIDENTS, username);
  1243. tryLoadSkillIntValuesFromRawData(skills, splitDataArray, PrimarySkillType.CROSSBOWS, FlatFileMappings.SKILLS_CROSSBOWS, username);
  1244. return skills;
  1245. }
  1246. private void tryLoadSkillCooldownFromRawData(@NotNull Map<SuperAbilityType, Integer> cooldownMap, @NotNull String[] character, @NotNull SuperAbilityType superAbilityType, int cooldownSuperBreaker, @NotNull String userName) {
  1247. try {
  1248. cooldownMap.put(superAbilityType, Integer.valueOf(character[cooldownSuperBreaker]));
  1249. } catch (NumberFormatException e) {
  1250. mcMMO.p.getLogger().severe("Data corruption when trying to load the value for skill "+superAbilityType.toString()+" for player named " + userName+ " setting value to zero");
  1251. e.printStackTrace();
  1252. }
  1253. }
  1254. private void tryLoadSkillFloatValuesFromRawData(@NotNull Map<PrimarySkillType, Float> skillMap, @NotNull String[] character, @NotNull PrimarySkillType primarySkillType, int index, @NotNull String userName) {
  1255. try {
  1256. float valueFromString = Integer.parseInt(character[index]);
  1257. skillMap.put(primarySkillType, valueFromString);
  1258. } catch (NumberFormatException e) {
  1259. skillMap.put(primarySkillType, 0F);
  1260. mcMMO.p.getLogger().severe("Data corruption when trying to load the value for skill "+primarySkillType.toString()+" for player named " + userName+ " setting value to zero");
  1261. e.printStackTrace();
  1262. }
  1263. }
  1264. private void tryLoadSkillIntValuesFromRawData(@NotNull Map<PrimarySkillType, Integer> skillMap, @NotNull String[] character, @NotNull PrimarySkillType primarySkillType, int index, @NotNull String userName) {
  1265. try {
  1266. int valueFromString = Integer.parseInt(character[index]);
  1267. skillMap.put(primarySkillType, valueFromString);
  1268. } catch (NumberFormatException e) {
  1269. skillMap.put(primarySkillType, 0);
  1270. mcMMO.p.getLogger().severe("Data corruption when trying to load the value for skill "+primarySkillType.toString()+" for player named " + userName+ " setting value to zero");
  1271. e.printStackTrace();
  1272. }
  1273. }
  1274. public @NotNull DatabaseType getDatabaseType() {
  1275. return DatabaseType.FLATFILE;
  1276. }
  1277. private int getSkillIndex(@NotNull PrimarySkillType primarySkillType) {
  1278. switch (primarySkillType) {
  1279. case ACROBATICS:
  1280. return FlatFileMappings.SKILLS_ACROBATICS;
  1281. case ALCHEMY:
  1282. return FlatFileMappings.SKILLS_ALCHEMY;
  1283. case ARCHERY:
  1284. return FlatFileMappings.SKILLS_ARCHERY;
  1285. case AXES:
  1286. return FlatFileMappings.SKILLS_AXES;
  1287. case EXCAVATION:
  1288. return FlatFileMappings.SKILLS_EXCAVATION;
  1289. case FISHING:
  1290. return FlatFileMappings.SKILLS_FISHING;
  1291. case HERBALISM:
  1292. return FlatFileMappings.SKILLS_HERBALISM;
  1293. case MINING:
  1294. return FlatFileMappings.SKILLS_MINING;
  1295. case REPAIR:
  1296. return FlatFileMappings.SKILLS_REPAIR;
  1297. case SWORDS:
  1298. return FlatFileMappings.SKILLS_SWORDS;
  1299. case TAMING:
  1300. return FlatFileMappings.SKILLS_TAMING;
  1301. case UNARMED:
  1302. return FlatFileMappings.SKILLS_UNARMED;
  1303. case WOODCUTTING:
  1304. return FlatFileMappings.SKILLS_WOODCUTTING;
  1305. case TRIDENTS:
  1306. return FlatFileMappings.SKILLS_TRIDENTS;
  1307. case CROSSBOWS:
  1308. return FlatFileMappings.SKILLS_CROSSBOWS;
  1309. default:
  1310. throw new RuntimeException("Primary Skills only");
  1311. }
  1312. }
  1313. @Override
  1314. public void onDisable() { }
  1315. public void resetMobHealthSettings() {
  1316. BufferedReader in = null;
  1317. FileWriter out = null;
  1318. String usersFilePath = mcMMO.getUsersFilePath();
  1319. synchronized (fileWritingLock) {
  1320. try {
  1321. in = new BufferedReader(new FileReader(usersFilePath));
  1322. StringBuilder writer = new StringBuilder();
  1323. String line;
  1324. while ((line = in.readLine()) != null) {
  1325. // Remove empty lines from the file
  1326. if (line.isEmpty()) {
  1327. continue;
  1328. }
  1329. String[] character = line.split(":");
  1330. character[FlatFileMappings.HEALTHBAR] = Config.getInstance().getMobHealthbarDefault().toString();
  1331. line = org.apache.commons.lang.StringUtils.join(character, ":") + ":";
  1332. writer.append(line).append("\r\n");
  1333. }
  1334. // Write the new file
  1335. out = new FileWriter(usersFilePath);
  1336. out.write(writer.toString());
  1337. }
  1338. catch (IOException e) {
  1339. mcMMO.p.getLogger().severe("Exception while reading " + usersFilePath + " (Are you sure you formatted it correctly?)" + e.toString());
  1340. }
  1341. finally {
  1342. if (in != null) {
  1343. try {
  1344. in.close();
  1345. }
  1346. catch (IOException e) {
  1347. // Ignore
  1348. }
  1349. }
  1350. if (out != null) {
  1351. try {
  1352. out.close();
  1353. }
  1354. catch (IOException e) {
  1355. // Ignore
  1356. }
  1357. }
  1358. }
  1359. }
  1360. }
  1361. }