Thanks, Ewe Loon for a patch that

provides persistent AvatarAppearance for SQLite.
Fixes Mantis #3296.
0.6.5-rc1
Homer Horwitz 2009-04-25 20:55:55 +00:00
parent ad4738ff33
commit 94ab683fe1
3 changed files with 279 additions and 15 deletions

View File

@ -70,6 +70,7 @@ what it is today.
* Gerhard * Gerhard
* Godfrey * Godfrey
* Grumly57 * Grumly57
* Ewe Loon
* Fly-Man * Fly-Man
* Flyte Xevious * Flyte Xevious
* Intimidated * Intimidated

View File

@ -0,0 +1,37 @@
BEGIN TRANSACTION;
CREATE TABLE IF NOT EXISTS avatarappearance(
Owner varchar(36) NOT NULL primary key,
BodyItem varchar(36) DEFAULT NULL,
BodyAsset varchar(36) DEFAULT NULL,
SkinItem varchar(36) DEFAULT NULL,
SkinAsset varchar(36) DEFAULT NULL,
HairItem varchar(36) DEFAULT NULL,
HairAsset varchar(36) DEFAULT NULL,
EyesItem varchar(36) DEFAULT NULL,
EyesAsset varchar(36) DEFAULT NULL,
ShirtItem varchar(36) DEFAULT NULL,
ShirtAsset varchar(36) DEFAULT NULL,
PantsItem varchar(36) DEFAULT NULL,
PantsAsset varchar(36) DEFAULT NULL,
ShoesItem varchar(36) DEFAULT NULL,
ShoesAsset varchar(36) DEFAULT NULL,
SocksItem varchar(36) DEFAULT NULL,
SocksAsset varchar(36) DEFAULT NULL,
JacketItem varchar(36) DEFAULT NULL,
JacketAsset varchar(36) DEFAULT NULL,
GlovesItem varchar(36) DEFAULT NULL,
GlovesAsset varchar(36) DEFAULT NULL,
UnderShirtItem varchar(36) DEFAULT NULL,
UnderShirtAsset varchar(36) DEFAULT NULL,
UnderPantsItem varchar(36) DEFAULT NULL,
UnderPantsAsset varchar(36) DEFAULT NULL,
SkirtItem varchar(36) DEFAULT NULL,
SkirtAsset varchar(36) DEFAULT NULL,
Texture blob,
VisualParams blob,
Serial int DEFAULT NULL,
AvatarHeight float DEFAULT NULL
);
COMMIT;

View File

