using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Data; using MySql.Data.MySqlClient; using PkmnFoundations.Structures; using PkmnFoundations.Support; using System.Security.Cryptography; using PkmnFoundations.Pokedex; namespace PkmnFoundations.Data { // todo: This class is getting quite large. We should move some things with // limited usefulness (eg. database creation) into separate classes. public class DataMysql : Database { #region Initialization public DataMysql(String connString) { ConnectionString = connString; } public String ConnectionString { get; set; } private MySqlConnection CreateConnection() { return new MySqlConnection(ConnectionString); } #endregion #region Utility public static String SqlSanitize(String s) { return SqlSanitize(s, ""); } public static String SqlSanitize(String s, String newChar) { String result = s.Replace("\'", newChar).Replace("[", newChar).Replace("]", newChar).Replace("`", newChar); int x = result.IndexOf("--"); if (x != -1) result = result.Substring(0, x); return result; } #endregion #region GTS 4 public GtsRecord4 GtsDataForUser4(MySqlTransaction tran, int pid) { MySqlDataReader reader = (MySqlDataReader)tran.ExecuteReader("SELECT Data, Species, Gender, Level, " + "RequestedSpecies, RequestedGender, RequestedMinLevel, RequestedMaxLevel, " + "Unknown1, TrainerGender, Unknown2, TimeDeposited, TimeExchanged, pid, " + "TrainerName, TrainerOT, TrainerCountry, TrainerRegion, TrainerClass, " + "IsExchanged, TrainerVersion, TrainerLanguage FROM GtsPokemon4 WHERE pid = @pid", new MySqlParameter("@pid", pid)); if (!reader.Read()) { reader.Close(); return null; } GtsRecord4 result = Record4FromReader(reader); #if DEBUG AssertHelper.Equals(result.PID, pid); #endif reader.Close(); return result; } public override GtsRecord4 GtsDataForUser4(int pid) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { GtsRecord4 result = GtsDataForUser4(tran, pid); tran.Commit(); return result; } } } public bool GtsDepositPokemon4(MySqlTransaction tran, GtsRecord4 record) { if (record.Data.Length != 236) throw new FormatException("pkm data must be 236 bytes."); if (record.TrainerName.RawData.Length != 16) throw new FormatException("Trainer name must be 16 bytes."); // note that IsTraded being true in the record is not an error condition // since it might have use later on. You should check for this in the upload handler. long count = (long)tran.ExecuteScalar("SELECT Count(*) FROM GtsPokemon4 WHERE pid = @pid", new MySqlParameter("@pid", record.PID)); if (count > 0) { // This player already has a pokemon in the system. // we can possibly allow multiples under some future conditions return false; } tran.ExecuteNonQuery("INSERT INTO GtsPokemon4 " + "(Data, Species, Gender, Level, RequestedSpecies, RequestedGender, " + "RequestedMinLevel, RequestedMaxLevel, Unknown1, TrainerGender, " + "Unknown2, TimeDeposited, TimeExchanged, pid, TrainerName, TrainerOT, " + "TrainerCountry, TrainerRegion, TrainerClass, IsExchanged, TrainerVersion, " + "TrainerLanguage) " + "VALUES (@Data, @Species, @Gender, @Level, @RequestedSpecies, " + "@RequestedGender, @RequestedMinLevel, @RequestedMaxLevel, @Unknown1, " + "@TrainerGender, @Unknown2, @TimeDeposited, @TimeExchanged, @pid, " + "@TrainerName, @TrainerOT, @TrainerCountry, @TrainerRegion, @TrainerClass, " + "@IsExchanged, @TrainerVersion, @TrainerLanguage)", ParamsFromRecord4(record)); return true; } public override bool GtsDepositPokemon4(GtsRecord4 record) { if (record.Data.Length != 236) throw new FormatException("pkm data must be 236 bytes."); if (record.TrainerName.RawData.Length != 16) throw new FormatException("Trainer name must be 16 bytes."); // note that IsTraded being true in the record is not an error condition // since it might have use later on. You should check for this in the upload handler. using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { if (!GtsDepositPokemon4(tran, record)) { tran.Rollback(); return false; } tran.Commit(); return true; } } } public ulong GtsGetDepositId4(int pid, MySqlTransaction tran) { object o = tran.ExecuteScalar("SELECT id FROM GtsPokemon4 WHERE pid = @pid " + "ORDER BY IsExchanged DESC, TimeExchanged, TimeDeposited LIMIT 1", new MySqlParameter("@pid", pid)); if (o == null || o == DBNull.Value) return 0; return Convert.ToUInt64(o); } public bool GtsDeletePokemon4(MySqlTransaction tran, int pid) { ulong pkmnId = GtsGetDepositId4(pid, tran); if (pkmnId == 0) return false; tran.ExecuteNonQuery("DELETE FROM GtsPokemon4 WHERE id = @id", new MySqlParameter("@id", pkmnId)); return true; } public override bool GtsDeletePokemon4(int pid) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { if (!GtsDeletePokemon4(tran, pid)) { tran.Rollback(); return false; } tran.Commit(); return true; } } } public override bool GtsTradePokemon4(int pidSrc, int pidDest) { // not needed yet. return false; } public override bool GtsTradePokemon4(GtsRecord4 upload, GtsRecord4 result) { GtsRecord4 traded = upload.Clone(); traded.FlagTraded(result); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { GtsRecord4 resultOrig = GtsDataForUser4(tran, result.PID); if (resultOrig == null || resultOrig != result) { // looks like the pokemon was ninja'd between the Exchange and Exchange_finish tran.Rollback(); return false; } if (!GtsDeletePokemon4(tran, result.PID)) { tran.Rollback(); return false; } if (!GtsDepositPokemon4(tran, traded)) { tran.Rollback(); return false; } tran.Commit(); return true; } } } public override GtsRecord4[] GtsSearch4(int pid, ushort species, Genders gender, byte minLevel, byte maxLevel, byte country, int count) { using (MySqlConnection db = CreateConnection()) { List _params = new List(); String where = "WHERE pid != @pid AND IsExchanged = 0"; _params.Add(new MySqlParameter("@pid", pid)); if (species > 0) { where += " AND Species = @species"; _params.Add(new MySqlParameter("@species", species)); } if (gender != Genders.Either) { where += " AND Gender = @gender"; _params.Add(new MySqlParameter("@gender", (byte)gender)); } if (minLevel > 0 && maxLevel > 0) { where += " AND Level BETWEEN @min_level AND @max_level"; _params.Add(new MySqlParameter("@min_level", minLevel)); _params.Add(new MySqlParameter("@max_level", maxLevel)); } else if (minLevel > 0) { where += " AND Level >= @min_level"; _params.Add(new MySqlParameter("@min_level", minLevel)); } else if (maxLevel > 0) { where += " AND Level <= @max_level"; _params.Add(new MySqlParameter("@max_level", maxLevel)); } if (country > 0) { where += " AND TrainerCountry = @country"; _params.Add(new MySqlParameter("@country", country)); } String limit = ""; if (count > 0) { _params.Add(new MySqlParameter("@count", count)); limit = " LIMIT @count"; } db.Open(); // todo: sort me in creative ways MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT Data, Species, Gender, Level, " + "RequestedSpecies, RequestedGender, RequestedMinLevel, RequestedMaxLevel, " + "Unknown1, TrainerGender, Unknown2, TimeDeposited, TimeExchanged, pid, " + "TrainerName, TrainerOT, TrainerCountry, TrainerRegion, TrainerClass, " + "IsExchanged, TrainerVersion, TrainerLanguage FROM GtsPokemon4 " + where + " ORDER BY TimeDeposited DESC" + limit, _params.ToArray()); List records; if (count > 0) records = new List(count); else records = new List(); while (reader.Read()) { records.Add(Record4FromReader(reader)); } return records.ToArray(); } } private static GtsRecord4 Record4FromReader(MySqlDataReader reader) { GtsRecord4 result = new GtsRecord4(); byte[] data = new byte[236]; reader.GetBytes(0, 0, data, 0, 236); result.Data = data; data = null; result.Species = reader.GetUInt16(1); result.Gender = (Genders)reader.GetByte(2); result.Level = reader.GetByte(3); result.RequestedSpecies = reader.GetUInt16(4); result.RequestedGender = (Genders)reader.GetByte(5); result.RequestedMinLevel = reader.GetByte(6); result.RequestedMaxLevel = reader.GetByte(7); result.Unknown1 = reader.GetByte(8); result.TrainerGender = (TrainerGenders)reader.GetByte(9); result.Unknown2 = reader.GetByte(10); if (reader.IsDBNull(11)) result.TimeDeposited = null; else result.TimeDeposited = reader.GetDateTime(11); if (reader.IsDBNull(12)) result.TimeExchanged = null; else result.TimeExchanged = reader.GetDateTime(12); result.PID = reader.GetInt32(13); data = new byte[16]; reader.GetBytes(14, 0, data, 0, 16); result.TrainerName = new EncodedString4(data); data = null; result.TrainerOT = reader.GetUInt16(15); result.TrainerCountry = reader.GetByte(16); result.TrainerRegion = reader.GetByte(17); result.TrainerClass = reader.GetByte(18); result.IsExchanged = reader.GetByte(19); result.TrainerVersion = reader.GetByte(20); result.TrainerLanguage = reader.GetByte(21); return result; } private static MySqlParameter[] ParamsFromRecord4(GtsRecord4 record) { MySqlParameter[] result = new MySqlParameter[22]; result[0] = new MySqlParameter("@Data", record.Data); result[1] = new MySqlParameter("@Species", record.Species); result[2] = new MySqlParameter("@Gender", (byte)record.Gender); result[3] = new MySqlParameter("@Level", record.Level); result[4] = new MySqlParameter("@RequestedSpecies", record.RequestedSpecies); result[5] = new MySqlParameter("@RequestedGender", (byte)record.RequestedGender); result[6] = new MySqlParameter("@RequestedMinLevel", record.RequestedMinLevel); result[7] = new MySqlParameter("@RequestedMaxLevel", record.RequestedMaxLevel); result[8] = new MySqlParameter("@Unknown1", record.Unknown1); result[9] = new MySqlParameter("@TrainerGender", (byte)record.TrainerGender); result[10] = new MySqlParameter("@Unknown2", record.Unknown2); result[11] = new MySqlParameter("@TimeDeposited", record.TimeDeposited); result[12] = new MySqlParameter("@TimeExchanged", record.TimeExchanged); result[13] = new MySqlParameter("@pid", record.PID); result[14] = new MySqlParameter("@TrainerName", record.TrainerName.RawData); result[15] = new MySqlParameter("@TrainerOT", record.TrainerOT); result[16] = new MySqlParameter("@TrainerCountry", record.TrainerCountry); result[17] = new MySqlParameter("@TrainerRegion", record.TrainerRegion); result[18] = new MySqlParameter("@TrainerClass", record.TrainerClass); result[19] = new MySqlParameter("@IsExchanged", record.IsExchanged); result[20] = new MySqlParameter("@TrainerVersion", record.TrainerVersion); result[21] = new MySqlParameter("@TrainerLanguage", record.TrainerLanguage); return result; } public override int GtsAvailablePokemon4() { using (MySqlConnection db = CreateConnection()) { db.Open(); return (int)(long)db.ExecuteScalar("SELECT Count(*) FROM GtsPokemon4 WHERE IsExchanged = 0"); } } public override void GtsLogTrade4(GtsRecord4 record, DateTime ? timeWithdrawn, int ? partner_pid) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { GtsLogTrade4(tran, record, timeWithdrawn, partner_pid); tran.Commit(); } } } public void GtsLogTrade4(MySqlTransaction tran, GtsRecord4 record, DateTime? timeWithdrawn, int ? partner_pid) { if (record.Data.Length != 236) throw new FormatException("pkm data must be 236 bytes."); if (record.TrainerName.RawData.Length != 16) throw new FormatException("Trainer name must be 16 bytes."); // note that IsTraded being true in the record is not an error condition // since it might have use later on. You should check for this in the upload handler. int trade_id = (int)tran.ExecuteScalar("SELECT id FROM GtsPokemon4 WHERE pid = @pid", new MySqlParameter("@pid", record.PID)); // when calling delete.asp, the partner pid can't be told from the request alone, // so obtain it from the database instead. if (record.IsExchanged != 0) partner_pid = (int ?)tran.ExecuteScalar("SELECT partner_pid FROM GtsHistory4 " + "WHERE trade_id = @trade_id AND IsExchanged = 0", new MySqlParameter("@trade_id", trade_id)); MySqlParameter[] _params = ParamsFromRecord4(record); MySqlParameter[] _params2 = new MySqlParameter[25]; Array.Copy(_params, _params2, 22); _params2[22] = new MySqlParameter("@TimeWithdrawn", timeWithdrawn); _params2[23] = new MySqlParameter("@trade_id", trade_id); _params2[24] = new MySqlParameter("@partner_pid", partner_pid); tran.ExecuteNonQuery("INSERT INTO GtsHistory4 " + "(Data, Species, Gender, Level, RequestedSpecies, RequestedGender, " + "RequestedMinLevel, RequestedMaxLevel, Unknown1, TrainerGender, " + "Unknown2, TimeDeposited, TimeExchanged, pid, TrainerName, TrainerOT, " + "TrainerCountry, TrainerRegion, TrainerClass, IsExchanged, TrainerVersion, " + "TrainerLanguage, TimeWithdrawn, trade_id, partner_pid) " + "VALUES (@Data, @Species, @Gender, @Level, @RequestedSpecies, " + "@RequestedGender, @RequestedMinLevel, @RequestedMaxLevel, @Unknown1, " + "@TrainerGender, @Unknown2, @TimeDeposited, @TimeExchanged, @pid, " + "@TrainerName, @TrainerOT, @TrainerCountry, @TrainerRegion, @TrainerClass, " + "@IsExchanged, @TrainerVersion, @TrainerLanguage, @TimeWithdrawn, " + "@trade_id, @partner_pid)", _params2); } #endregion #region Battle Tower 4 public override ulong BattleTowerUpdateRecord4(BattleTowerRecord4 record) { if (record.BattlesWon > 7) throw new ArgumentException("Battles won can not be greater than 7."); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { ulong result = BattleTowerUpdateRecord4(tran, record); tran.Commit(); return result; } } } private ulong BattleTowerUpdateRecord4(MySqlTransaction tran, BattleTowerRecord4 record) { if (record.BattlesWon > 7) throw new ArgumentException("Battles won can not be greater than 7."); // Does this player already have a record in this room? // Also get primary key if it does. (We need it for updating party) // // The official server doesn't seem to ever replace existing // records. This worked fine for them, but we don't have nearly // as many active players, so doing this will cause too many // duplicates. Instead, we require the trainers in a given room to // be unique by replacing their old record with a new one. ulong pkey = FindBattleTowerRecord4(tran, record, false); if (pkey != 0) { // If the player already has a record, move everyone below it up one position // (effectively removing this record from the ordering) // todo: In the case that the player's rank hasn't changed, // we can optimize this and the next down to a single BETWEEN // query. // This does require retrieving their old rank from the db. tran.ExecuteNonQuery("SELECT @old_rank = Rank, @old_position = Position " + "FROM GtsBattleTower4 WHERE pid = @pid AND RoomNum = @room); " + "UPDATE GtsBattleTower4 SET Position = Position - 1 " + "WHERE RoomNum = @room AND Rank = @old_rank AND Position > @old_position", new MySqlParameter("@pid", record.PID), new MySqlParameter("@room", record.RoomNum)); } uint position = (uint)(7 - record.BattlesWon); // Shift down all the players in the player's new rank by one. tran.ExecuteNonQuery("UPDATE GtsBattleTower4 SET Position = Position + 1 " + "WHERE RoomNum = @room AND Rank = @rank AND Position >= @position", new MySqlParameter("@room", record.RoomNum), new MySqlParameter("@rank", record.Rank), new MySqlParameter("@position", position)); object lastPosition = tran.ExecuteScalar("SELECT MAX(Position) " + "FROM GtsBattleTower4 WHERE RoomNum = @room AND Rank = @rank", new MySqlParameter("@room", record.RoomNum), new MySqlParameter("@rank", record.Rank)); // If the room has fewer than 7 trainers, insert this one at the // end but don't leave any gaps in the numbering. if (lastPosition is DBNull) position = 0; else position = Math.Min(position, (uint)lastPosition + 1); // Update the actual record if (pkey != 0) { List _params = ParamsFromBattleTowerRecord4(record, false); _params.Add(new MySqlParameter("@position", position)); _params.Add(new MySqlParameter("@id", pkey)); tran.ExecuteNonQuery("UPDATE GtsBattleTower4 SET pid = @pid, Name = @name, " + "Version = @version, Language = @language, Country = @country, " + "Region = @region, TrainerID = @trainer_id, " + "PhraseLeader = @phrase_leader, Gender = @gender, " + "Unknown2 = @unknown2, PhraseChallenged = @phrase_challenged, " + "PhraseWon = @phrase_won, PhraseLost = @phrase_lost, " + "Unknown3 = @unknown3, " + "Unknown5 = @unknown5, ParseVersion = 1, Rank = @rank, " + "BattlesWon = @battles_won, Position = @position, " + "TimeUpdated = UTC_TIMESTAMP() WHERE id = @id", _params.ToArray()); UpdateBattleTowerPokemon4(tran, record.Party[0], pkey, 0); UpdateBattleTowerPokemon4(tran, record.Party[1], pkey, 1); UpdateBattleTowerPokemon4(tran, record.Party[2], pkey, 2); } else { List _params = ParamsFromBattleTowerRecord4(record, false); _params.Add(new MySqlParameter("@position", position)); pkey = Convert.ToUInt64(tran.ExecuteScalar("INSERT INTO GtsBattleTower4 " + "(pid, Name, Version, Language, Country, Region, TrainerID, " + "PhraseLeader, Gender, Unknown2, PhraseChallenged, PhraseWon, " + "PhraseLost, Unknown3, Unknown5, ParseVersion, " + "Rank, RoomNum, BattlesWon, Position, TimeAdded, TimeUpdated) VALUES " + "(@pid, @name, @version, @language, @country, @region, @trainer_id, " + "@phrase_leader, @gender, @unknown2, @phrase_challenged, @phrase_won, " + "@phrase_lost, @unknown3, @unknown5, 1, " + "@rank, @room, @battles_won, @position, UTC_TIMESTAMP(), UTC_TIMESTAMP()); " + "SELECT LAST_INSERT_ID()", _params.ToArray())); InsertBattleTowerPokemon4(tran, record.Party[0], pkey, 0); InsertBattleTowerPokemon4(tran, record.Party[1], pkey, 1); InsertBattleTowerPokemon4(tran, record.Party[2], pkey, 2); } return pkey; } private void InsertBattleTowerPokemon4(MySqlTransaction tran, BattleTowerPokemon4 pokemon, ulong partyId, byte slot) { List _params = ParamsFromBattleTowerPokemon4(pokemon); _params.Add(new MySqlParameter("@id", partyId)); _params.Add(new MySqlParameter("@slot", slot)); tran.ExecuteNonQuery("INSERT INTO GtsBattleTowerPokemon4 " + "(party_id, Slot, Species, HeldItem, Move1, Move2, Move3, Move4, TrainerID, " + "Personality, IVs, EVs, Unknown1, Language, Ability, Happiness, Nickname) VALUES " + "(@id, @slot, @species, @held_item, @move1, @move2, @move3, @move4, @trainer_id, " + "@personality, @ivs, @evs, @unknown1, @language, @ability, @happiness, @nickname)", _params.ToArray()); } private void UpdateBattleTowerPokemon4(MySqlTransaction tran, BattleTowerPokemon4 pokemon, ulong partyId, byte slot) { List _params = ParamsFromBattleTowerPokemon4(pokemon); _params.Add(new MySqlParameter("@id", partyId)); _params.Add(new MySqlParameter("@slot", slot)); tran.ExecuteNonQuery("UPDATE GtsBattleTowerPokemon4 SET Species = @species, " + "HeldItem = @held_item, Move1 = @move1, Move2 = @move2, Move3 = @move3, " + "Move4 = @move4, TrainerID = @trainer_id, Personality = @personality, " + "IVs = @ivs, EVs = @evs, Unknown1 = @unknown1, Language = @language, " + "Ability = @ability, Happiness = @happiness, Nickname = @nickname " + "WHERE party_id = @id AND Slot = @slot", _params.ToArray()); } private List ParamsFromBattleTowerRecord4(BattleTowerRecord4 record, bool leader) { List result = new List(15); result.Add(new MySqlParameter("@pid", record.PID)); result.Add(new MySqlParameter("@name", record.Profile.Name.RawData)); result.Add(new MySqlParameter("@version", (byte)record.Profile.Version)); result.Add(new MySqlParameter("@language", (byte)record.Profile.Language)); result.Add(new MySqlParameter("@country", record.Profile.Country)); result.Add(new MySqlParameter("@region", record.Profile.Region)); result.Add(new MySqlParameter("@trainer_id", record.Profile.OT)); result.Add(new MySqlParameter("@phrase_leader", record.Profile.PhraseLeader.Data)); result.Add(new MySqlParameter("@gender", record.Profile.Gender)); result.Add(new MySqlParameter("@unknown2", record.Profile.Unknown)); result.Add(new MySqlParameter("@rank", record.Rank)); result.Add(new MySqlParameter("@room", record.RoomNum)); if (!leader) { result.Add(new MySqlParameter("@phrase_challenged", record.PhraseChallenged.Data)); result.Add(new MySqlParameter("@phrase_won", record.PhraseWon.Data)); result.Add(new MySqlParameter("@phrase_lost", record.PhraseLost.Data)); result.Add(new MySqlParameter("@unknown3", record.Unknown3)); result.Add(new MySqlParameter("@unknown5", record.Unknown5)); result.Add(new MySqlParameter("@battles_won", record.BattlesWon)); } return result; } private List ParamsFromBattleTowerPokemon4(BattleTowerPokemon4 pokemon) { List result = new List(15); result.Add(new MySqlParameter("@species", pokemon.Species)); result.Add(new MySqlParameter("@held_item", pokemon.HeldItem)); result.Add(new MySqlParameter("@move1", pokemon.Moveset[0])); result.Add(new MySqlParameter("@move2", pokemon.Moveset[1])); result.Add(new MySqlParameter("@move3", pokemon.Moveset[2])); result.Add(new MySqlParameter("@move4", pokemon.Moveset[3])); result.Add(new MySqlParameter("@trainer_id", pokemon.OT)); result.Add(new MySqlParameter("@personality", pokemon.Personality)); result.Add(new MySqlParameter("@ivs", pokemon.IVs)); result.Add(new MySqlParameter("@evs", pokemon.EVs)); result.Add(new MySqlParameter("@unknown1", pokemon.Unknown1)); result.Add(new MySqlParameter("@language", (byte)pokemon.Language)); result.Add(new MySqlParameter("@ability", pokemon.Ability)); result.Add(new MySqlParameter("@happiness", pokemon.Happiness)); result.Add(new MySqlParameter("@nickname", pokemon.Nickname.RawData)); return result; } public override ulong BattleTowerAddLeader4(BattleTowerRecord4 record) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { ulong result = BattleTowerAddLeader4(tran, record); tran.Commit(); return result; } } } private ulong BattleTowerAddLeader4(MySqlTransaction tran, BattleTowerRecord4 record) { ulong pkey = FindBattleTowerRecord4(tran, record, true); // Update the actual record if (pkey != 0) { List _params = ParamsFromBattleTowerRecord4(record, true); _params.Add(new MySqlParameter("@id", pkey)); tran.ExecuteNonQuery("UPDATE GtsBattleTowerLeaders4 SET " + "pid = @pid, Name = @name, Version = @version, " + "Language = @language, Country = @country, Region = @region, " + "TrainerID = @trainer_id, " + "PhraseLeader = @phrase_leader, Gender = @gender, Unknown2 = @unknown2, " + "ParseVersion = 1, Rank = @rank, " + "TimeUpdated = UTC_TIMESTAMP() WHERE id = @id", _params.ToArray()); } else { List _params = ParamsFromBattleTowerRecord4(record, true); pkey = Convert.ToUInt64(tran.ExecuteScalar("INSERT INTO " + "GtsBattleTowerLeaders4 " + "(pid, Name, Version, Language, Country, Region, TrainerID, " + "PhraseLeader, Gender, Unknown2, ParseVersion, Rank, " + "RoomNum, TimeAdded, TimeUpdated) VALUES " + "(@pid, @name, @version, @language, @country, @region, @trainer_id, " + "@phrase_leader, @gender, @unknown2, 1, @rank, " + "@room, UTC_TIMESTAMP(), UTC_TIMESTAMP()); " + "SELECT LAST_INSERT_ID()", _params.ToArray())); } return pkey; } /// /// Tries to find an existing database record for the provided player /// record. The match must be found in the same rank and room number. /// /// /// /// If true, look up against the Leaders table. /// Otherwise looks up against the opponents table. /// The match's primary key or 0 if no match is found /// private ulong FindBattleTowerRecord4(MySqlTransaction tran, BattleTowerRecord4 record, bool leader) { String tblName = leader ? "GtsBattleTowerLeaders4" : "GtsBattleTower4"; // If PID is missing, this is restored data. // We assume the original server took care of matching existing // records, so we don't allow it to match here. if (record.PID == 0) return 0; // Match normally. object oPkey = tran.ExecuteScalar("SELECT id FROM " + tblName + " WHERE pid = @pid AND RoomNum = @room AND Rank = @rank", new MySqlParameter("@pid", record.PID), new MySqlParameter("@rank", record.Rank), new MySqlParameter("@room", record.RoomNum)); if (oPkey == null) { // PID isn't found. Try to match one of Pikachu025's saved // records based on unchanging properties of the savegame. oPkey = tran.ExecuteScalar("SELECT id FROM " + tblName + " WHERE pid = 0 AND RoomNum = @room AND Rank = @rank " + "AND Name = @name AND Version = @version " + "AND Language = @language AND TrainerID = @trainer_id", new MySqlParameter("@rank", record.Rank), new MySqlParameter("@room", record.RoomNum), new MySqlParameter("@name", record.Profile.Name), new MySqlParameter("@version", (byte)record.Profile.Version), new MySqlParameter("@language", (byte)record.Profile.Language), new MySqlParameter("@trainer_id", record.Profile.OT) ); } // Don't need to worry about DBNull since the column is non-null. return (ulong)(oPkey ?? 0UL); } public override BattleTowerRecord4[] BattleTowerGetOpponents4(int pid, byte rank, byte roomNum) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { List records = new List(7); List keys = new List(7); MySqlDataReader reader = (MySqlDataReader)tran.ExecuteReader( "SELECT id, pid, Name, " + "Version, Language, Country, Region, TrainerID, " + "PhraseLeader, Gender, Unknown2, PhraseChallenged, " + "PhraseWon, PhraseLost, Unknown3, Unknown5 FROM GtsBattleTower4 " + "WHERE Rank = @rank AND RoomNum = @room AND pid != @pid " + "ORDER BY Position LIMIT 7", new MySqlParameter("@rank", rank), new MySqlParameter("@room", roomNum), new MySqlParameter("@pid", pid)); while (reader.Read()) { BattleTowerRecord4 record = BattleTowerRecord4FromReader(reader); record.Party = new BattleTowerPokemon4[3]; records.Add(record); keys.Add(reader.GetUInt64(0)); } reader.Close(); if (records.Count == 0) return new BattleTowerRecord4[0]; String inClause = String.Join(", ", keys.Select(i => i.ToString()).ToArray()); reader = (MySqlDataReader)tran.ExecuteReader("SELECT party_id, " + "Slot, Species, HeldItem, Move1, Move2, Move3, Move4, " + "TrainerID, Personality, IVs, EVs, Unknown1, Language, " + "Ability, Happiness, Nickname FROM GtsBattleTowerPokemon4 " + "WHERE party_id IN (" + inClause + ")"); while (reader.Read()) { BattleTowerRecord4 record = records[keys.IndexOf(reader.GetUInt64(0))]; record.Party[reader.GetByte(1)] = BattleTowerPokemon4FromReader(reader); } reader.Close(); tran.Commit(); return Enumerable.Reverse(records).ToArray(); } } } private BattleTowerRecord4 BattleTowerRecord4FromReader(MySqlDataReader reader) { // todo: Stop using ordinals everywhere. BattleTowerRecord4 result = new BattleTowerRecord4(); result.PID = reader.GetInt32(1); if (reader.FieldCount > 11) result.PhraseChallenged = new TrendyPhrase4(reader.GetByteArray(11, 8)); if (reader.FieldCount > 12) result.PhraseWon = new TrendyPhrase4(reader.GetByteArray(12, 8)); if (reader.FieldCount > 13) result.PhraseLost = new TrendyPhrase4(reader.GetByteArray(13, 8)); if (reader.FieldCount > 14) result.Unknown3 = reader.GetUInt16(14); if (reader.FieldCount > 15) result.Unknown5 = reader.GetUInt64(15); BattleTowerProfile4 profile = new BattleTowerProfile4(); profile.Name = new EncodedString4(reader.GetByteArray(2, 16)); profile.Version = (Versions)reader.GetByte(3); profile.Language = (Languages)reader.GetByte(4); profile.Country = reader.GetByte(5); profile.Region = reader.GetByte(6); profile.OT = reader.GetUInt32(7); profile.PhraseLeader = new TrendyPhrase4(reader.GetByteArray(8, 8)); profile.Gender = reader.GetByte(9); profile.Unknown = reader.GetByte(10); result.Profile = profile; return result; } private BattleTowerPokemon4 BattleTowerPokemon4FromReader(MySqlDataReader reader) { BattleTowerPokemon4 result = new BattleTowerPokemon4(); result.Species = reader.GetUInt16(2); result.HeldItem = reader.GetUInt16(3); result.Moveset = new ushort[4]; result.Moveset[0] = reader.GetUInt16(4); result.Moveset[1] = reader.GetUInt16(5); result.Moveset[2] = reader.GetUInt16(6); result.Moveset[3] = reader.GetUInt16(7); result.OT = reader.GetUInt32(8); result.Personality = reader.GetUInt32(9); result.IVs = reader.GetUInt32(10); result.EVs = reader.GetByteArray(11, 6); result.Unknown1 = reader.GetByte(12); result.Language = (Languages)reader.GetByte(13); result.Ability = reader.GetByte(14); result.Happiness = reader.GetByte(15); result.Nickname = new EncodedString4(reader.GetByteArray(16, 22)); return result; } public override BattleTowerProfile4[] BattleTowerGetLeaders4(byte rank, byte roomNum) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { List profiles = new List(30); MySqlDataReader reader = (MySqlDataReader)tran.ExecuteReader( "SELECT id, pid, Name, " + "Version, Language, Country, Region, TrainerID, " + "PhraseLeader, Gender, Unknown2 FROM GtsBattleTowerLeaders4 " + "WHERE Rank = @rank AND RoomNum = @room " + "ORDER BY TimeUpdated DESC, id LIMIT 30", new MySqlParameter("@rank", rank), new MySqlParameter("@room", roomNum)); while (reader.Read()) { profiles.Add(BattleTowerRecord4FromReader(reader).Profile); } reader.Close(); tran.Commit(); return profiles.ToArray(); } } } #endregion #region Other Gamestats 4 public override bool GamestatsSetProfile4(TrainerProfile4 profile) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { bool result = GamestatsSetProfile4(tran, profile); tran.Commit(); return result; } } } public bool GamestatsSetProfile4(MySqlTransaction tran, TrainerProfile4 profile) { if (profile.Data.Length != 100) throw new FormatException("Profile data must be 100 bytes."); long exists = (long)tran.ExecuteScalar("SELECT EXISTS(SELECT * FROM GtsProfiles4 WHERE pid = @pid)", new MySqlParameter("@pid", profile.PID)); MySqlParameter[] _params = new MySqlParameter[]{ new MySqlParameter("@pid", profile.PID), new MySqlParameter("@data", profile.Data), new MySqlParameter("@version", (byte)profile.Version), new MySqlParameter("@language", (byte)profile.Language), new MySqlParameter("@country", profile.Country), new MySqlParameter("@region", profile.Region), new MySqlParameter("@ot", profile.OT), new MySqlParameter("@name", profile.Name.RawData) }; if (exists != 0) { return tran.ExecuteNonQuery("UPDATE GtsProfiles4 SET Data = @data, " + "Version = @version, Language = @language, Country = @country, " + "Region = @region, OT = @ot, Name = @name, ParseVersion = 1, " + "TimeUpdated = UTC_TIMESTAMP() " + "WHERE pid = @pid", _params) > 0; } else { return tran.ExecuteNonQuery("INSERT INTO GtsProfiles4 " + "(pid, Data, Version, Language, Country, Region, OT, Name, " + "ParseVersion, TimeAdded, TimeUpdated) VALUES " + "(@pid, @data, @version, @language, @country, @region, @ot, " + "@name, 1, UTC_TIMESTAMP(), UTC_TIMESTAMP())", _params) > 0; } } #endregion #region GTS 5 public GtsRecord5 GtsDataForUser5(MySqlTransaction tran, int pid) { MySqlDataReader reader = (MySqlDataReader)tran.ExecuteReader("SELECT Data, Unknown0, " + "Species, Gender, Level, " + "RequestedSpecies, RequestedGender, RequestedMinLevel, RequestedMaxLevel, " + "Unknown1, TrainerGender, Unknown2, TimeDeposited, TimeExchanged, pid, " + "TrainerOT, TrainerName, TrainerCountry, TrainerRegion, TrainerClass, " + "IsExchanged, TrainerVersion, TrainerLanguage, TrainerBadges, TrainerUnityTower " + "FROM GtsPokemon5 WHERE pid = @pid", new MySqlParameter("@pid", pid)); if (!reader.Read()) { reader.Close(); return null; } GtsRecord5 result = Record5FromReader(reader); #if DEBUG AssertHelper.Equals(result.PID, pid); #endif reader.Close(); return result; } public override GtsRecord5 GtsDataForUser5(int pid) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { GtsRecord5 result = GtsDataForUser5(tran, pid); tran.Commit(); return result; } } } public bool GtsDepositPokemon5(MySqlTransaction tran, GtsRecord5 record) { if (record == null) throw new ArgumentNullException("record"); if (record.Data.Length != 220) throw new FormatException("pkm data must be 220 bytes."); if (record.Unknown0.Length != 16) throw new FormatException("pkm padding must be 16 bytes."); if (record.TrainerName.RawData.Length != 16) throw new FormatException("Trainer name must be 16 bytes."); // note that IsTraded being true in the record is not an error condition // since it might have use later on. You should check for this in the upload handler. long count = (long)tran.ExecuteScalar("SELECT Count(*) FROM GtsPokemon5 WHERE pid = @pid", new MySqlParameter("@pid", record.PID)); if (count > 0) { // This player already has a pokemon in the system. // we can possibly allow multiples under some future conditions return false; } tran.ExecuteNonQuery("INSERT INTO GtsPokemon5 " + "(Data, Unknown0, Species, Gender, Level, RequestedSpecies, RequestedGender, " + "RequestedMinLevel, RequestedMaxLevel, Unknown1, TrainerGender, " + "Unknown2, TimeDeposited, TimeExchanged, pid, TrainerOT, TrainerName, " + "TrainerCountry, TrainerRegion, TrainerClass, IsExchanged, TrainerVersion, " + "TrainerLanguage, TrainerBadges, TrainerUnityTower) " + "VALUES (@Data, @Unknown0, @Species, @Gender, @Level, @RequestedSpecies, " + "@RequestedGender, @RequestedMinLevel, @RequestedMaxLevel, @Unknown1, " + "@TrainerGender, @Unknown2, @TimeDeposited, @TimeExchanged, @pid, " + "@TrainerOT, @TrainerName, @TrainerCountry, @TrainerRegion, @TrainerClass, " + "@IsExchanged, @TrainerVersion, @TrainerLanguage, @TrainerBadges, @TrainerUnityTower)", ParamsFromRecord5(record)); return true; } public override bool GtsDepositPokemon5(GtsRecord5 record) { if (record == null) throw new ArgumentNullException("record"); if (record.Data.Length != 220) throw new FormatException("pkm data must be 220 bytes."); if (record.Unknown0.Length != 16) throw new FormatException("pkm padding must be 16 bytes."); if (record.TrainerName.RawData.Length != 16) throw new FormatException("Trainer name must be 16 bytes."); // note that IsTraded being true in the record is not an error condition // since it might have use later on. You should check for this in the upload handler. using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { if (!GtsDepositPokemon5(tran, record)) { tran.Rollback(); return false; } tran.Commit(); return true; } } } public ulong GtsGetDepositId5(int pid, MySqlTransaction tran) { object o = tran.ExecuteScalar("SELECT id FROM GtsPokemon5 WHERE pid = @pid " + "ORDER BY IsExchanged DESC, TimeExchanged, TimeDeposited LIMIT 1", new MySqlParameter("@pid", pid)); if (o == null || o == DBNull.Value) return 0; return Convert.ToUInt64(o); } public bool GtsDeletePokemon5(MySqlTransaction tran, int pid) { ulong pkmnId = GtsGetDepositId5(pid, tran); if (pkmnId == 0) return false; tran.ExecuteNonQuery("DELETE FROM GtsPokemon5 WHERE id = @id", new MySqlParameter("@id", pkmnId)); return true; } public override bool GtsDeletePokemon5(int pid) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { if (!GtsDeletePokemon5(tran, pid)) { tran.Rollback(); return false; } tran.Commit(); return true; } } } public override bool GtsTradePokemon5(int pidSrc, int pidDest) { // not needed yet. return false; } public override bool GtsTradePokemon5(GtsRecord5 upload, GtsRecord5 result) { GtsRecord5 traded = upload.Clone(); traded.FlagTraded(result); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { GtsRecord5 resultOrig = GtsDataForUser5(tran, result.PID); if (resultOrig == null || resultOrig != result) { // looks like the pokemon was ninja'd between the Exchange and Exchange_finish tran.Rollback(); return false; } if (!GtsDeletePokemon5(tran, result.PID)) { tran.Rollback(); return false; } if (!GtsDepositPokemon5(tran, traded)) { tran.Rollback(); return false; } tran.Commit(); return true; } } } public override GtsRecord5[] GtsSearch5(int pid, ushort species, Genders gender, byte minLevel, byte maxLevel, byte country, int count) { using (MySqlConnection db = CreateConnection()) { List _params = new List(); String where = "WHERE pid != @pid AND IsExchanged = 0"; _params.Add(new MySqlParameter("@pid", pid)); if (species > 0) { where += " AND Species = @species"; _params.Add(new MySqlParameter("@species", species)); } if (gender != Genders.Either) { where += " AND Gender = @gender"; _params.Add(new MySqlParameter("@gender", (byte)gender)); } if (minLevel > 0 && maxLevel > 0) { where += " AND Level BETWEEN @min_level AND @max_level"; _params.Add(new MySqlParameter("@min_level", minLevel)); _params.Add(new MySqlParameter("@max_level", maxLevel)); } else if (minLevel > 0) { where += " AND Level >= @min_level"; _params.Add(new MySqlParameter("@min_level", minLevel)); } else if (maxLevel > 0) { where += " AND Level <= @max_level"; _params.Add(new MySqlParameter("@max_level", maxLevel)); } if (country > 0) { where += " AND TrainerCountry = @country"; _params.Add(new MySqlParameter("@country", country)); } String limit = ""; if (count > 0) { _params.Add(new MySqlParameter("@count", count)); limit = " LIMIT @count"; } db.Open(); // todo: sort me in creative ways MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT Data, Unknown0, " + "Species, Gender, Level, " + "RequestedSpecies, RequestedGender, RequestedMinLevel, RequestedMaxLevel, " + "Unknown1, TrainerGender, Unknown2, TimeDeposited, TimeExchanged, pid, " + "TrainerOT, TrainerName, TrainerCountry, TrainerRegion, TrainerClass, " + "IsExchanged, TrainerVersion, TrainerLanguage, TrainerBadges, TrainerUnityTower " + "FROM GtsPokemon5 " + where + " ORDER BY TimeDeposited DESC" + limit, _params.ToArray()); List records; if (count > 0) records = new List(count); else records = new List(); while (reader.Read()) { records.Add(Record5FromReader(reader)); } return records.ToArray(); } } private static GtsRecord5 Record5FromReader(MySqlDataReader reader) { GtsRecord5 result = new GtsRecord5(); byte[] data = new byte[220]; reader.GetBytes(0, 0, data, 0, 220); result.Data = data; data = null; data = new byte[16]; reader.GetBytes(1, 0, data, 0, 16); result.Unknown0 = data; data = null; result.Species = reader.GetUInt16(2); result.Gender = (Genders)reader.GetByte(3); result.Level = reader.GetByte(4); result.RequestedSpecies = reader.GetUInt16(5); result.RequestedGender = (Genders)reader.GetByte(6); result.RequestedMinLevel = reader.GetByte(7); result.RequestedMaxLevel = reader.GetByte(8); result.Unknown1 = reader.GetByte(9); result.TrainerGender = (TrainerGenders)reader.GetByte(10); result.Unknown2 = reader.GetByte(11); if (reader.IsDBNull(12)) result.TimeDeposited = null; else result.TimeDeposited = reader.GetDateTime(12); if (reader.IsDBNull(13)) result.TimeExchanged = null; else result.TimeExchanged = reader.GetDateTime(13); result.PID = reader.GetInt32(14); result.TrainerOT = reader.GetUInt32(15); data = new byte[16]; reader.GetBytes(16, 0, data, 0, 16); result.TrainerName = new EncodedString5(data); data = null; result.TrainerCountry = reader.GetByte(17); result.TrainerRegion = reader.GetByte(18); result.TrainerClass = reader.GetByte(19); result.IsExchanged = reader.GetByte(20); result.TrainerVersion = reader.GetByte(21); result.TrainerLanguage = reader.GetByte(22); result.TrainerBadges = reader.GetByte(23); result.TrainerUnityTower = reader.GetByte(24); return result; } private static MySqlParameter[] ParamsFromRecord5(GtsRecord5 record) { MySqlParameter[] result = new MySqlParameter[25]; result[0] = new MySqlParameter("@Data", record.Data); result[1] = new MySqlParameter("@Unknown0", record.Unknown0); result[2] = new MySqlParameter("@Species", record.Species); result[3] = new MySqlParameter("@Gender", (byte)record.Gender); result[4] = new MySqlParameter("@Level", record.Level); result[5] = new MySqlParameter("@RequestedSpecies", record.RequestedSpecies); result[6] = new MySqlParameter("@RequestedGender", (byte)record.RequestedGender); result[7] = new MySqlParameter("@RequestedMinLevel", record.RequestedMinLevel); result[8] = new MySqlParameter("@RequestedMaxLevel", record.RequestedMaxLevel); result[9] = new MySqlParameter("@Unknown1", record.Unknown1); result[10] = new MySqlParameter("@TrainerGender", (byte)record.TrainerGender); result[11] = new MySqlParameter("@Unknown2", record.Unknown2); result[12] = new MySqlParameter("@TimeDeposited", record.TimeDeposited); result[13] = new MySqlParameter("@TimeExchanged", record.TimeExchanged); result[14] = new MySqlParameter("@pid", record.PID); result[15] = new MySqlParameter("@TrainerOT", record.TrainerOT); result[16] = new MySqlParameter("@TrainerName", record.TrainerName.RawData); result[17] = new MySqlParameter("@TrainerCountry", record.TrainerCountry); result[18] = new MySqlParameter("@TrainerRegion", record.TrainerRegion); result[19] = new MySqlParameter("@TrainerClass", record.TrainerClass); result[20] = new MySqlParameter("@IsExchanged", record.IsExchanged); result[21] = new MySqlParameter("@TrainerVersion", record.TrainerVersion); result[22] = new MySqlParameter("@TrainerLanguage", record.TrainerLanguage); result[23] = new MySqlParameter("@TrainerBadges", record.TrainerBadges); result[24] = new MySqlParameter("@TrainerUnityTower", record.TrainerUnityTower); return result; } public override int GtsAvailablePokemon5() { using (MySqlConnection db = CreateConnection()) { db.Open(); return (int)(long)db.ExecuteScalar("SELECT Count(*) FROM GtsPokemon5 WHERE IsExchanged = 0"); } } public override void GtsLogTrade5(GtsRecord5 record, DateTime ? timeWithdrawn, int ? partner_pid) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { GtsLogTrade5(tran, record, timeWithdrawn, partner_pid); tran.Commit(); } } } public void GtsLogTrade5(MySqlTransaction tran, GtsRecord5 record, DateTime ? timeWithdrawn, int ? partner_pid) { // todo: Bring these out into a ValidateRecord5 method if (record == null) throw new ArgumentNullException("record"); if (record.Data.Length != 220) throw new FormatException("pkm data must be 220 bytes."); if (record.Unknown0.Length != 16) throw new FormatException("pkm padding must be 16 bytes."); if (record.TrainerName.RawData.Length != 16) throw new FormatException("Trainer name must be 16 bytes."); // note that IsTraded being true in the record is not an error condition // since it might have use later on. You should check for this in the upload handler. int trade_id = (int)tran.ExecuteScalar("SELECT id FROM GtsPokemon5 WHERE pid = @pid", new MySqlParameter("@pid", record.PID)); // when calling delete.asp, the partner pid can't be told from the request alone, // so obtain it from the database instead. if (record.IsExchanged != 0) partner_pid = (int ?)tran.ExecuteScalar("SELECT partner_pid FROM GtsHistory5 " + "WHERE trade_id = @trade_id AND IsExchanged = 0", new MySqlParameter("@trade_id", trade_id)); MySqlParameter[] _params = ParamsFromRecord5(record); MySqlParameter[] _params2 = new MySqlParameter[28]; Array.Copy(_params, _params2, 25); _params2[25] = new MySqlParameter("@TimeWithdrawn", timeWithdrawn); _params2[26] = new MySqlParameter("@trade_id", trade_id); _params2[27] = new MySqlParameter("@partner_pid", partner_pid); tran.ExecuteNonQuery("INSERT INTO GtsHistory5 " + "(Data, Unknown0, Species, Gender, Level, RequestedSpecies, RequestedGender, " + "RequestedMinLevel, RequestedMaxLevel, Unknown1, TrainerGender, " + "Unknown2, TimeDeposited, TimeExchanged, pid, TrainerOT, TrainerName, " + "TrainerCountry, TrainerRegion, TrainerClass, IsExchanged, TrainerVersion, " + "TrainerLanguage, TrainerBadges, TrainerUnityTower, TimeWithdrawn, " + "trade_id, partner_pid) " + "VALUES (@Data, @Unknown0, @Species, @Gender, @Level, @RequestedSpecies, " + "@RequestedGender, @RequestedMinLevel, @RequestedMaxLevel, @Unknown1, " + "@TrainerGender, @Unknown2, @TimeDeposited, @TimeExchanged, @pid, " + "@TrainerOT, @TrainerName, @TrainerCountry, @TrainerRegion, @TrainerClass, " + "@IsExchanged, @TrainerVersion, @TrainerLanguage, @TrainerBadges, " + "@TrainerUnityTower, @TimeWithdrawn, @trade_id, @partner_pid)", _params2); } #endregion #region Battle Subway 5 public override ulong BattleSubwayUpdateRecord5(BattleSubwayRecord5 record) { if (record.BattlesWon > 7) throw new ArgumentException("Battles won can not be greater than 7."); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { ulong result = BattleSubwayUpdateRecord5(tran, record); tran.Commit(); return result; } } } private ulong BattleSubwayUpdateRecord5(MySqlTransaction tran, BattleSubwayRecord5 record) { if (record.BattlesWon > 7) throw new ArgumentException("Battles won can not be greater than 7."); // Does this player already have a record in this room? // Also get primary key if it does. (We need it for updating party) ulong pkey = FindBattleSubwayRecord5(tran, record, false); if (pkey != 0) { // If the player already has a record, move everyone below it up one position // (effectively removing this record from the ordering) // todo: In the case that the player's rank hasn't changed, // we can optimize this and the next down to a single BETWEEN // query. // This does require retrieving their old rank from the db. tran.ExecuteNonQuery("SELECT @old_rank = Rank, @old_position = Position " + "FROM GtsBattleSubway5 WHERE pid = @pid AND RoomNum = @room); " + "UPDATE GtsBattleSubway5 SET Position = Position - 1 " + "WHERE RoomNum = @room AND Rank = @old_rank AND Position > @old_position", new MySqlParameter("@pid", record.PID), new MySqlParameter("@room", record.RoomNum)); } uint position = (uint)(7 - record.BattlesWon); // Shift down all the players in the player's new rank by one. tran.ExecuteNonQuery("UPDATE GtsBattleSubway5 SET Position = Position + 1 " + "WHERE RoomNum = @room AND Rank = @rank AND Position >= @position", new MySqlParameter("@room", record.RoomNum), new MySqlParameter("@rank", record.Rank), new MySqlParameter("@position", position)); object lastPosition = tran.ExecuteScalar("SELECT MAX(Position) " + "FROM GtsBattleSubway5 WHERE RoomNum = @room AND Rank = @rank", new MySqlParameter("@room", record.RoomNum), new MySqlParameter("@rank", record.Rank)); // If the room has fewer than 7 trainers, insert this one at the // end but don't leave any gaps in the numbering. if (lastPosition is DBNull) position = 0; else position = Math.Min(position, (uint)lastPosition + 1); // Update the actual record if (pkey != 0) { List _params = ParamsFromBattleSubwayRecord5(record, false); _params.Add(new MySqlParameter("@position", position)); _params.Add(new MySqlParameter("@id", pkey)); tran.ExecuteNonQuery("UPDATE GtsBattleSubway5 SET pid = @pid, Name = @name, " + "Version = @version, Language = @language, Country = @country, " + "Region = @region, TrainerID = @trainer_id, " + "PhraseLeader = @phrase_leader, Gender = @gender, " + "Unknown2 = @unknown2, PhraseChallenged = @phrase_challenged, " + "PhraseWon = @phrase_won, PhraseLost = @phrase_lost, Unknown3 = @unknown3, " + "Unknown4 = @unknown4, Unknown5 = @unknown5, ParseVersion = 1, Rank = @rank, " + "BattlesWon = @battles_won, Position = @position, " + "TimeUpdated = UTC_TIMESTAMP() WHERE id = @id", _params.ToArray()); UpdateBattleSubwayPokemon5(tran, record.Party[0], pkey, 0); UpdateBattleSubwayPokemon5(tran, record.Party[1], pkey, 1); UpdateBattleSubwayPokemon5(tran, record.Party[2], pkey, 2); } else { List _params = ParamsFromBattleSubwayRecord5(record, false); _params.Add(new MySqlParameter("@position", position)); pkey = Convert.ToUInt64(tran.ExecuteScalar("INSERT INTO GtsBattleSubway5 " + "(pid, Name, Version, Language, Country, Region, TrainerID, " + "PhraseLeader, Gender, Unknown2, PhraseChallenged, PhraseWon, " + "PhraseLost, Unknown3, Unknown4, Unknown5, ParseVersion, " + "Rank, RoomNum, BattlesWon, Position, TimeAdded, TimeUpdated) VALUES " + "(@pid, @name, @version, @language, @country, @region, @trainer_id, " + "@phrase_leader, @gender, @unknown2, @phrase_challenged, @phrase_won, " + "@phrase_lost, @unknown3, @unknown4, @unknown5, 1, " + "@rank, @room, @battles_won, @position, UTC_TIMESTAMP(), UTC_TIMESTAMP()); " + "SELECT LAST_INSERT_ID()", _params.ToArray())); InsertBattleSubwayPokemon5(tran, record.Party[0], pkey, 0); InsertBattleSubwayPokemon5(tran, record.Party[1], pkey, 1); InsertBattleSubwayPokemon5(tran, record.Party[2], pkey, 2); } return pkey; } private void InsertBattleSubwayPokemon5(MySqlTransaction tran, BattleSubwayPokemon5 pokemon, ulong partyId, byte slot) { List _params = ParamsFromBattleSubwayPokemon5(pokemon); _params.Add(new MySqlParameter("@id", partyId)); _params.Add(new MySqlParameter("@slot", slot)); tran.ExecuteNonQuery("INSERT INTO GtsBattleSubwayPokemon5 " + "(party_id, Slot, Species, HeldItem, Move1, Move2, Move3, Move4, TrainerID, " + "Personality, IVs, EVs, Unknown1, Language, Ability, Happiness, " + "Nickname, Unknown2) VALUES " + "(@id, @slot, @species, @held_item, @move1, @move2, @move3, @move4, @trainer_id, " + "@personality, @ivs, @evs, @unknown1, @language, @ability, @happiness, " + "@nickname, @unknown2)", _params.ToArray()); } private void UpdateBattleSubwayPokemon5(MySqlTransaction tran, BattleSubwayPokemon5 pokemon, ulong partyId, byte slot) { List _params = ParamsFromBattleSubwayPokemon5(pokemon); _params.Add(new MySqlParameter("@id", partyId)); _params.Add(new MySqlParameter("@slot", slot)); tran.ExecuteNonQuery("UPDATE GtsBattleSubwayPokemon5 SET Species = @species, " + "HeldItem = @held_item, Move1 = @move1, Move2 = @move2, Move3 = @move3, " + "Move4 = @move4, TrainerID = @trainer_id, Personality = @personality, " + "IVs = @ivs, EVs = @evs, Unknown1 = @unknown1, Language = @language, " + "Ability = @ability, Happiness = @happiness, Nickname = @nickname, " + "Unknown2 = @unknown2 " + "WHERE party_id = @id AND Slot = @slot", _params.ToArray()); } private List ParamsFromBattleSubwayRecord5(BattleSubwayRecord5 record, bool leader) { List result = new List(15); result.Add(new MySqlParameter("@pid", record.PID)); result.Add(new MySqlParameter("@name", record.Profile.Name.RawData)); result.Add(new MySqlParameter("@version", (byte)record.Profile.Version)); result.Add(new MySqlParameter("@language", (byte)record.Profile.Language)); result.Add(new MySqlParameter("@country", record.Profile.Country)); result.Add(new MySqlParameter("@region", record.Profile.Region)); result.Add(new MySqlParameter("@trainer_id", record.Profile.OT)); result.Add(new MySqlParameter("@phrase_leader", record.Profile.PhraseLeader.Data)); result.Add(new MySqlParameter("@gender", record.Profile.Gender)); result.Add(new MySqlParameter("@unknown2", record.Profile.Unknown)); result.Add(new MySqlParameter("@rank", record.Rank)); result.Add(new MySqlParameter("@room", record.RoomNum)); if (!leader) { result.Add(new MySqlParameter("@phrase_challenged", record.PhraseChallenged.Data)); result.Add(new MySqlParameter("@phrase_won", record.PhraseWon.Data)); result.Add(new MySqlParameter("@phrase_lost", record.PhraseLost.Data)); result.Add(new MySqlParameter("@unknown3", record.Unknown3)); result.Add(new MySqlParameter("@unknown4", record.Unknown4 ?? new byte[5])); result.Add(new MySqlParameter("@unknown5", record.Unknown5)); result.Add(new MySqlParameter("@battles_won", record.BattlesWon)); } return result; } private List ParamsFromBattleSubwayPokemon5(BattleSubwayPokemon5 pokemon) { List result = new List(15); result.Add(new MySqlParameter("@species", pokemon.Species)); result.Add(new MySqlParameter("@held_item", pokemon.HeldItem)); result.Add(new MySqlParameter("@move1", pokemon.Moveset[0])); result.Add(new MySqlParameter("@move2", pokemon.Moveset[1])); result.Add(new MySqlParameter("@move3", pokemon.Moveset[2])); result.Add(new MySqlParameter("@move4", pokemon.Moveset[3])); result.Add(new MySqlParameter("@trainer_id", pokemon.OT)); result.Add(new MySqlParameter("@personality", pokemon.Personality)); result.Add(new MySqlParameter("@ivs", pokemon.IVs)); result.Add(new MySqlParameter("@evs", pokemon.EVs)); result.Add(new MySqlParameter("@unknown1", pokemon.Unknown1)); result.Add(new MySqlParameter("@language", (byte)pokemon.Language)); result.Add(new MySqlParameter("@ability", pokemon.Ability)); result.Add(new MySqlParameter("@happiness", pokemon.Happiness)); result.Add(new MySqlParameter("@nickname", pokemon.Nickname.RawData)); result.Add(new MySqlParameter("@unknown2", pokemon.Unknown2)); return result; } public override ulong BattleSubwayAddLeader5(BattleSubwayRecord5 record) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { ulong result = BattleSubwayAddLeader5(tran, record); tran.Commit(); return result; } } } private ulong BattleSubwayAddLeader5(MySqlTransaction tran, BattleSubwayRecord5 record) { ulong pkey = FindBattleSubwayRecord5(tran, record, true); // Update the actual record if (pkey != 0) { List _params = ParamsFromBattleSubwayRecord5(record, true); _params.Add(new MySqlParameter("@id", pkey)); tran.ExecuteNonQuery("UPDATE GtsBattleSubwayLeaders5 SET " + "pid = @pid, Name = @name, Version = @version, " + "Language = @language, Country = @country, Region = @region, " + "TrainerID = @trainer_id, " + "PhraseLeader = @phrase_leader, Gender = @gender, Unknown2 = @unknown2, " + "ParseVersion = 1, Rank = @rank, " + "TimeUpdated = UTC_TIMESTAMP() WHERE id = @id", _params.ToArray()); } else { List _params = ParamsFromBattleSubwayRecord5(record, true); pkey = Convert.ToUInt64(tran.ExecuteScalar("INSERT INTO " + "GtsBattleSubwayLeaders5 " + "(pid, Name, Version, Language, Country, Region, TrainerID, " + "PhraseLeader, Gender, Unknown2, ParseVersion, Rank, " + "RoomNum, TimeAdded, TimeUpdated) VALUES " + "(@pid, @name, @version, @language, @country, @region, @trainer_id, " + "@phrase_leader, @gender, @unknown2, 1, @rank, " + "@room, UTC_TIMESTAMP(), UTC_TIMESTAMP()); " + "SELECT LAST_INSERT_ID()", _params.ToArray())); } return pkey; } /// /// Tries to find an existing database record for the provided player /// record. The match must be found in the same rank and room number. /// /// /// /// If true, look up against the Leaders table. /// Otherwise looks up against the opponents table. /// The match's primary key or 0 if no match is found /// private ulong FindBattleSubwayRecord5(MySqlTransaction tran, BattleSubwayRecord5 record, bool leader) { String tblName = leader ? "GtsBattleSubwayLeaders5" : "GtsBattleSubway5"; // If PID is missing, this is restored data. // We assume the original server took care of matching existing // records, so we don't allow it to match here. if (record.PID == 0) return 0; // Match normally. object oPkey = tran.ExecuteScalar("SELECT id FROM " + tblName + " WHERE pid = @pid AND RoomNum = @room AND Rank = @rank", // Only require rank to match if this is the leaderboard. new MySqlParameter("@pid", record.PID), new MySqlParameter("@rank", record.Rank), new MySqlParameter("@room", record.RoomNum)); if (oPkey == null) { // PID isn't found. Try to match one of Pikachu025's saved // records based on unchanging properties of the savegame. oPkey = tran.ExecuteScalar("SELECT id FROM " + tblName + " WHERE pid = 0 AND RoomNum = @room AND Rank = @rank " + // Only require rank to match if this is the leaderboard. "AND Name = @name AND Version = @version " + "AND Language = @language AND TrainerID = @trainer_id", new MySqlParameter("@rank", record.Rank), new MySqlParameter("@room", record.RoomNum), new MySqlParameter("@name", record.Profile.Name), new MySqlParameter("@version", (byte)record.Profile.Version), new MySqlParameter("@language", (byte)record.Profile.Language), new MySqlParameter("@trainer_id", record.Profile.OT) ); } // Don't need to worry about DBNull since the column is non-null. return (ulong)(oPkey ?? 0UL); } public override BattleSubwayRecord5[] BattleSubwayGetOpponents5(int pid, byte rank, byte roomNum) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { List records = new List(7); List keys = new List(7); MySqlDataReader reader = (MySqlDataReader)tran.ExecuteReader( "SELECT id, pid, Name, " + "Version, Language, Country, Region, TrainerID, " + "PhraseLeader, Gender, Unknown2, PhraseChallenged, " + "PhraseWon, PhraseLost, Unknown3, Unknown4, Unknown5 " + "FROM GtsBattleSubway5 " + "WHERE Rank = @rank AND RoomNum = @room AND pid != @pid " + "ORDER BY Position LIMIT 7", new MySqlParameter("@rank", rank), new MySqlParameter("@room", roomNum), new MySqlParameter("@pid", pid)); while (reader.Read()) { BattleSubwayRecord5 record = BattleSubwayRecord5FromReader(reader); record.Party = new BattleSubwayPokemon5[3]; records.Add(record); keys.Add(reader.GetUInt64(0)); } reader.Close(); if (records.Count == 0) return new BattleSubwayRecord5[0]; String inClause = String.Join(", ", keys.Select(i => i.ToString()).ToArray()); reader = (MySqlDataReader)tran.ExecuteReader("SELECT party_id, " + "Slot, Species, HeldItem, Move1, Move2, Move3, Move4, " + "TrainerID, Personality, IVs, EVs, Unknown1, Language, " + "Ability, Happiness, Nickname, Unknown2 FROM GtsBattleSubwayPokemon5 " + "WHERE party_id IN (" + inClause + ")"); while (reader.Read()) { BattleSubwayRecord5 record = records[keys.IndexOf(reader.GetUInt64(0))]; record.Party[reader.GetByte(1)] = BattleSubwayPokemon5FromReader(reader); } reader.Close(); tran.Commit(); return Enumerable.Reverse(records).ToArray(); } } } private BattleSubwayRecord5 BattleSubwayRecord5FromReader(MySqlDataReader reader) { // todo: Stop using ordinals everywhere. BattleSubwayRecord5 result = new BattleSubwayRecord5(); result.PID = reader.GetInt32(1); // this is unsustainable. What happens if I add columns to Leaders? if (reader.FieldCount > 11) result.PhraseChallenged = new TrendyPhrase5(reader.GetByteArray(11, 8)); if (reader.FieldCount > 12) result.PhraseWon = new TrendyPhrase5(reader.GetByteArray(12, 8)); if (reader.FieldCount > 13) result.PhraseLost = new TrendyPhrase5(reader.GetByteArray(13, 8)); if (reader.FieldCount > 14) result.Unknown3 = reader.GetUInt16(14); if (reader.FieldCount > 15 && !reader.IsDBNull(15)) result.Unknown4 = reader.GetByteArray(15, 5); if (reader.FieldCount > 16) result.Unknown5 = reader.GetUInt64(16); BattleSubwayProfile5 profile = new BattleSubwayProfile5(); profile.Name = new EncodedString5(reader.GetByteArray(2, 16)); profile.Version = (Versions)reader.GetByte(3); profile.Language = (Languages)reader.GetByte(4); profile.Country = reader.GetByte(5); profile.Region = reader.GetByte(6); profile.OT = reader.GetUInt32(7); profile.PhraseLeader = new TrendyPhrase5(reader.GetByteArray(8, 8)); profile.Gender = reader.GetByte(9); profile.Unknown = reader.GetByte(10); result.Profile = profile; return result; } private BattleSubwayPokemon5 BattleSubwayPokemon5FromReader(MySqlDataReader reader) { BattleSubwayPokemon5 result = new BattleSubwayPokemon5(); result.Species = reader.GetUInt16(2); result.HeldItem = reader.GetUInt16(3); result.Moveset = new ushort[4]; result.Moveset[0] = reader.GetUInt16(4); result.Moveset[1] = reader.GetUInt16(5); result.Moveset[2] = reader.GetUInt16(6); result.Moveset[3] = reader.GetUInt16(7); result.OT = reader.GetUInt32(8); result.Personality = reader.GetUInt32(9); result.IVs = reader.GetUInt32(10); result.EVs = reader.GetByteArray(11, 6); result.Unknown1 = reader.GetByte(12); result.Language = (Languages)reader.GetByte(13); result.Ability = reader.GetByte(14); result.Happiness = reader.GetByte(15); result.Nickname = new EncodedString5(reader.GetByteArray(16, 22)); result.Unknown2 = reader.GetUInt32(17); return result; } public override BattleSubwayProfile5[] BattleSubwayGetLeaders5(byte rank, byte roomNum) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { List profiles = new List(30); MySqlDataReader reader = (MySqlDataReader)tran.ExecuteReader( "SELECT id, pid, Name, " + "Version, Language, Country, Region, TrainerID, " + "PhraseLeader, Gender, Unknown2 FROM GtsBattleSubwayLeaders5 " + "WHERE Rank = @rank AND RoomNum = @room " + "ORDER BY TimeUpdated DESC, id LIMIT 30", new MySqlParameter("@rank", rank), new MySqlParameter("@room", roomNum)); while (reader.Read()) { profiles.Add(BattleSubwayRecord5FromReader(reader).Profile); } reader.Close(); tran.Commit(); return profiles.ToArray(); } } } #endregion #region Other Gamestats 5 public override bool GamestatsSetProfile5(TrainerProfile5 profile) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { bool result = GamestatsSetProfile5(tran, profile); tran.Commit(); return result; } } } public bool GamestatsSetProfile5(MySqlTransaction tran, TrainerProfile5 profile) { if (profile.Data.Length != 100) throw new FormatException("Profile data must be 100 bytes."); long exists = (long)tran.ExecuteScalar("SELECT EXISTS(SELECT * FROM GtsProfiles5 WHERE pid = @pid)", new MySqlParameter("@pid", profile.PID)); MySqlParameter[] _params = new MySqlParameter[]{ new MySqlParameter("@pid", profile.PID), new MySqlParameter("@data", profile.Data), new MySqlParameter("@version", (byte)profile.Version), new MySqlParameter("@language", (byte)profile.Language), new MySqlParameter("@country", profile.Country), new MySqlParameter("@region", profile.Region), new MySqlParameter("@ot", profile.OT), new MySqlParameter("@name", profile.Name.RawData) }; if (exists != 0) { return tran.ExecuteNonQuery("UPDATE GtsProfiles5 SET Data = @data, " + "Version = @version, Language = @language, Country = @country, " + "Region = @region, OT = @ot, Name = @name, ParseVersion = 1, " + "TimeUpdated = UTC_TIMESTAMP() " + "WHERE pid = @pid", _params) > 0; } else { return tran.ExecuteNonQuery("INSERT INTO GtsProfiles5 " + "(pid, Data, Version, Language, Country, Region, OT, Name, " + "ParseVersion, TimeAdded, TimeUpdated) VALUES " + "(@pid, @data, @version, @language, @country, @region, @ot, " + "@name, 1, UTC_TIMESTAMP(), UTC_TIMESTAMP())", _params) > 0; } } #endregion #region Global Terminal 4 public override ulong DressupUpload4(DressupRecord4 record) { if (record.Data.Length != 224) throw new ArgumentException("Dressup data must be 224 bytes."); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { long exists = (long)tran.ExecuteScalar("SELECT EXISTS(SELECT * FROM TerminalDressup4 WHERE md5 = unhex(md5(@data)) AND Data = @data)", new MySqlParameter("@data", record.Data)); if (exists != 0) return 0; if (record.SerialNumber == 0) { ulong serial = Convert.ToUInt64(tran.ExecuteScalar("INSERT INTO TerminalDressup4 (pid, " + "Data, md5, TimeAdded, ParseVersion, Species) VALUES (@pid, @data, " + "unhex(md5(@data)), UTC_TIMESTAMP(), 1, @species); SELECT LAST_INSERT_ID()", new MySqlParameter("@pid", record.PID), new MySqlParameter("@data", record.Data), new MySqlParameter("@species", record.Species))); tran.Commit(); return serial; } else { int rows = tran.ExecuteNonQuery("INSERT INTO TerminalDressup4 (pid, SerialNumber, " + "Data, md5, TimeAdded, ParseVersion, Species) VALUES (@pid, @serial, @data, " + "unhex(md5(@data)), UTC_TIMESTAMP(), 1, @species)", new MySqlParameter("@pid", record.PID), new MySqlParameter("@serial", record.SerialNumber), new MySqlParameter("@data", record.Data), new MySqlParameter("@species", record.Species)); tran.Commit(); return rows > 0 ? record.SerialNumber : 0; } } } } public override DressupRecord4[] DressupSearch4(ushort species, int count) { using (MySqlConnection db = CreateConnection()) { db.Open(); List results = new List(count); MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT pid, " + "SerialNumber, Data FROM TerminalDressup4 WHERE Species = @species " + "ORDER BY TimeAdded DESC LIMIT @count", new MySqlParameter("@species", species), new MySqlParameter("@count", count)); while (reader.Read()) { results.Add(Dressup4FromReader(reader)); } reader.Close(); db.Close(); return results.ToArray(); } } private DressupRecord4 Dressup4FromReader(MySqlDataReader reader) { byte[] data = new byte[224]; reader.GetBytes(2, 0, data, 0, 224); return new DressupRecord4(reader.GetInt32(0), reader.GetUInt64(1), data); } public override ulong BoxUpload4(BoxRecord4 record) { if (record.Data.Length != 540) throw new ArgumentException("Box data must be 540 bytes."); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { long exists = (long)tran.ExecuteScalar("SELECT EXISTS(SELECT * FROM TerminalBoxes4 WHERE md5 = unhex(md5(@data)) AND Data = @data)", new MySqlParameter("@data", record.Data)); if (exists != 0) return 0; if (record.SerialNumber == 0) { ulong serial = Convert.ToUInt64(tran.ExecuteScalar("INSERT INTO TerminalBoxes4 (pid, " + "Data, md5, TimeAdded, ParseVersion, Label) VALUES (@pid, @data, " + "unhex(md5(@data)), UTC_TIMESTAMP(), 1, @label); SELECT LAST_INSERT_ID()", new MySqlParameter("@pid", record.PID), new MySqlParameter("@data", record.Data), new MySqlParameter("@label", (int)record.Label))); tran.Commit(); return serial; } else { int rows = tran.ExecuteNonQuery("INSERT INTO TerminalBoxes4 (pid, SerialNumber, " + "Data, md5, TimeAdded, ParseVersion, Label) VALUES (@pid, @serial, @data, " + "unhex(md5(@data)), UTC_TIMESTAMP(), 1, @label)", new MySqlParameter("@pid", record.PID), new MySqlParameter("@serial", record.SerialNumber), new MySqlParameter("@data", record.Data), new MySqlParameter("@label", (int)record.Label)); tran.Commit(); return rows > 0 ? record.SerialNumber : 0; } } } } public override BoxRecord4[] BoxSearch4(BoxLabels4 label, int count) { using (MySqlConnection db = CreateConnection()) { db.Open(); List results = new List(count); MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT pid, " + "Label, SerialNumber, Data FROM TerminalBoxes4 WHERE Label = @label " + "ORDER BY TimeAdded DESC LIMIT @count", new MySqlParameter("@label", (int)label), new MySqlParameter("@count", count)); while (reader.Read()) { results.Add(Box4FromReader(reader)); } reader.Close(); db.Close(); return results.ToArray(); } } private BoxRecord4 Box4FromReader(MySqlDataReader reader) { byte[] data = new byte[540]; reader.GetBytes(3, 0, data, 0, 540); return new BoxRecord4(reader.GetInt32(0), (BoxLabels4)reader.GetInt32(1), reader.GetUInt64(2), data); } public override ulong BattleVideoUpload4(BattleVideoRecord4 record) { if (record.Data.Length != 7272) throw new ArgumentException(); if (record.Header.Data.Length != 228) throw new ArgumentException(); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { long exists = (long)tran.ExecuteScalar("SELECT EXISTS(SELECT * " + "FROM TerminalBattleVideos4 WHERE md5 = unhex(md5(CONCAT(@header, @data))) " + "AND Data = @data AND Header = @header)", new MySqlParameter("@header", record.Header.Data), new MySqlParameter("@data", record.Data)); if (exists != 0) return 0; if (record.SerialNumber == 0) { ulong key = Convert.ToUInt64(tran.ExecuteScalar("INSERT INTO TerminalBattleVideos4 " + "(pid, Header, Data, md5, TimeAdded, ParseVersion, Streak, TrainerName, " + "Metagame, Country, Region) " + "VALUES (@pid, @header, @data, unhex(md5(CONCAT(@header, @data))), " + "UTC_TIMESTAMP(), 1, @streak, @trainer, @metagame, @country, @region); " + "SELECT LAST_INSERT_ID()", new MySqlParameter("@pid", record.PID), new MySqlParameter("@header", record.Header.Data), new MySqlParameter("@data", record.Data), new MySqlParameter("@streak", record.Header.Streak), new MySqlParameter("@trainer", record.Header.TrainerName), new MySqlParameter("@metagame", (byte)record.Header.Metagame), new MySqlParameter("@country", (byte)record.Header.Country), new MySqlParameter("@region", (byte)record.Header.Region) )); ulong serial = BattleVideoHeader4.KeyToSerial(key); tran.ExecuteNonQuery("UPDATE TerminalBattleVideos4 SET " + "SerialNumber = @serial WHERE id = @key", new MySqlParameter("@serial", serial), new MySqlParameter("@key", key)); // todo: make a proc to insert both video and party. InsertBattleVideoParty4(record.Header, key, tran); tran.Commit(); return serial; } else { ulong key = BattleVideoHeader4.SerialToKey(record.SerialNumber); int rows = tran.ExecuteNonQuery("INSERT INTO TerminalBattleVideos4 " + "(id, pid, SerialNumber, Header, Data, md5, TimeAdded, " + "ParseVersion, Streak, TrainerName, " + "Metagame, Country, Region) " + "VALUES (@key, @pid, @serial, @header, @data, " + "unhex(md5(CONCAT(@header, @data))), " + "UTC_TIMESTAMP(), 1, @streak, @trainer, @metagame, @country, @region)", new MySqlParameter("@key", key), new MySqlParameter("@pid", record.PID), new MySqlParameter("@serial", record.SerialNumber), new MySqlParameter("@header", record.Header.Data), new MySqlParameter("@data", record.Data), new MySqlParameter("@streak", record.Header.Streak), new MySqlParameter("@trainer", record.Header.TrainerName), new MySqlParameter("@metagame", (byte)record.Header.Metagame), new MySqlParameter("@country", (byte)record.Header.Country), new MySqlParameter("@region", (byte)record.Header.Region) ); if (rows == 0) return 0; InsertBattleVideoParty4(record.Header, key, tran); tran.Commit(); return record.SerialNumber; } } } } private void InsertBattleVideoParty4(BattleVideoHeader4 header, ulong key, MySqlTransaction tran) { MySqlCommand cmd = new MySqlCommand("INSERT INTO " + "TerminalBattleVideoPokemon4 (video_id, Slot, Species) VALUES " + "(@key, @slot, @species)", tran.Connection, tran); cmd.Parameters.Add("@key", MySqlDbType.UInt64).Value = key; cmd.Parameters.Add("@slot", MySqlDbType.UByte); cmd.Parameters.Add("@species", MySqlDbType.UInt16); ushort[] party = header.Party; for (byte x = 0; x < 12; x++) { ushort species = party[x]; if (species == 0) continue; cmd.Parameters["@slot"].Value = x; cmd.Parameters["@species"].Value = species; cmd.ExecuteNonQuery(); } } public override BattleVideoHeader4[] BattleVideoSearch4(ushort species, BattleVideoRankings4 ranking, BattleVideoMetagames4 metagame, byte country, byte region, int count) { using (MySqlConnection db = CreateConnection()) { List _params = new List(); String where = ""; String sort = ""; bool hasSearch = false; if (ranking == BattleVideoRankings4.None) { if (species != 0xffff) { where += " WHERE EXISTS(SELECT * FROM TerminalBattleVideoPokemon4 " + "WHERE video_id = TerminalBattleVideos4.id AND Species = @species)"; _params.Add(new MySqlParameter("@species", species)); hasSearch = true; } if (metagame == BattleVideoMetagames4.SearchColosseumSingleNoRestrictions) metagame = BattleVideoMetagames4.ColosseumSingleNoRestrictions; if (metagame == BattleVideoMetagames4.SearchColosseumDoubleNoRestrictions) metagame = BattleVideoMetagames4.ColosseumDoubleNoRestrictions; if (metagame == BattleVideoMetagames4.SearchColosseumSingleCupMatch) { where += (hasSearch ? " AND " : " WHERE ") + "Metagame BETWEEN 1 AND 6"; hasSearch = true; } else if (metagame == BattleVideoMetagames4.SearchColosseumDoubleCupMatch) { where += (hasSearch ? " AND " : " WHERE ") + "Metagame BETWEEN 8 AND 13"; hasSearch = true; } else if (metagame != BattleVideoMetagames4.SearchLatest30) { where += (hasSearch ? " AND " : " WHERE ") + "Metagame = @metagame"; _params.Add(new MySqlParameter("@metagame", (byte)metagame)); hasSearch = true; } if (country != 0xff) { where += (hasSearch ? " AND " : " WHERE ") + "Country = @country"; _params.Add(new MySqlParameter("@country", country)); hasSearch = true; } if (region != 0xff) { where += (hasSearch ? " AND " : " WHERE ") + "Region = @region"; _params.Add(new MySqlParameter("@region", region)); } sort = " ORDER BY TimeAdded DESC, id DESC"; } else if (ranking == BattleVideoRankings4.Colosseum) { // todo: sort by .. something. where = " WHERE Metagame BETWEEN 0 AND 14"; sort = " ORDER BY Streak DESC, TimeAdded DESC, id DESC"; } else if (ranking == BattleVideoRankings4.BattleFrontier) { where = " WHERE NOT (Metagame BETWEEN 0 AND 14)"; sort = " ORDER BY Streak DESC, TimeAdded DESC, id DESC"; } else { sort = " ORDER BY TimeAdded DESC, id DESC"; } _params.Add(new MySqlParameter("@count", count)); db.Open(); List results = new List(count); MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT pid, " + "SerialNumber, Header FROM TerminalBattleVideos4" + where + sort + " LIMIT @count", _params.ToArray()); while (reader.Read()) { results.Add(BattleVideoHeader4FromReader(reader)); } reader.Close(); db.Close(); return results.ToArray(); } } private BattleVideoHeader4 BattleVideoHeader4FromReader(MySqlDataReader reader) { byte[] data = new byte[228]; reader.GetBytes(2, 0, data, 0, 228); return new BattleVideoHeader4(reader.GetInt32(0), reader.GetUInt64(1), data); } public override BattleVideoRecord4 BattleVideoGet4(ulong serial, bool incrementViews = false) { String update = incrementViews ? "UPDATE TerminalBattleVideos4 " + "SET Views = Views + 1 WHERE SerialNumber = @serial; " : ""; using (MySqlConnection db = CreateConnection()) { db.Open(); MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader(update + "SELECT pid, " + "SerialNumber, Header, Data FROM TerminalBattleVideos4 " + "WHERE SerialNumber = @serial", new MySqlParameter("@serial", serial)); if (reader.Read()) return BattleVideo4FromReader(reader); else return null; } } private BattleVideoRecord4 BattleVideo4FromReader(MySqlDataReader reader) { byte[] data = new byte[7272]; reader.GetBytes(3, 0, data, 0, 7272); BattleVideoHeader4 header = BattleVideoHeader4FromReader(reader); return new BattleVideoRecord4(header.PID, header.SerialNumber, header, data); } public override bool BattleVideoFlagSaved4(ulong serial) { using (MySqlConnection db = CreateConnection()) { db.Open(); int results = db.ExecuteNonQuery("UPDATE TerminalBattleVideos4 " + "SET Saves = Saves + 1 WHERE SerialNumber = @serial", new MySqlParameter("@serial", serial)); db.Close(); return results > 0; } } public override ulong BattleVideoCount4() { using (MySqlConnection db = CreateConnection()) { db.Open(); return Convert.ToUInt64(db.ExecuteScalar("SELECT Count(*) FROM TerminalBattleVideos4")); } } #endregion #region Global Terminal 5 public override ulong MusicalUpload5(MusicalRecord5 record) { if (record.Data.Length != 560) throw new ArgumentException(); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { long exists = (long)tran.ExecuteScalar("SELECT EXISTS(SELECT * " + "FROM TerminalMusicals5 WHERE md5 = unhex(md5(@data)) " + "AND Data = @data)", new MySqlParameter("@data", record.Data)); if (exists != 0) return 0; if (record.SerialNumber == 0) { ulong serial = Convert.ToUInt64(tran.ExecuteScalar("INSERT INTO TerminalMusicals5 " + "(pid, Data, md5, TimeAdded, ParseVersion) " + "VALUES (@pid, @data, unhex(md5(@data)), " + "UTC_TIMESTAMP(), 1); " + "SELECT LAST_INSERT_ID()", new MySqlParameter("@pid", record.PID), new MySqlParameter("@data", record.Data) )); // todo: make a proc to insert both musical and party. InsertMusicalParticipants5(record, serial, tran); tran.Commit(); return serial; } else { int rows = tran.ExecuteNonQuery("INSERT INTO TerminalMusicals5 " + "(pid, SerialNumber, Data, md5, TimeAdded, ParseVersion) " + "VALUES (@pid, @serial, @data, unhex(md5(@data)), " + "UTC_TIMESTAMP(), 1)", new MySqlParameter("@pid", record.PID), new MySqlParameter("@serial", record.SerialNumber), new MySqlParameter("@data", record.Data) ); if (rows == 0) return 0; InsertMusicalParticipants5(record, record.SerialNumber, tran); tran.Commit(); return record.SerialNumber; } } } } private void InsertMusicalParticipants5(MusicalRecord5 record, ulong SerialNumber, MySqlTransaction tran) { MySqlCommand cmd = new MySqlCommand("INSERT INTO " + "TerminalMusicalPokemon5 (musical_id, Slot, Species) VALUES " + "(@serial, @slot, @species)", tran.Connection, tran); cmd.Parameters.Add("@serial", MySqlDbType.UInt64).Value = SerialNumber; cmd.Parameters.Add("@slot", MySqlDbType.UByte); cmd.Parameters.Add("@species", MySqlDbType.UInt16); MusicalParticipant5[] participants = record.Participants; for (byte x = 0; x < 4; x++) { ushort species = participants[x].Species; if (species == 0) continue; cmd.Parameters["@slot"].Value = x; cmd.Parameters["@species"].Value = species; cmd.ExecuteNonQuery(); } } public override MusicalRecord5[] MusicalSearch5(ushort species, int count) { using (MySqlConnection db = CreateConnection()) { db.Open(); List results = new List(count); MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT pid, " + "SerialNumber, Data FROM TerminalMusicals5 " + "WHERE EXISTS(SELECT * FROM TerminalMusicalPokemon5 " + "WHERE musical_id = TerminalMusicals5.SerialNumber AND Species = @species) " + "ORDER BY TimeAdded DESC LIMIT @count", new MySqlParameter("@species", species), new MySqlParameter("@count", count)); while (reader.Read()) { results.Add(Musical5FromReader(reader)); } reader.Close(); db.Close(); return results.ToArray(); } } private MusicalRecord5 Musical5FromReader(MySqlDataReader reader) { byte[] data = new byte[560]; reader.GetBytes(2, 0, data, 0, 560); return new MusicalRecord5(reader.GetInt32(0), reader.GetUInt64(1), data); } public override ulong BattleVideoUpload5(BattleVideoRecord5 record) { if (record.Data.Length != 6112) throw new ArgumentException(); if (record.Header.Data.Length != 196) throw new ArgumentException(); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlTransaction tran = db.BeginTransaction()) { long exists = (long)tran.ExecuteScalar("SELECT EXISTS(SELECT * " + "FROM TerminalBattleVideos5 WHERE md5 = unhex(md5(CONCAT(@header, @data))) " + "AND Data = @data AND Header = @header)", new MySqlParameter("@header", record.Header.Data), new MySqlParameter("@data", record.Data)); if (exists != 0) return 0; if (record.SerialNumber == 0) { ulong key = Convert.ToUInt64(tran.ExecuteScalar("INSERT INTO TerminalBattleVideos5 " + "(pid, Header, Data, md5, TimeAdded, ParseVersion, Streak, TrainerName, " + "Metagame, Country, Region) " + "VALUES (@pid, @header, @data, unhex(md5(CONCAT(@header, @data))), " + "UTC_TIMESTAMP(), 1, @streak, @trainer, @metagame, @country, @region); " + "SELECT LAST_INSERT_ID()", new MySqlParameter("@pid", record.PID), new MySqlParameter("@header", record.Header.Data), new MySqlParameter("@data", record.Data), new MySqlParameter("@streak", record.Header.Streak), new MySqlParameter("@trainer", record.Header.TrainerName), new MySqlParameter("@metagame", (byte)record.Header.Metagame), new MySqlParameter("@country", (byte)record.Header.Country), new MySqlParameter("@region", (byte)record.Header.Region) )); ulong serial = BattleVideoHeader4.KeyToSerial(key); tran.ExecuteNonQuery("UPDATE TerminalBattleVideos5 SET " + "SerialNumber = @serial WHERE id = @key", new MySqlParameter("@serial", serial), new MySqlParameter("@key", key)); // todo: make a proc to insert both video and party. InsertBattleVideoParty5(record.Header, key, tran); tran.Commit(); return serial; } else { ulong key = (ulong)BattleVideoHeader4.SerialToKey(record.SerialNumber); int rows = tran.ExecuteNonQuery("INSERT INTO TerminalBattleVideos5 " + "(id, pid, SerialNumber, Header, Data, md5, TimeAdded, " + "ParseVersion, Streak, TrainerName, " + "Metagame, Country, Region) " + "VALUES (@key, @pid, @serial, @header, @data, " + "unhex(md5(CONCAT(@header, @data))), " + "UTC_TIMESTAMP(), 1, @streak, @trainer, @metagame, @country, @region)", new MySqlParameter("@key", key), new MySqlParameter("@pid", record.PID), new MySqlParameter("@serial", record.SerialNumber), new MySqlParameter("@header", record.Header.Data), new MySqlParameter("@data", record.Data), new MySqlParameter("@streak", record.Header.Streak), new MySqlParameter("@trainer", record.Header.TrainerName), new MySqlParameter("@metagame", (byte)record.Header.Metagame), new MySqlParameter("@country", (byte)record.Header.Country), new MySqlParameter("@region", (byte)record.Header.Region) ); if (rows == 0) return 0; InsertBattleVideoParty5(record.Header, key, tran); tran.Commit(); return record.SerialNumber; } } } } private void InsertBattleVideoParty5(BattleVideoHeader5 header, ulong key, MySqlTransaction tran) { MySqlCommand cmd = new MySqlCommand("INSERT INTO " + "TerminalBattleVideoPokemon5 (video_id, Slot, Species) VALUES " + "(@key, @slot, @species)", tran.Connection, tran); cmd.Parameters.Add("@key", MySqlDbType.UInt64).Value = key; cmd.Parameters.Add("@slot", MySqlDbType.UByte); cmd.Parameters.Add("@species", MySqlDbType.UInt16); ushort[] party = header.Party; for (byte x = 0; x < 12; x++) { ushort species = party[x]; if (species == 0) continue; cmd.Parameters["@slot"].Value = x; cmd.Parameters["@species"].Value = species; cmd.ExecuteNonQuery(); } } public override BattleVideoHeader5[] BattleVideoSearch5(ushort species, BattleVideoRankings5 ranking, BattleVideoMetagames5 metagame, byte country, byte region, int count) { using (MySqlConnection db = CreateConnection()) { List _params = new List(); String where = ""; String sort = ""; bool hasSearch = false; if (ranking == BattleVideoRankings5.None) { if (species != 0xffff) { where += (hasSearch ? " AND " : " WHERE ") + "EXISTS(SELECT * FROM TerminalBattleVideoPokemon5 " + "WHERE video_id = TerminalBattleVideos5.id AND Species = @species)"; _params.Add(new MySqlParameter("@species", species)); hasSearch = true; } if (metagame == BattleVideoMetagames5.RandomMatchupSingle) { where += (hasSearch ? " AND " : " WHERE ") + "Metagame IN (40, 104)"; hasSearch = true; } else if (metagame == BattleVideoMetagames5.RandomMatchupDouble) { where += (hasSearch ? " AND " : " WHERE ") + "Metagame IN (41, 105)"; hasSearch = true; } else if (metagame == BattleVideoMetagames5.RandomMatchupTriple) { where += (hasSearch ? " AND " : " WHERE ") + "Metagame IN (42, 106)"; hasSearch = true; } else if (metagame == BattleVideoMetagames5.RandomMatchupRotation) { where += (hasSearch ? " AND " : " WHERE ") + "Metagame IN (43, 107)"; hasSearch = true; } else if (metagame == BattleVideoMetagames5.SearchBattleCompetition) { where += (hasSearch ? " AND " : " WHERE ") + "Metagame BETWEEN 56 AND 59"; hasSearch = true; } else if (metagame != BattleVideoMetagames5.SearchNone) { where += (hasSearch ? " AND " : " WHERE ") + "Metagame = @metagame"; _params.Add(new MySqlParameter("@metagame", (byte)metagame)); hasSearch = true; } if (country != 0xff) { where += (hasSearch ? " AND " : " WHERE ") + "Country = @country"; _params.Add(new MySqlParameter("@country", country)); hasSearch = true; } if (region != 0xff) { where += (hasSearch ? " AND " : " WHERE ") + "Region = @region"; _params.Add(new MySqlParameter("@region", region)); } sort = " ORDER BY TimeAdded DESC, id DESC"; } else if (ranking == BattleVideoRankings5.LinkBattles) { // todo: sort by .. something. where = " WHERE NOT (Metagame BETWEEN 0 AND 4)"; sort = " ORDER BY Streak DESC, TimeAdded DESC, id DESC"; } else if (ranking == BattleVideoRankings5.SubwayBattles) { where = " WHERE Metagame BETWEEN 0 AND 4"; sort = " ORDER BY Streak DESC, TimeAdded DESC, id DESC"; } else { sort = " ORDER BY TimeAdded DESC, id DESC"; } _params.Add(new MySqlParameter("@count", count)); db.Open(); List results = new List(count); MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT pid, " + "SerialNumber, Header FROM TerminalBattleVideos5" + where + sort + " LIMIT @count", _params.ToArray()); while (reader.Read()) { results.Add(BattleVideoHeader5FromReader(reader)); } reader.Close(); db.Close(); return results.ToArray(); } } private BattleVideoHeader5 BattleVideoHeader5FromReader(MySqlDataReader reader) { byte[] data = new byte[196]; reader.GetBytes(2, 0, data, 0, 196); return new BattleVideoHeader5(reader.GetInt32(0), reader.GetUInt64(1), data); } public override BattleVideoRecord5 BattleVideoGet5(ulong serial, bool incrementViews = false) { String update = incrementViews ? "UPDATE TerminalBattleVideos5 " + "SET Views = Views + 1 WHERE SerialNumber = @serial; " : ""; using (MySqlConnection db = CreateConnection()) { db.Open(); MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader(update + "SELECT pid, " + "SerialNumber, Header, Data FROM TerminalBattleVideos5 " + "WHERE SerialNumber = @serial", new MySqlParameter("@serial", serial)); if (reader.Read()) return BattleVideo5FromReader(reader); else return null; } } private BattleVideoRecord5 BattleVideo5FromReader(MySqlDataReader reader) { byte[] data = new byte[6112]; reader.GetBytes(3, 0, data, 0, 6112); BattleVideoHeader5 header = BattleVideoHeader5FromReader(reader); return new BattleVideoRecord5(header.PID, header.SerialNumber, header, data); } public override bool BattleVideoFlagSaved5(ulong serial) { using (MySqlConnection db = CreateConnection()) { db.Open(); int results = db.ExecuteNonQuery("UPDATE TerminalBattleVideos5 " + "SET Saves = Saves + 1 WHERE SerialNumber = @serial", new MySqlParameter("@serial", serial)); db.Close(); return results > 0; } } public override ulong BattleVideoCount5() { using (MySqlConnection db = CreateConnection()) { db.Open(); return Convert.ToUInt64(db.ExecuteScalar("SELECT Count(*) FROM TerminalBattleVideos5")); } } #endregion #region Pokedex creation private const String INSERT_COLUMNS = "Name_JA, Name_EN, Name_FR, Name_IT, Name_DE, Name_ES, Name_KO"; private const String INSERT_VALUES = "@name_ja, @name_en, @name_fr, @name_it, @name_de, @name_es, @name_ko"; private void CreateLocalizedStringQueryPieces(LocalizedString s, List insertParams) { String[] langs = new String[] { "JA", "EN", "FR", "IT", "DE", "ES", "KO" }; foreach (String lang in langs) { MySqlParameter param = new MySqlParameter("@name_" + lang.ToLowerInvariant(), s.ContainsKey(lang) ? s[lang] : (object)DBNull.Value); insertParams.Add(param); } } public override void PokedexInsertSpecies(Species s) { using (MySqlConnection db = CreateConnection()) { db.Open(); List insertParams = new List(); insertParams.Add(new MySqlParameter("@national_dex", s.NationalDex)); insertParams.Add(new MySqlParameter("@family_id", s.FamilyID)); insertParams.Add(new MySqlParameter("@growth_rate", (int)s.GrowthRate)); insertParams.Add(new MySqlParameter("@gender_ratio", s.GenderRatio)); insertParams.Add(new MySqlParameter("@egg_group_1", (byte)s.EggGroup1)); insertParams.Add(new MySqlParameter("@egg_group_2", (byte)s.EggGroup2)); insertParams.Add(new MySqlParameter("@egg_steps", s.EggSteps)); insertParams.Add(new MySqlParameter("@gender_variations", s.GenderVariations)); CreateLocalizedStringQueryPieces(s.Name, insertParams); db.ExecuteNonQuery("INSERT INTO pkmncf_pokedex_pokemon (NationalDex, family_id, " + INSERT_COLUMNS + ", GrowthRate, GenderRatio, EggGroup1, EggGroup2, EggSteps, " + "GenderVariations) VALUES (@national_dex, @family_id, " + INSERT_VALUES + ", @growth_rate, @gender_ratio, @egg_group_1, @egg_group_2, " + "@egg_steps, @gender_variations)", insertParams.ToArray()); db.Close(); } } public override void PokedexInsertForm(Form f) { using (MySqlConnection db = CreateConnection()) { db.Open(); List insertParams = new List(); insertParams.Add(new MySqlParameter("@id", f.ID)); insertParams.Add(new MySqlParameter("@national_dex", f.SpeciesID)); insertParams.Add(new MySqlParameter("@form_value", f.Value)); insertParams.Add(new MySqlParameter("@form_suffix", f.Suffix)); insertParams.Add(new MySqlParameter("@height", f.Height)); insertParams.Add(new MySqlParameter("@weight", f.Weight)); insertParams.Add(new MySqlParameter("@experience", f.Experience)); CreateLocalizedStringQueryPieces(f.Name, insertParams); db.ExecuteNonQuery("INSERT INTO pkmncf_pokedex_pokemon_forms (id, NationalDex, " + "FormValue, " + INSERT_COLUMNS + ", FormSuffix, Height, Weight, Experience) VALUES (" + "@id, @national_dex, @form_value, " + INSERT_VALUES + ", @form_suffix, @height, @weight, @experience)", insertParams.ToArray()); db.Close(); } } public override void PokedexInsertFormStats(FormStats f) { using (MySqlConnection db = CreateConnection()) { db.Open(); db.ExecuteNonQuery("INSERT INTO pkmncf_pokedex_pokemon_form_stats " + "(form_id, MinGeneration, Type1, Type2, " + "BaseHP, BaseAttack, BaseDefense, BaseSpeed, BaseSpAttack, BaseSpDefense, " + "RewardHP, RewardAttack, RewardDefense, RewardSpeed, RewardSpAttack, RewardSpDefense) " + "VALUES (@form_id, @min_generation, @type1, @type2, " + "@base_hp, @base_attack, @base_defense, @base_speed, @base_sp_attack, @base_sp_defense, " + "@reward_hp, @reward_attack, @reward_defense, @reward_speed, @reward_sp_attack, @reward_sp_defense)", new MySqlParameter("@form_id", f.FormID), new MySqlParameter("@min_generation", (int)f.MinGeneration), new MySqlParameter("@type1", f.Type1ID), new MySqlParameter("@type2", f.Type2ID), new MySqlParameter("@base_hp", f.BaseStats.Hp), new MySqlParameter("@base_attack", f.BaseStats.Attack), new MySqlParameter("@base_defense", f.BaseStats.Defense), new MySqlParameter("@base_speed", f.BaseStats.Speed), new MySqlParameter("@base_sp_attack", f.BaseStats.SpecialAttack), new MySqlParameter("@base_sp_defense", f.BaseStats.SpecialDefense), new MySqlParameter("@reward_hp", (byte)f.RewardEvs.Hp), new MySqlParameter("@reward_attack", (byte)f.RewardEvs.Attack), new MySqlParameter("@reward_defense", (byte)f.RewardEvs.Defense), new MySqlParameter("@reward_speed", (byte)f.RewardEvs.Speed), new MySqlParameter("@reward_sp_attack", (byte)f.RewardEvs.SpecialAttack), new MySqlParameter("@reward_sp_defense", (byte)f.RewardEvs.SpecialDefense) ); db.Close(); } } public override void PokedexInsertFamily(Family f) { using (MySqlConnection db = CreateConnection()) { db.Open(); db.ExecuteNonQuery("INSERT INTO pkmncf_pokedex_pokemon_families " + "(id, BasicMale, BasicFemale, BabyMale, BabyFemale, " + "Incense, GenderRatio) VALUES (@id, @basic_male, " + "@basic_female, @baby_male, @baby_female, @incense, " + "@gender_ratio)", // todo: collapse 0 to null sometimes new MySqlParameter("@id", f.ID), new MySqlParameter("@basic_male", f.BasicMaleID), new MySqlParameter("@basic_female", f.BasicFemaleID), new MySqlParameter("@baby_male", f.BabyMaleID), new MySqlParameter("@baby_female", f.BabyFemaleID), new MySqlParameter("@incense", f.IncenseID), new MySqlParameter("@gender_ratio", f.GenderRatio) ); db.Close(); } } public override void PokedexInsertEvolution(Evolution f) { throw new NotImplementedException(); } public override void PokedexInsertType(Pokedex.Type t) { using (MySqlConnection db = CreateConnection()) { db.Open(); List insertParams = new List(); insertParams.Add(new MySqlParameter("@id", t.ID)); insertParams.Add(new MySqlParameter("@damage_class", (byte)t.DamageClass)); CreateLocalizedStringQueryPieces(t.Name, insertParams); db.ExecuteNonQuery("INSERT INTO pkmncf_pokedex_types (id, " + INSERT_COLUMNS + ", DamageClass) VALUES (@id, " + INSERT_VALUES + ", @damage_class)", insertParams.ToArray()); db.Close(); } } public override void PokedexInsertItem(Item i) { using (MySqlConnection db = CreateConnection()) { db.Open(); List insertParams = new List(); insertParams.Add(new MySqlParameter("@id", i.ID)); insertParams.Add(new MySqlParameter("@value3", i.Value3)); insertParams.Add(new MySqlParameter("@value4", i.Value4)); insertParams.Add(new MySqlParameter("@value5", i.Value5)); insertParams.Add(new MySqlParameter("@value6", i.Value6)); insertParams.Add(new MySqlParameter("@price", i.Price)); CreateLocalizedStringQueryPieces(i.Name, insertParams); db.ExecuteNonQuery("INSERT INTO pkmncf_pokedex_items (id, Value3, " + "Value4, Value5, Value6, " + INSERT_COLUMNS + ", Price) VALUES (" + "@id, @value3, @value4, @value5, @value6, " + INSERT_VALUES + ", @price)", insertParams.ToArray()); db.Close(); } } public override void PokedexInsertMove(Move m) { using (MySqlConnection db = CreateConnection()) { db.Open(); List insertParams = new List(); insertParams.Add(new MySqlParameter("@value", m.ID)); insertParams.Add(new MySqlParameter("@type_id", m.TypeID)); insertParams.Add(new MySqlParameter("@damage_class", (int)m.DamageClass)); insertParams.Add(new MySqlParameter("@damage", m.Damage)); insertParams.Add(new MySqlParameter("@pp", m.PP)); insertParams.Add(new MySqlParameter("@accuracy", m.Accuracy)); insertParams.Add(new MySqlParameter("@priority", m.Priority)); insertParams.Add(new MySqlParameter("@target", (int)m.Target)); CreateLocalizedStringQueryPieces(m.Name, insertParams); db.ExecuteNonQuery("INSERT INTO pkmncf_pokedex_moves (Value, type_id, " + "DamageClass, " + INSERT_COLUMNS + ", Damage, PP, Accuracy, " + "Priority, Target) VALUES (@value, @type_id, @damage_class, " + INSERT_VALUES + ", @damage, @pp, " + "@accuracy, @priority, @target)", insertParams.ToArray()); db.Close(); } } public override void PokedexInsertAbility(Ability a) { using (MySqlConnection db = CreateConnection()) { db.Open(); List insertParams = new List(); insertParams.Add(new MySqlParameter("@value", a.Value)); CreateLocalizedStringQueryPieces(a.Name, insertParams); db.ExecuteNonQuery("INSERT INTO pkmncf_pokedex_abilities (Value, " + INSERT_COLUMNS + ") VALUES (@value, " + INSERT_VALUES + ")", insertParams.ToArray()); db.Close(); } } #endregion #region Pokedex retrieval private List ReaderToList(MySqlDataReader reader, Pokedex.Pokedex pokedex, Func ctor) where T : PokedexRecordBase { List result = new List(); while (reader.Read()) { result.Add(ctor()); } return result; } public override List PokedexGetAllSpecies(Pokedex.Pokedex pokedex) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT " + "NationalDex, family_id, " + INSERT_COLUMNS + ", GrowthRate, " + "GenderRatio, EggGroup1, EggGroup2, EggSteps, GenderVariations " + "FROM pkmncf_pokedex_pokemon")) { return ReaderToList(reader, pokedex, () => new Species(pokedex, reader)); } } } public override List
PokedexGetAllForms(Pokedex.Pokedex pokedex) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT " + "id, NationalDex, FormValue, " + INSERT_COLUMNS + ", FormSuffix, " + "Height, Weight, Experience " + "FROM pkmncf_pokedex_pokemon_forms")) { return ReaderToList(reader, pokedex, () => new Form(pokedex, reader)); } } } public override List PokedexGetAllFormStats(Pokedex.Pokedex pokedex) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT " + "form_id, MinGeneration, Type1, Type2, BaseHP, BaseAttack, " + "BaseDefense, BaseSpeed, BaseSpAttack, BaseSpDefense, RewardHP, " + "RewardAttack, RewardDefense, RewardSpeed, RewardSpAttack, RewardSpDefense " + "FROM pkmncf_pokedex_pokemon_form_Stats")) { return ReaderToList(reader, pokedex, () => new FormStats(pokedex, reader)); } } } public override List PokedexGetAllFamilies(Pokedex.Pokedex pokedex) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT " + "id, BasicMale, BasicFemale, BabyMale, BabyFemale, Incense, GenderRatio " + "FROM pkmncf_pokedex_pokemon_families")) { return ReaderToList(reader, pokedex, () => new Family(pokedex, reader)); } } } public override List PokedexGetAllEvolutions(Pokedex.Pokedex pokedex) { throw new NotImplementedException(); using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT " + "id " + "FROM pkmncf_pokedex_pokemon_evolutions")) { //return ReaderToList(reader, pokedex, () => new Evolution(pokedex, reader)); } } } public override List PokedexGetAllTypes(Pokedex.Pokedex pokedex) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT " + "id, " + INSERT_COLUMNS + ", DamageClass " + "FROM pkmncf_pokedex_types")) { return ReaderToList(reader, pokedex, () => new Pokedex.Type(pokedex, reader)); } } } public override List PokedexGetAllItems(Pokedex.Pokedex pokedex) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT " + "id, Value3, Value4, Value5, Value6, " + INSERT_COLUMNS + ", Price " + "FROM pkmncf_pokedex_items")) { return ReaderToList(reader, pokedex, () => new Item(pokedex, reader)); } } } public override List PokedexGetAllMoves(Pokedex.Pokedex pokedex) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT " + "Value, type_id, DamageClass, " + INSERT_COLUMNS + ", Damage, " + "PP, Accuracy, Priority, Target " + "FROM pkmncf_pokedex_moves")) { return ReaderToList(reader, pokedex, () => new Move(pokedex, reader)); } } } public override List PokedexGetAllAbilities(Pokedex.Pokedex pokedex) { using (MySqlConnection db = CreateConnection()) { db.Open(); using (MySqlDataReader reader = (MySqlDataReader)db.ExecuteReader("SELECT " + "Value, " + INSERT_COLUMNS + " FROM pkmncf_pokedex_abilities")) { return ReaderToList(reader, pokedex, () => new Ability(pokedex, reader)); } } } #endregion } }