using MySql.Data.MySqlClient; using Dapper; using System; using System.Collections.Generic; using System.Linq; using FFXIVClassic.Common; using FFXIVClassic_Map_Server.utils; using FFXIVClassic_Map_Server.packets.send.player; using FFXIVClassic_Map_Server.dataobjects; using FFXIVClassic_Map_Server.Actors; using FFXIVClassic_Map_Server.actors.chara.player; namespace FFXIVClassic_Map_Server { class Database { public static uint GetUserIdFromSession(String sessionId) { uint id = 0; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); MySqlCommand cmd = new MySqlCommand("SELECT * FROM sessions WHERE id = @sessionId AND expiration > NOW()", conn); cmd.Parameters.AddWithValue("@sessionId", sessionId); using (MySqlDataReader Reader = cmd.ExecuteReader()) { while (Reader.Read()) { id = Reader.GetUInt32("userId"); } } } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } return id; } public static List GetNpcList() { using (var conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { List npcList = null; try { conn.Open(); npcList = conn.Query("SELECT * FROM npc_list").ToList(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } return npcList; } } public static Dictionary GetItemGamedata() { using (var conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { Dictionary gamedataItems = new Dictionary(); try { conn.Open(); string query = @" SELECT * FROM gamedata_items LEFT JOIN gamedata_items_equipment ON gamedata_items.catalogID = gamedata_items_equipment.catalogID LEFT JOIN gamedata_items_accessory ON gamedata_items.catalogID = gamedata_items_accessory.catalogID LEFT JOIN gamedata_items_armor ON gamedata_items.catalogID = gamedata_items_armor.catalogID LEFT JOIN gamedata_items_weapon ON gamedata_items.catalogID = gamedata_items_weapon.catalogID LEFT JOIN gamedata_items_graphics ON gamedata_items.catalogID = gamedata_items_graphics.catalogID "; MySqlCommand cmd = new MySqlCommand(query, conn); using (MySqlDataReader reader = cmd.ExecuteReader()) { while (reader.Read()) { uint id = reader.GetUInt32("catalogID"); Item item = null; if (Item.IsWeapon(id)) item = new WeaponItem(reader); else if (Item.IsArmor(id)) item = new ArmorItem(reader); else if (Item.IsAccessory(id)) item = new AccessoryItem(reader); else item = new Item(reader); gamedataItems.Add(item.catalogID, item); } } } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } return gamedataItems; } } public static void SavePlayerAppearance(Player player) { string query; MySqlCommand cmd; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); query = @" UPDATE characters_appearance SET mainHand = @mainHand, offHand = @offHand, head = @head, body = @body, legs = @legs, hands = @hands, feet = @feet, waist = @waist, leftFinger = @leftFinger, rightFinger = @rightFinger, leftEar = @leftEar, rightEar = @rightEar WHERE characterId = @charaId "; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charaId", player.actorId); cmd.Parameters.AddWithValue("@mainHand", player.appearanceIds[Character.MAINHAND]); cmd.Parameters.AddWithValue("@offHand", player.appearanceIds[Character.OFFHAND]); cmd.Parameters.AddWithValue("@head", player.appearanceIds[Character.HEADGEAR]); cmd.Parameters.AddWithValue("@body", player.appearanceIds[Character.BODYGEAR]); cmd.Parameters.AddWithValue("@legs", player.appearanceIds[Character.LEGSGEAR]); cmd.Parameters.AddWithValue("@hands", player.appearanceIds[Character.HANDSGEAR]); cmd.Parameters.AddWithValue("@feet", player.appearanceIds[Character.FEETGEAR]); cmd.Parameters.AddWithValue("@waist", player.appearanceIds[Character.WAISTGEAR]); cmd.Parameters.AddWithValue("@leftFinger", player.appearanceIds[Character.L_RINGFINGER]); cmd.Parameters.AddWithValue("@rightFinger", player.appearanceIds[Character.R_RINGFINGER]); cmd.Parameters.AddWithValue("@leftEar", player.appearanceIds[Character.L_EAR]); cmd.Parameters.AddWithValue("@rightEar", player.appearanceIds[Character.R_EAR]); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static void SavePlayerCurrentClass(Player player) { string query; MySqlCommand cmd; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); query = @" UPDATE characters_parametersave SET mainSkill = @classId, mainSkillLevel = @classLevel WHERE characterId = @charaId "; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charaId", player.actorId); cmd.Parameters.AddWithValue("@classId", player.charaWork.parameterSave.state_mainSkill[0]); cmd.Parameters.AddWithValue("@classLevel", player.charaWork.parameterSave.state_mainSkillLevel); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static void SavePlayerPosition(Player player) { string query; MySqlCommand cmd; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); query = @" UPDATE characters SET positionX = @x, positionY = @y, positionZ = @z, rotation = @rot, destinationZoneId = @destZone, destinationSpawnType = @destSpawn, currentZoneId = @zoneId, currentPrivateArea = @privateArea, currentPrivateAreaType = @privateAreaType WHERE id = @charaId "; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charaId", player.actorId); cmd.Parameters.AddWithValue("@x", player.positionX); cmd.Parameters.AddWithValue("@y", player.positionY); cmd.Parameters.AddWithValue("@z", player.positionZ); cmd.Parameters.AddWithValue("@rot", player.rotation); cmd.Parameters.AddWithValue("@zoneId", player.zoneId); cmd.Parameters.AddWithValue("@privateArea", player.privateArea); cmd.Parameters.AddWithValue("@privateAreaType", player.privateAreaType); cmd.Parameters.AddWithValue("@destZone", player.destinationZone); cmd.Parameters.AddWithValue("@destSpawn", player.destinationSpawnType); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static void SavePlayerPlayTime(Player player) { string query; MySqlCommand cmd; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); query = @" UPDATE characters SET playTime = @playtime WHERE id = @charaId "; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charaId", player.actorId); cmd.Parameters.AddWithValue("@playtime", player.GetPlayTime(true)); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static void SaveQuest(Player player, Quest quest) { int slot = player.GetQuestSlot(quest.actorId); if (slot == -1) { Program.Log.Error("Tried saving quest player didn't have: Player: {0:x}, QuestId: {0:x}", player.actorId, quest.actorId); return; } else SaveQuest(player, quest, slot); } public static void SaveQuest(Player player, Quest quest, int slot) { string query; MySqlCommand cmd; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); query = @" INSERT INTO characters_quest_scenario (characterId, slot, questId, currentPhase, questData, questFlags) VALUES (@charaId, @slot, @questId, @phase, @questData, @questFlags) ON DUPLICATE KEY UPDATE questId = @questId, currentPhase = @phase, questData = @questData, questFlags = @questFlags "; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charaId", player.actorId); cmd.Parameters.AddWithValue("@slot", slot); cmd.Parameters.AddWithValue("@questId", 0xFFFFF & quest.actorId); cmd.Parameters.AddWithValue("@phase", quest.GetPhase()); cmd.Parameters.AddWithValue("@questData", quest.GetSerializedQuestData()); cmd.Parameters.AddWithValue("@questFlags", quest.GetQuestFlags()); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static void RemoveQuest(Player player, uint questId) { string query; MySqlCommand cmd; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); query = @" DELETE FROM characters_quest_scenario WHERE characterId = @charaId and questId = @questId "; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charaId", player.actorId); cmd.Parameters.AddWithValue("@questId", 0xFFFFF & questId); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static void CompleteQuest(Player player, uint questId) { string query; MySqlCommand cmd; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); query = @" INSERT INTO characters_quest_completed (characterId, questId) VALUES (@charaId, @questId) ON DUPLICATE KEY UPDATE characterId=characterId "; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charaId", player.actorId); cmd.Parameters.AddWithValue("@questId", 0xFFFFF & questId); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static bool IsQuestCompleted(Player player, uint questId) { bool isCompleted = false; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); MySqlCommand cmd = new MySqlCommand("SELECT * FROM characters_quest_completed WHERE characterId = @charaId and questId = @questId", conn); cmd.Parameters.AddWithValue("@charaId", player.actorId); cmd.Parameters.AddWithValue("@questId", questId); isCompleted = cmd.ExecuteScalar() != null; } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } return isCompleted; } public static void LoadPlayerCharacter(Player player) { string query; MySqlCommand cmd; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); //Load basic info query = @" SELECT name, positionX, positionY, positionZ, rotation, actorState, currentZoneId, gcCurrent, gcLimsaRank, gcGridaniaRank, gcUldahRank, currentTitle, guardian, birthDay, birthMonth, initialTown, tribe, restBonus, achievementPoints, playTime, destinationZoneId, destinationSpawnType, currentPrivateArea, currentPrivateAreaType FROM characters WHERE id = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { if (reader.Read()) { player.displayNameId = 0xFFFFFFFF; player.customDisplayName = reader.GetString(0); player.oldPositionX = player.positionX = reader.GetFloat(1); player.oldPositionY = player.positionY = reader.GetFloat(2); player.oldPositionZ = player.positionZ = reader.GetFloat(3); player.oldRotation = player.rotation = reader.GetFloat(4); player.currentMainState = reader.GetUInt16(5); player.zoneId = reader.GetUInt32(6); player.isZoning = true; player.gcCurrent = reader.GetByte(7); player.gcRankLimsa = reader.GetByte(8); player.gcRankGridania = reader.GetByte(9); player.gcRankUldah = reader.GetByte(10); player.currentTitle = reader.GetUInt32(11); player.playerWork.guardian = reader.GetByte(12); player.playerWork.birthdayDay = reader.GetByte(13); player.playerWork.birthdayMonth = reader.GetByte(14); player.playerWork.initialTown = reader.GetByte(15); player.playerWork.tribe = reader.GetByte(16); player.playerWork.restBonusExpRate = reader.GetInt32(17); player.achievementPoints = reader.GetUInt32(18); player.playTime = reader.GetUInt32(19); player.destinationZone = reader.GetUInt32("destinationZoneId"); player.destinationSpawnType = reader.GetByte("destinationSpawnType"); if (!reader.IsDBNull(reader.GetOrdinal("currentPrivateArea"))) player.privateArea = reader.GetString("currentPrivateArea"); player.privateAreaType = reader.GetUInt32("currentPrivateAreaType"); if (player.destinationZone != 0) player.zoneId = player.destinationZone; if (player.privateArea != null && !player.privateArea.Equals("")) player.zone = Server.GetWorldManager().GetPrivateArea(player.zoneId, player.privateArea, player.privateAreaType); else player.zone = Server.GetWorldManager().GetZone(player.zoneId); } } //Get class levels query = @" SELECT pug, gla, mrd, arc, lnc, thm, cnj, crp, bsm, arm, gsm, ltw, wvr, alc, cul, min, btn, fsh FROM characters_class_levels WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { if (reader.Read()) { player.charaWork.battleSave.skillLevel[Player.CLASSID_PUG-1] = reader.GetInt16("pug"); player.charaWork.battleSave.skillLevel[Player.CLASSID_GLA - 1] = reader.GetInt16("gla"); player.charaWork.battleSave.skillLevel[Player.CLASSID_MRD - 1] = reader.GetInt16("mrd"); player.charaWork.battleSave.skillLevel[Player.CLASSID_ARC - 1] = reader.GetInt16("arc"); player.charaWork.battleSave.skillLevel[Player.CLASSID_LNC - 1] = reader.GetInt16("lnc"); player.charaWork.battleSave.skillLevel[Player.CLASSID_THM - 1] = reader.GetInt16("thm"); player.charaWork.battleSave.skillLevel[Player.CLASSID_CNJ - 1] = reader.GetInt16("cnj"); player.charaWork.battleSave.skillLevel[Player.CLASSID_CRP - 1] = reader.GetInt16("crp"); player.charaWork.battleSave.skillLevel[Player.CLASSID_BSM - 1] = reader.GetInt16("bsm"); player.charaWork.battleSave.skillLevel[Player.CLASSID_ARM - 1] = reader.GetInt16("arm"); player.charaWork.battleSave.skillLevel[Player.CLASSID_GSM - 1] = reader.GetInt16("gsm"); player.charaWork.battleSave.skillLevel[Player.CLASSID_LTW - 1] = reader.GetInt16("ltw"); player.charaWork.battleSave.skillLevel[Player.CLASSID_WVR - 1] = reader.GetInt16("wvr"); player.charaWork.battleSave.skillLevel[Player.CLASSID_ALC - 1] = reader.GetInt16("alc"); player.charaWork.battleSave.skillLevel[Player.CLASSID_CUL - 1] = reader.GetInt16("cul"); player.charaWork.battleSave.skillLevel[Player.CLASSID_MIN - 1] = reader.GetInt16("min"); player.charaWork.battleSave.skillLevel[Player.CLASSID_BTN - 1] = reader.GetInt16("btn"); player.charaWork.battleSave.skillLevel[Player.CLASSID_FSH - 1] = reader.GetInt16("fsh"); } } //Load Saved Parameters query = @" SELECT hp, hpMax, mp, mpMax, mainSkill FROM characters_parametersave WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { if (reader.Read()) { player.charaWork.parameterSave.hp[0] = reader.GetInt16(0); player.charaWork.parameterSave.hpMax[0] = reader.GetInt16(1); player.charaWork.parameterSave.mp = reader.GetInt16(2); player.charaWork.parameterSave.mpMax = reader.GetInt16(3); player.charaWork.parameterSave.state_mainSkill[0] = reader.GetByte(4); player.charaWork.parameterSave.state_mainSkillLevel = player.charaWork.battleSave.skillLevel[reader.GetByte(4) - 1]; } } //Load appearance query = @" SELECT baseId, size, voice, skinColor, hairStyle, hairColor, hairHighlightColor, eyeColor, characteristics, characteristicsColor, faceType, ears, faceMouth, faceFeatures, faceNose, faceEyeShape, faceIrisSize, faceEyebrows, mainHand, offHand, head, body, legs, hands, feet, waist, leftFinger, rightFinger, leftEar, rightEar FROM characters_appearance WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { if (reader.Read()) { if (reader.GetUInt32(0) == 0xFFFFFFFF) player.modelId = CharacterUtils.GetTribeModel(player.playerWork.tribe); else player.modelId = reader.GetUInt32(0); player.appearanceIds[Character.SIZE] = reader.GetByte(1); player.appearanceIds[Character.COLORINFO] = (uint)(reader.GetUInt16(3) | (reader.GetUInt16(5) << 10) | (reader.GetUInt16(7) << 20)); player.appearanceIds[Character.FACEINFO] = PrimitiveConversion.ToUInt32(CharacterUtils.GetFaceInfo(reader.GetByte(8), reader.GetByte(9), reader.GetByte(10), reader.GetByte(11), reader.GetByte(12), reader.GetByte(13), reader.GetByte(14), reader.GetByte(15), reader.GetByte(16), reader.GetByte(17))); player.appearanceIds[Character.HIGHLIGHT_HAIR] = (uint)(reader.GetUInt16(6) | reader.GetUInt16(4) << 10); player.appearanceIds[Character.VOICE] = reader.GetByte(2); player.appearanceIds[Character.MAINHAND] = reader.GetUInt32(18); player.appearanceIds[Character.OFFHAND] = reader.GetUInt32(19); player.appearanceIds[Character.HEADGEAR] = reader.GetUInt32(20); player.appearanceIds[Character.BODYGEAR] = reader.GetUInt32(21); player.appearanceIds[Character.LEGSGEAR] = reader.GetUInt32(22); player.appearanceIds[Character.HANDSGEAR] = reader.GetUInt32(23); player.appearanceIds[Character.FEETGEAR] = reader.GetUInt32(24); player.appearanceIds[Character.WAISTGEAR] = reader.GetUInt32(25); player.appearanceIds[Character.R_EAR] = reader.GetUInt32(26); player.appearanceIds[Character.L_EAR] = reader.GetUInt32(27); player.appearanceIds[Character.R_RINGFINGER] = reader.GetUInt32(28); player.appearanceIds[Character.L_RINGFINGER] = reader.GetUInt32(29); } } //Load Status Effects query = @" SELECT statusId, expireTime FROM characters_statuseffect WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { int count = 0; while (reader.Read()) { player.charaWork.status[count] = reader.GetUInt16(0); player.charaWork.statusShownTime[count] = reader.GetUInt32(1); } } //Load Chocobo query = @" SELECT hasChocobo, hasGoobbue, chocoboAppearance, chocoboName FROM characters_chocobo WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { if (reader.Read()) { player.hasChocobo = reader.GetBoolean(0); player.hasGoobbue = reader.GetBoolean(1); player.chocoboAppearance = reader.GetByte(2); player.chocoboName = reader.GetString(3); } } //Load Timers query = @" SELECT thousandmaws, dzemaeldarkhold, bowlofembers_hard, bowlofembers, thornmarch, aurumvale, cutterscry, battle_aleport, battle_hyrstmill, battle_goldenbazaar, howlingeye_hard, howlingeye, castrumnovum, bowlofembers_extreme, rivenroad, rivenroad_hard, behests, companybehests, returntimer, skirmish FROM characters_timers WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { if (reader.Read()) { for (int i = 0; i < player.timers.Length; i++) player.timers[i] = reader.GetUInt32(i); } } //Load Hotbar query = @" SELECT hotbarSlot, commandId, recastTime FROM characters_hotbar WHERE characterId = @charId AND classId = @classId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); cmd.Parameters.AddWithValue("@classId", player.charaWork.parameterSave.state_mainSkill[0]); using (MySqlDataReader reader = cmd.ExecuteReader()) { while (reader.Read()) { int index = reader.GetUInt16(0); player.charaWork.command[index+32] = reader.GetUInt32(1); player.charaWork.parameterSave.commandSlot_recastTime[index] = reader.GetUInt32(2); } } //Load Scenario Quests query = @" SELECT slot, questId, questData, questFlags, currentPhase FROM characters_quest_scenario WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { while (reader.Read()) { int index = reader.GetUInt16(0); player.playerWork.questScenario[index] = 0xA0F00000 | reader.GetUInt32(1); string questData = null; uint questFlags = 0; uint currentPhase = 0; if (!reader.IsDBNull(2)) questData = reader.GetString(2); else questData = "{}"; if (!reader.IsDBNull(3)) questFlags = reader.GetUInt32(3); else questFlags = 0; if (!reader.IsDBNull(4)) currentPhase = reader.GetUInt32(4); string questName = Server.GetStaticActors(player.playerWork.questScenario[index]).actorName; player.questScenario[index] = new Quest(player, player.playerWork.questScenario[index], questName, questData, questFlags, currentPhase); } } //Load Local Guildleves query = @" SELECT slot, questId, abandoned, completed FROM characters_quest_guildleve_local WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { while (reader.Read()) { int index = reader.GetUInt16(0); player.playerWork.questGuildleve[index] = 0xA0F00000 | reader.GetUInt32(1); } } //Load Regional Guildleve Quests query = @" SELECT slot, guildleveId, abandoned, completed FROM characters_quest_guildleve_regional WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { while (reader.Read()) { int index = reader.GetUInt16(0); player.work.guildleveId[index] = reader.GetUInt16(1); player.work.guildleveDone[index] = reader.GetBoolean(2); player.work.guildleveChecked[index] = reader.GetBoolean(3); } } //Load NPC Linkshell query = @" SELECT npcLinkshellId, isCalling, isExtra FROM characters_npclinkshell WHERE characterId = @charId"; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { while (reader.Read()) { int npcLSId = reader.GetUInt16(0); player.playerWork.npcLinkshellChatCalling[npcLSId] = reader.GetBoolean(1); player.playerWork.npcLinkshellChatExtra[npcLSId] = reader.GetBoolean(2); } } player.GetInventory(Inventory.NORMAL).InitList(GetInventory(player, 0, Inventory.NORMAL)); player.GetInventory(Inventory.KEYITEMS).InitList(GetInventory(player, 0, Inventory.KEYITEMS)); player.GetInventory(Inventory.CURRENCY).InitList(GetInventory(player, 0, Inventory.CURRENCY)); player.GetInventory(Inventory.BAZAAR).InitList(GetInventory(player, 0, Inventory.BAZAAR)); player.GetInventory(Inventory.MELDREQUEST).InitList(GetInventory(player, 0, Inventory.MELDREQUEST)); player.GetInventory(Inventory.LOOT).InitList(GetInventory(player, 0, Inventory.LOOT)); player.GetEquipment().SetEquipment(GetEquipment(player, player.charaWork.parameterSave.state_mainSkill[0])); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static InventoryItem[] GetEquipment(Player player, ushort classId) { InventoryItem[] equipment = new InventoryItem[player.GetEquipment().GetCapacity()]; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" SELECT equipSlot, itemId FROM characters_inventory_equipment WHERE characterId = @charId AND (classId = @classId OR classId = 0) ORDER BY equipSlot"; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); cmd.Parameters.AddWithValue("@classId", classId); using (MySqlDataReader reader = cmd.ExecuteReader()) { while (reader.Read()) { ushort equipSlot = reader.GetUInt16(0); ulong uniqueItemId = reader.GetUInt16(1); InventoryItem item = player.GetInventory(Inventory.NORMAL).GetItemById(uniqueItemId); equipment[equipSlot] = item; } } } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } return equipment; } public static void EquipItem(Player player, ushort equipSlot, ulong uniqueItemId) { using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" INSERT INTO characters_inventory_equipment (characterId, classId, equipSlot, itemId) VALUES (@characterId, @classId, @equipSlot, @uniqueItemId) ON DUPLICATE KEY UPDATE itemId=@uniqueItemId; "; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@characterId", player.actorId); cmd.Parameters.AddWithValue("@classId", (equipSlot == Equipment.SLOT_UNDERSHIRT || equipSlot == Equipment.SLOT_UNDERGARMENT) ? 0 : player.charaWork.parameterSave.state_mainSkill[0]); cmd.Parameters.AddWithValue("@equipSlot", equipSlot); cmd.Parameters.AddWithValue("@uniqueItemId", uniqueItemId); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static void UnequipItem(Player player, ushort equipSlot) { using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" DELETE FROM characters_inventory_equipment WHERE characterId = @characterId AND classId = @classId AND equipSlot = @equipSlot; "; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@characterId", player.actorId); cmd.Parameters.AddWithValue("@classId", player.charaWork.parameterSave.state_mainSkill[0]); cmd.Parameters.AddWithValue("@equipSlot", equipSlot); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static List GetInventory(Player player, uint slotOffset, uint type) { List items = new List(); using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" SELECT serverItemId, itemId, quantity, slot, itemType, quality, durability, spiritBind, materia1, materia2, materia3, materia4, materia5 FROM characters_inventory INNER JOIN server_items ON serverItemId = server_items.id WHERE characterId = @charId AND inventoryType = @type AND slot >= @slot ORDER BY slot"; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); cmd.Parameters.AddWithValue("@slot", slotOffset); cmd.Parameters.AddWithValue("@type", type); using (MySqlDataReader reader = cmd.ExecuteReader()) { while (reader.Read()) { uint uniqueId = reader.GetUInt32(0); uint itemId = reader.GetUInt32(1); int quantity = reader.GetInt32(2); ushort slot = reader.GetUInt16(3); byte itemType = reader.GetByte(4); byte qualityNumber = reader.GetByte(5); int durability = reader.GetInt32(6); ushort spiritBind = reader.GetUInt16(7); byte materia1 = reader.GetByte(8); byte materia2 = reader.GetByte(9); byte materia3 = reader.GetByte(10); byte materia4 = reader.GetByte(11); byte materia5 = reader.GetByte(12); items.Add(new InventoryItem(uniqueId, itemId, quantity, slot, itemType, qualityNumber, durability, spiritBind, materia1, materia2, materia3, materia4, materia5)); } } } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } return items; } public static InventoryItem AddItem(Player player, uint itemId, int quantity, byte quality, byte itemType, int durability, ushort type) { InventoryItem insertedItem = null; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" INSERT INTO server_items (itemId, quality, itemType, durability) VALUES (@itemId, @quality, @itemType, @durability); "; MySqlCommand cmd = new MySqlCommand(query, conn); string query2 = @" INSERT INTO characters_inventory (characterId, slot, inventoryType, serverItemId, quantity) SELECT @charId, IFNULL(MAX(SLOT)+1, 0), @inventoryType, LAST_INSERT_ID(), @quantity FROM characters_inventory WHERE characterId = @charId AND inventoryType = @inventoryType; "; MySqlCommand cmd2 = new MySqlCommand(query2, conn); cmd.Parameters.AddWithValue("@itemId", itemId); cmd.Parameters.AddWithValue("@quality", quality); cmd.Parameters.AddWithValue("@itemType", itemType); cmd.Parameters.AddWithValue("@durability", durability); cmd2.Parameters.AddWithValue("@charId", player.actorId); cmd2.Parameters.AddWithValue("@inventoryType", type); cmd2.Parameters.AddWithValue("@quantity", quantity); cmd.ExecuteNonQuery(); cmd2.ExecuteNonQuery(); insertedItem = new InventoryItem((uint)cmd.LastInsertedId, itemId, quantity, (ushort)player.GetInventory(type).GetNextEmptySlot(), itemType, quality, durability, 0, 0, 0, 0, 0, 0); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } return insertedItem; } public static void SetQuantity(Player player, uint slot, ushort type, int quantity) { using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" UPDATE characters_inventory SET quantity = @quantity WHERE characterId = @charId AND slot = @slot AND inventoryType = @type; "; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); cmd.Parameters.AddWithValue("@quantity", quantity); cmd.Parameters.AddWithValue("@slot", slot); cmd.Parameters.AddWithValue("@type", type); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static void RemoveItem(Player player, ulong serverItemId, ushort type) { using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}; Allow User Variables=True", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" SELECT slot INTO @slotToDelete FROM characters_inventory WHERE serverItemId = @serverItemId; UPDATE characters_inventory SET slot = slot - 1 WHERE characterId = @charId AND slot > @slotToDelete AND inventoryType = @type; DELETE FROM characters_inventory WHERE serverItemId = @serverItemId AND inventoryType = @type; DELETE FROM server_items WHERE id = @serverItemId; "; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); cmd.Parameters.AddWithValue("@serverItemId", serverItemId); cmd.Parameters.AddWithValue("@type", type); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static void RemoveItem(Player player, ushort slot, ushort type) { using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}; Allow User Variables=True", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" SELECT serverItemId INTO @serverItemId FROM characters_inventory WHERE characterId = @charId AND slot = @slot; DELETE FROM characters_inventory WHERE characterId = @charId AND slot = @slot AND inventoryType = @type; DELETE FROM server_items WHERE id = @serverItemId; UPDATE characters_inventory SET slot = slot - 1 WHERE characterId = @charId AND slot > @slot AND inventoryType = @type; "; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); cmd.Parameters.AddWithValue("@slot", slot); cmd.Parameters.AddWithValue("@type", type); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } public static SubPacket GetLatestAchievements(Player player) { uint[] latestAchievements = new uint[5]; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); //Load Last 5 Completed string query = @" SELECT characters_achievements.achievementId FROM characters_achievements INNER JOIN gamedata_achievements ON characters_achievements.achievementId = gamedata_achievements.achievementId WHERE characterId = @charId AND rewardPoints <> 0 ORDER BY timeDone LIMIT 5"; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { int count = 0; while (reader.Read()) { uint id = reader.GetUInt32(0); latestAchievements[count++] = id; } } } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } return SetLatestAchievementsPacket.BuildPacket(player.actorId, latestAchievements); } public static SubPacket GetAchievementsPacket(Player player) { SetCompletedAchievementsPacket cheevosPacket = new SetCompletedAchievementsPacket(); using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" SELECT packetOffsetId FROM characters_achievements INNER JOIN gamedata_achievements ON characters_achievements.achievementId = gamedata_achievements.achievementId WHERE characterId = @charId AND timeDone IS NOT NULL"; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charId", player.actorId); using (MySqlDataReader reader = cmd.ExecuteReader()) { while (reader.Read()) { uint offset = reader.GetUInt32(0); if (offset < 0 || offset >= cheevosPacket.achievementFlags.Length) { Program.Log.Error("SQL Error; achievement flag offset id out of range: " + offset); continue; } cheevosPacket.achievementFlags[offset] = true; } } } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } return cheevosPacket.BuildPacket(player.actorId); } public static bool CreateLinkshell(Player player, string lsName, ushort lsCrest) { bool success = false; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); string query = @" INSERT INTO server_linkshells (name, master, crest) VALUES (@lsName, @master, @crest) ; "; MySqlCommand cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@lsName", lsName); cmd.Parameters.AddWithValue("@master", player.actorId); cmd.Parameters.AddWithValue("@crest", lsCrest); cmd.ExecuteNonQuery(); success = true; } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } return success; } public static void SaveNpcLS(Player player, uint npcLSId, bool isCalling, bool isExtra) { string query; MySqlCommand cmd; using (MySqlConnection conn = new MySqlConnection(String.Format("Server={0}; Port={1}; Database={2}; UID={3}; Password={4}", ConfigConstants.DATABASE_HOST, ConfigConstants.DATABASE_PORT, ConfigConstants.DATABASE_NAME, ConfigConstants.DATABASE_USERNAME, ConfigConstants.DATABASE_PASSWORD))) { try { conn.Open(); query = @" INSERT INTO characters_npclinkshell (characterId, npcLinkshellId, isCalling, isExtra) VALUES (@charaId, @lsId, @calling, @extra) ON DUPLICATE KEY UPDATE characterId = @charaId, npcLinkshellId = @lsId, isCalling = @calling, isExtra = @extra "; cmd = new MySqlCommand(query, conn); cmd.Parameters.AddWithValue("@charaId", player.actorId); cmd.Parameters.AddWithValue("@lsId", npcLSId); cmd.Parameters.AddWithValue("@calling", isCalling ? 1 : 0); cmd.Parameters.AddWithValue("@extra", isExtra ? 1 : 0); cmd.ExecuteNonQuery(); } catch (MySqlException e) { Program.Log.Error(e.ToString()); } finally { conn.Dispose(); } } } } }