@ -56,6 +56,7 @@ namespace OpenSim.Data.SQLite
private const string userSelect = "select * from users"; private const string userSelect = "select * from users";
private const string userFriendsSelect = "select a.ownerID as ownerID,a.friendID as friendID,a.friendPerms as friendPerms,b.friendPerms as ownerperms, b.ownerID as fownerID, b.friendID as ffriendID from userfriends as a, userfriends as b"; private const string userFriendsSelect = "select a.ownerID as ownerID,a.friendID as friendID,a.friendPerms as friendPerms,b.friendPerms as ownerperms, b.ownerID as fownerID, b.friendID as ffriendID from userfriends as a, userfriends as b";
private const string userAgentSelect = "select * from useragents"; private const string userAgentSelect = "select * from useragents";
private const string AvatarAppearanceSelect = "select * from avatarappearance";
private const string AvatarPickerAndSQL = "select * from users where username like :username and surname like :surname"; private const string AvatarPickerAndSQL = "select * from users where username like :username and surname like :surname";
private const string AvatarPickerOrSQL = "select * from users where username like :username or surname like :surname"; private const string AvatarPickerOrSQL = "select * from users where username like :username or surname like :surname";
@ -65,6 +66,7 @@ namespace OpenSim.Data.SQLite
private SqliteDataAdapter da; private SqliteDataAdapter da;
private SqliteDataAdapter daf; private SqliteDataAdapter daf;
private SqliteDataAdapter dua; private SqliteDataAdapter dua;
private SqliteDataAdapter daa;
SqliteConnection g_conn; SqliteConnection g_conn;
public override void Initialise() public override void Initialise()
@ -102,12 +104,15 @@ namespace OpenSim.Data.SQLite
da = new SqliteDataAdapter(new SqliteCommand(userSelect, conn)); da = new SqliteDataAdapter(new SqliteCommand(userSelect, conn));
dua = new SqliteDataAdapter(new SqliteCommand(userAgentSelect, conn)); dua = new SqliteDataAdapter(new SqliteCommand(userAgentSelect, conn));
daf = new SqliteDataAdapter(new SqliteCommand(userFriendsSelect, conn)); daf = new SqliteDataAdapter(new SqliteCommand(userFriendsSelect, conn));
daa = new SqliteDataAdapter(new SqliteCommand(AvatarAppearanceSelect, conn));
//if (daa == null) m_log.Info("[SQLiteUserData]: daa = null");
lock (ds) lock (ds)
{ {
ds.Tables.Add(createUsersTable()); ds.Tables.Add(createUsersTable());
ds.Tables.Add(createUserAgentsTable()); ds.Tables.Add(createUserAgentsTable());
ds.Tables.Add(createUserFriendsTable()); ds.Tables.Add(createUserFriendsTable());
ds.Tables.Add(createAvatarAppearanceTable());
setupUserCommands(da, conn); setupUserCommands(da, conn);
da.Fill(ds.Tables["users"]); da.Fill(ds.Tables["users"]);
@ -117,6 +122,9 @@ namespace OpenSim.Data.SQLite
setupUserFriendsCommands(daf, conn); setupUserFriendsCommands(daf, conn);
daf.Fill(ds.Tables["userfriends"]); daf.Fill(ds.Tables["userfriends"]);
setupAvatarAppearanceCommands(daa, conn);
daa.Fill(ds.Tables["avatarappearance"]);
} }
return; return;
@ -149,6 +157,11 @@ namespace OpenSim.Data.SQLite
dua.Dispose(); dua.Dispose();
dua = null; dua = null;
} }
if (daa != null)
{
daa.Dispose();
daa = null;
}
aplist = null; aplist = null;
} }
@ -641,13 +654,83 @@ namespace OpenSim.Data.SQLite
/// <returns>Avatar Appearence</returns> /// <returns>Avatar Appearence</returns>
override public AvatarAppearance GetUserAppearance(UUID user) override public AvatarAppearance GetUserAppearance(UUID user)
{ {
AvatarAppearance aa = null; m_log.Info("[APPEARANCE] GetUserAppearance " + user.ToString());
try {
aa = aplist[user]; AvatarAppearance aa = new AvatarAppearance(user);
m_log.Info("[APPEARANCE] Found appearance for " + user.ToString() + aa.ToString()); //try {
} catch (KeyNotFoundException) { aa.Owner = user;
m_log.InfoFormat("[APPEARANCE] No appearance found for {0}", user.ToString()); //aplist[user] = appearance;
DataTable aap = ds.Tables["avatarappearance"];
lock (ds)
{
DataRow row = aap.Rows.Find(Util.ToRawUuidString(user));
if (row == null)
{
m_log.Info("[APPEARANCE] Could not find appearance for " + user.ToString());
//m_log.Debug("[USER DB]: Creating avatarappearance For: " + user.ToString());
//row = aap.NewRow();
//fillAvatarAppearanceRow(row, user, appearance);
//aap.Rows.Add(row);
// m_log.Debug("[USER DB]: Syncing user database: " + ds.Tables["users"].Rows.Count + " users stored");
// save changes off to disk
//daa.Update(ds, "avatarappearance");
} }
else
{
m_log.InfoFormat("[APPEARANCE] appearance found for {0}", user.ToString());
aa.BodyAsset = new UUID((String)row["BodyAsset"]);
aa.BodyItem = new UUID((String)row["BodyItem"]);
aa.SkinItem = new UUID((String)row["SkinItem"]);
aa.SkinAsset = new UUID((String)row["SkinAsset"]);
aa.HairItem = new UUID((String)row["HairItem"]);
aa.HairAsset = new UUID((String)row["HairAsset"]);
aa.EyesItem = new UUID((String)row["EyesItem"]);
aa.EyesAsset = new UUID((String)row["EyesAsset"]);
aa.ShirtItem = new UUID((String)row["ShirtItem"]);
aa.ShirtAsset = new UUID((String)row["ShirtAsset"]);
aa.PantsItem = new UUID((String)row["PantsItem"]);
aa.PantsAsset = new UUID((String)row["PantsAsset"]);
aa.ShoesItem = new UUID((String)row["ShoesItem"]);
aa.ShoesAsset = new UUID((String)row["ShoesAsset"]);
aa.SocksItem = new UUID((String)row["SocksItem"]);
aa.SocksAsset = new UUID((String)row["SocksAsset"]);
aa.JacketItem = new UUID((String)row["JacketItem"]);
aa.JacketAsset = new UUID((String)row["JacketAsset"]);
aa.GlovesItem = new UUID((String)row["GlovesItem"]);
aa.GlovesAsset = new UUID((String)row["GlovesAsset"]);
aa.UnderShirtItem = new UUID((String)row["UnderShirtItem"]);
aa.UnderShirtAsset = new UUID((String)row["UnderShirtAsset"]);
aa.UnderPantsItem = new UUID((String)row["UnderPantsItem"]);
aa.UnderPantsAsset = new UUID((String)row["UnderPantsAsset"]);
aa.SkirtItem = new UUID((String)row["SkirtItem"]);
aa.SkirtAsset = new UUID((String)row["SkirtAsset"]);
// Ewe Loon
// Used Base64String because for some reason it wont accept using Byte[] (which works in Region date)
String str = (String)row["Texture"];
byte[] texture = Convert.FromBase64String(str);
aa.Texture = new Primitive.TextureEntry(texture, 0, texture.Length);
str = (String)row["VisualParams"];
byte[] VisualParams = Convert.FromBase64String(str);
aa.VisualParams = VisualParams;
aa.Serial = Convert.ToInt32(row["Serial"]);
aa.AvatarHeight = Convert.ToSingle(row["AvatarHeight"]);
m_log.InfoFormat("[APPEARANCE] appearance set for {0}", user.ToString());
}
}
// aa = aplist[user];
// m_log.Info("[APPEARANCE] Found appearance for " + user.ToString() + aa.ToString());
// } catch (KeyNotFoundException) {
// m_log.InfoFormat("[APPEARANCE] No appearance found for {0}", user.ToString());
// }
return aa; return aa;
} }
@ -659,7 +742,29 @@ namespace OpenSim.Data.SQLite
override public void UpdateUserAppearance(UUID user, AvatarAppearance appearance) override public void UpdateUserAppearance(UUID user, AvatarAppearance appearance)
{ {
appearance.Owner = user; appearance.Owner = user;
aplist[user] = appearance; //aplist[user] = appearance;
DataTable aap = ds.Tables["avatarappearance"];
lock (ds)
{
DataRow row = aap.Rows.Find(Util.ToRawUuidString(user));
if (row == null)
{
m_log.Debug("[USER DB]: Creating UserAppearance For: " + user.ToString());
row = aap.NewRow();
fillAvatarAppearanceRow(row, user, appearance);
aap.Rows.Add(row);
// m_log.Debug("[USER DB]: Syncing user database: " + ds.Tables["users"].Rows.Count + " users stored");
// save changes off to disk
daa.Update(ds, "avatarappearance");
}
else
{
m_log.Debug("[USER DB]: Updating UserAppearance For: " + user.ToString());
fillAvatarAppearanceRow(row, user, appearance);
daa.Update(ds, "avatarappearance");
}
}
} }
/// <summary> /// <summary>
@ -784,6 +889,55 @@ namespace OpenSim.Data.SQLite
return ua; return ua;
} }
/// <summary>
/// Create the "avatarappearance" table
/// </summary>
/// <returns>Data Table</returns>
private static DataTable createAvatarAppearanceTable()
{
DataTable aa = new DataTable("avatarappearance");
// table contains user appearance items
SQLiteUtil.createCol(aa, "Owner", typeof(String));
SQLiteUtil.createCol(aa, "BodyItem", typeof(String));
SQLiteUtil.createCol(aa, "BodyAsset", typeof(String));
SQLiteUtil.createCol(aa, "SkinItem", typeof(String));
SQLiteUtil.createCol(aa, "SkinAsset", typeof(String));
SQLiteUtil.createCol(aa, "HairItem", typeof(String));
SQLiteUtil.createCol(aa, "HairAsset", typeof(String));
SQLiteUtil.createCol(aa, "EyesItem", typeof(String));
SQLiteUtil.createCol(aa, "EyesAsset", typeof(String));
SQLiteUtil.createCol(aa, "ShirtItem", typeof(String));
SQLiteUtil.createCol(aa, "ShirtAsset", typeof(String));
SQLiteUtil.createCol(aa, "PantsItem", typeof(String));
SQLiteUtil.createCol(aa, "PantsAsset", typeof(String));
SQLiteUtil.createCol(aa, "ShoesItem", typeof(String));
SQLiteUtil.createCol(aa, "ShoesAsset", typeof(String));
SQLiteUtil.createCol(aa, "SocksItem", typeof(String));
SQLiteUtil.createCol(aa, "SocksAsset", typeof(String));
SQLiteUtil.createCol(aa, "JacketItem", typeof(String));
SQLiteUtil.createCol(aa, "JacketAsset", typeof(String));
SQLiteUtil.createCol(aa, "GlovesItem", typeof(String));
SQLiteUtil.createCol(aa, "GlovesAsset", typeof(String));
SQLiteUtil.createCol(aa, "UnderShirtItem", typeof(String));
SQLiteUtil.createCol(aa, "UnderShirtAsset", typeof(String));
SQLiteUtil.createCol(aa, "UnderPantsItem", typeof(String));
SQLiteUtil.createCol(aa, "UnderPantsAsset", typeof(String));
SQLiteUtil.createCol(aa, "SkirtItem", typeof(String));
SQLiteUtil.createCol(aa, "SkirtAsset", typeof(String));
// Used Base64String because for some reason it wont accept using Byte[] (which works in Region date)
SQLiteUtil.createCol(aa, "Texture", typeof (String));
SQLiteUtil.createCol(aa, "VisualParams", typeof (String));
SQLiteUtil.createCol(aa, "Serial", typeof(Int32));
SQLiteUtil.createCol(aa, "AvatarHeight", typeof(Double));
aa.PrimaryKey = new DataColumn[] { aa.Columns["Owner"] };
return aa;
}
/*********************************************************************** /***********************************************************************
* *
* Convert between ADO.NET <=> OpenSim Objects * Convert between ADO.NET <=> OpenSim Objects
@ -902,6 +1056,58 @@ namespace OpenSim.Data.SQLite
} }
} }
/// <summary>
///
/// </summary>
/// <param name="row"></param>
/// <param name="user"></param>
private void fillAvatarAppearanceRow(DataRow row, UUID user, AvatarAppearance appearance)
{
row["Owner"] = Util.ToRawUuidString(user);
row["BodyItem"] = appearance.BodyItem.ToString();
row["BodyAsset"] = appearance.BodyAsset.ToString();
row["SkinItem"] = appearance.SkinItem.ToString();
row["SkinAsset"] = appearance.SkinAsset.ToString();
row["HairItem"] = appearance.HairItem.ToString();
row["HairAsset"] = appearance.HairAsset.ToString();
row["EyesItem"] = appearance.EyesItem.ToString();
row["EyesAsset"] = appearance.EyesAsset.ToString();
row["ShirtItem"] = appearance.ShirtItem.ToString();
row["ShirtAsset"] = appearance.ShirtAsset.ToString();
row["PantsItem"] = appearance.PantsItem.ToString();
row["PantsAsset"] = appearance.PantsAsset.ToString();
row["ShoesItem"] = appearance.ShoesItem.ToString();
row["ShoesAsset"] = appearance.ShoesAsset.ToString();
row["SocksItem"] = appearance.SocksItem.ToString();
row["SocksAsset"] = appearance.SocksAsset.ToString();
row["JacketItem"] = appearance.JacketItem.ToString();
row["JacketAsset"] = appearance.JacketAsset.ToString();
row["GlovesItem"] = appearance.GlovesItem.ToString();
row["GlovesAsset"] = appearance.GlovesAsset.ToString();
row["UnderShirtItem"] = appearance.UnderShirtItem.ToString();
row["UnderShirtAsset"] = appearance.UnderShirtAsset.ToString();
row["UnderPantsItem"] = appearance.UnderPantsItem.ToString();
row["UnderPantsAsset"] = appearance.UnderPantsAsset.ToString();
row["SkirtItem"] = appearance.SkirtItem.ToString();
row["SkirtAsset"] = appearance.SkirtAsset.ToString();
// Used Base64String because for some reason it wont accept using Byte[] (which works in Region date)
row["Texture"] = Convert.ToBase64String(appearance.Texture.GetBytes());
row["VisualParams"] = Convert.ToBase64String(appearance.VisualParams);
row["Serial"] = appearance.Serial;
row["AvatarHeight"] = appearance.AvatarHeight;
// ADO.NET doesn't handle NULL very well
foreach (DataColumn col in ds.Tables["avatarappearance"].Columns)
{
if (row[col] == null)
{
row[col] = String.Empty;
}
}
}
/// <summary> /// <summary>
/// ///
/// </summary> /// </summary>
@ -1027,6 +1233,26 @@ namespace OpenSim.Data.SQLite
} }
/// <summary>
///
/// </summary>
/// <param name="daf"></param>
/// <param name="conn"></param>
private void setupAvatarAppearanceCommands(SqliteDataAdapter daa, SqliteConnection conn)
{
daa.InsertCommand = SQLiteUtil.createInsertCommand("avatarappearance", ds.Tables["avatarappearance"]);
daa.InsertCommand.Connection = conn;
daa.UpdateCommand = SQLiteUtil.createUpdateCommand("avatarappearance", "Owner=:Owner", ds.Tables["avatarappearance"]);
daa.UpdateCommand.Connection = conn;
SqliteCommand delete = new SqliteCommand("delete from avatarappearance where Owner=:Owner");
delete.Parameters.Add(SQLiteUtil.createSqliteParameter("Owner", typeof(String)));
delete.Connection = conn;
daa.DeleteCommand = delete;
}
override public void ResetAttachments(UUID userID) override public void ResetAttachments(UUID userID)
{ {
} }