mirror of
https://github.com/mm201/pkmn-classic-framework.git
synced 2026-03-22 09:54:09 -05:00
2920 lines
138 KiB
C#
2920 lines
138 KiB
C#
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<MySqlParameter> _params = new List<MySqlParameter>();
|
|
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<GtsRecord4> records;
|
|
if (count > 0) records = new List<GtsRecord4>(count);
|
|
else records = new List<GtsRecord4>();
|
|
|
|
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<MySqlParameter> _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<MySqlParameter> _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<MySqlParameter> _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<MySqlParameter> _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<MySqlParameter> ParamsFromBattleTowerRecord4(BattleTowerRecord4 record, bool leader)
|
|
{
|
|
List<MySqlParameter> result = new List<MySqlParameter>(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<MySqlParameter> ParamsFromBattleTowerPokemon4(BattleTowerPokemon4 pokemon)
|
|
{
|
|
List<MySqlParameter> result = new List<MySqlParameter>(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<MySqlParameter> _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<MySqlParameter> _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;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tries to find an existing database record for the provided player
|
|
/// record. The match must be found in the same rank and room number.
|
|
/// </summary>
|
|
/// <param name="tran"></param>
|
|
/// <param name="record"></param>
|
|
/// <param name="leader">If true, look up against the Leaders table.
|
|
/// Otherwise looks up against the opponents table.</param>
|
|
/// <returns>The match's primary key or 0 if no match is found
|
|
/// </returns>
|
|
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<BattleTowerRecord4> records = new List<BattleTowerRecord4>(7);
|
|
List<ulong> keys = new List<ulong>(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<BattleTowerProfile4> profiles = new List<BattleTowerProfile4>(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<MySqlParameter> _params = new List<MySqlParameter>();
|
|
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<GtsRecord5> records;
|
|
if (count > 0) records = new List<GtsRecord5>(count);
|
|
else records = new List<GtsRecord5>();
|
|
|
|
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<MySqlParameter> _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<MySqlParameter> _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<MySqlParameter> _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<MySqlParameter> _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<MySqlParameter> ParamsFromBattleSubwayRecord5(BattleSubwayRecord5 record, bool leader)
|
|
{
|
|
List<MySqlParameter> result = new List<MySqlParameter>(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<MySqlParameter> ParamsFromBattleSubwayPokemon5(BattleSubwayPokemon5 pokemon)
|
|
{
|
|
List<MySqlParameter> result = new List<MySqlParameter>(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<MySqlParameter> _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<MySqlParameter> _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;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tries to find an existing database record for the provided player
|
|
/// record. The match must be found in the same rank and room number.
|
|
/// </summary>
|
|
/// <param name="tran"></param>
|
|
/// <param name="record"></param>
|
|
/// <param name="leader">If true, look up against the Leaders table.
|
|
/// Otherwise looks up against the opponents table.</param>
|
|
/// <returns>The match's primary key or 0 if no match is found
|
|
/// </returns>
|
|
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<BattleSubwayRecord5> records = new List<BattleSubwayRecord5>(7);
|
|
List<ulong> keys = new List<ulong>(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<BattleSubwayProfile5> profiles = new List<BattleSubwayProfile5>(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<DressupRecord4> results = new List<DressupRecord4>(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<BoxRecord4> results = new List<BoxRecord4>(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<MySqlParameter> _params = new List<MySqlParameter>();
|
|
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<BattleVideoHeader4> results = new List<BattleVideoHeader4>(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<MusicalRecord5> results = new List<MusicalRecord5>(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<MySqlParameter> _params = new List<MySqlParameter>();
|
|
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<BattleVideoHeader5> results = new List<BattleVideoHeader5>(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<MySqlParameter> 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<MySqlParameter> insertParams = new List<MySqlParameter>();
|
|
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<MySqlParameter> insertParams = new List<MySqlParameter>();
|
|
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<MySqlParameter> insertParams = new List<MySqlParameter>();
|
|
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<MySqlParameter> insertParams = new List<MySqlParameter>();
|
|
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<MySqlParameter> insertParams = new List<MySqlParameter>();
|
|
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<MySqlParameter> insertParams = new List<MySqlParameter>();
|
|
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<T> ReaderToList<T>(MySqlDataReader reader, Pokedex.Pokedex pokedex, Func<T> ctor)
|
|
where T : PokedexRecordBase
|
|
{
|
|
List<T> result = new List<T>();
|
|
while (reader.Read())
|
|
{
|
|
result.Add(ctor());
|
|
}
|
|
return result;
|
|
}
|
|
|
|
public override List<Species> 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<Form> 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<FormStats> 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<Family> 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<Evolution> 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<Pokedex.Type> 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<Item> 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<Move> 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<Ability> 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
|
|
|
|
}
|
|
}
|