Merge branch 'avination-current' of ssh://3dhosting.de/var/git/careminster into avination-current
Conflicts: bin/Regions/Regions.ini.exampleavinationmerge
commit
4275d7a839
|
@ -98,3 +98,5 @@ OpenSim/Region/ScriptEngine/test-results/
|
||||||
OpenSim/Tests/Common/test-results/
|
OpenSim/Tests/Common/test-results/
|
||||||
OpenSim/Tests/test-results/
|
OpenSim/Tests/test-results/
|
||||||
test-results/
|
test-results/
|
||||||
|
doc/html
|
||||||
|
doc/doxygen.error.log
|
||||||
|
|
|
@ -43,10 +43,11 @@
|
||||||
<delete dir="${distbindir}/Prebuild"/>
|
<delete dir="${distbindir}/Prebuild"/>
|
||||||
<delete dir="${distbindir}/%temp%"/>
|
<delete dir="${distbindir}/%temp%"/>
|
||||||
<delete dir="${distbindir}/.nant"/>
|
<delete dir="${distbindir}/.nant"/>
|
||||||
|
<delete dir="${distbindir}/ThirdParty"/>
|
||||||
<delete>
|
<delete>
|
||||||
<fileset basedir="${distbindir}">
|
<fileset basedir="${distbindir}">
|
||||||
<include name="compile.bat"/>
|
<include name="compile.bat"/>
|
||||||
<include name="BUILDING.txt"/>
|
<include name="BUILDING.md"/>
|
||||||
<include name="Makefile"/>
|
<include name="Makefile"/>
|
||||||
<include name="nant-color"/>
|
<include name="nant-color"/>
|
||||||
<include name="OpenSim.*"/>
|
<include name="OpenSim.*"/>
|
||||||
|
|
|
@ -65,6 +65,7 @@ what it is today.
|
||||||
* A_Biondi
|
* A_Biondi
|
||||||
* alex_carnell
|
* alex_carnell
|
||||||
* Alan Webb (IBM)
|
* Alan Webb (IBM)
|
||||||
|
* Allen Kerensky
|
||||||
* BigFootAg
|
* BigFootAg
|
||||||
* BlueWall Slade
|
* BlueWall Slade
|
||||||
* brianw/Sir_Ahzz
|
* brianw/Sir_Ahzz
|
||||||
|
@ -154,6 +155,7 @@ what it is today.
|
||||||
* tglion
|
* tglion
|
||||||
* tlaukkan/Tommil (Tommi S. E. Laukkanen, Bubble Cloud)
|
* tlaukkan/Tommil (Tommi S. E. Laukkanen, Bubble Cloud)
|
||||||
* tyre
|
* tyre
|
||||||
|
* Vegaslon <vegaslon@gmail.com>
|
||||||
* VikingErik
|
* VikingErik
|
||||||
* Vytek
|
* Vytek
|
||||||
* webmage (IBM)
|
* webmage (IBM)
|
||||||
|
|
|
@ -0,0 +1,77 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Region.Framework.Interfaces;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public class ForeignImporter
|
||||||
|
{
|
||||||
|
IUserManagement m_UserManagement;
|
||||||
|
public ForeignImporter(IUserManagement uman)
|
||||||
|
{
|
||||||
|
m_UserManagement = uman;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupMembersData ConvertGroupMembersData(ExtendedGroupMembersData _m)
|
||||||
|
{
|
||||||
|
GroupMembersData m = new GroupMembersData();
|
||||||
|
m.AcceptNotices = _m.AcceptNotices;
|
||||||
|
m.AgentPowers = _m.AgentPowers;
|
||||||
|
m.Contribution = _m.Contribution;
|
||||||
|
m.IsOwner = _m.IsOwner;
|
||||||
|
m.ListInProfile = _m.ListInProfile;
|
||||||
|
m.OnlineStatus = _m.OnlineStatus;
|
||||||
|
m.Title = _m.Title;
|
||||||
|
|
||||||
|
string url = string.Empty, first = string.Empty, last = string.Empty, tmp = string.Empty;
|
||||||
|
Util.ParseUniversalUserIdentifier(_m.AgentID, out m.AgentID, out url, out first, out last, out tmp);
|
||||||
|
if (url != string.Empty)
|
||||||
|
m_UserManagement.AddUser(m.AgentID, first, last, url);
|
||||||
|
|
||||||
|
return m;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupRoleMembersData ConvertGroupRoleMembersData(ExtendedGroupRoleMembersData _rm)
|
||||||
|
{
|
||||||
|
GroupRoleMembersData rm = new GroupRoleMembersData();
|
||||||
|
rm.RoleID = _rm.RoleID;
|
||||||
|
|
||||||
|
string url = string.Empty, first = string.Empty, last = string.Empty, tmp = string.Empty;
|
||||||
|
Util.ParseUniversalUserIdentifier(_rm.MemberID, out rm.MemberID, out url, out first, out last, out tmp);
|
||||||
|
if (url != string.Empty)
|
||||||
|
m_UserManagement.AddUser(rm.MemberID, first, last, url);
|
||||||
|
|
||||||
|
return rm;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,509 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenMetaverse;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public class ExtendedGroupRecord : GroupRecord
|
||||||
|
{
|
||||||
|
public int MemberCount;
|
||||||
|
public int RoleCount;
|
||||||
|
public string ServiceLocation;
|
||||||
|
public string FounderUUI;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class ExtendedGroupMembershipData : GroupMembershipData
|
||||||
|
{
|
||||||
|
public string AccessToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class ExtendedGroupMembersData
|
||||||
|
{
|
||||||
|
// This is the only difference: this is a string
|
||||||
|
public string AgentID;
|
||||||
|
public int Contribution;
|
||||||
|
public string OnlineStatus;
|
||||||
|
public ulong AgentPowers;
|
||||||
|
public string Title;
|
||||||
|
public bool IsOwner;
|
||||||
|
public bool ListInProfile;
|
||||||
|
public bool AcceptNotices;
|
||||||
|
public string AccessToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class ExtendedGroupRoleMembersData
|
||||||
|
{
|
||||||
|
public UUID RoleID;
|
||||||
|
// This is the only difference: this is a string
|
||||||
|
public string MemberID;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct ExtendedGroupNoticeData
|
||||||
|
{
|
||||||
|
public UUID NoticeID;
|
||||||
|
public uint Timestamp;
|
||||||
|
public string FromName;
|
||||||
|
public string Subject;
|
||||||
|
public bool HasAttachment;
|
||||||
|
public byte AttachmentType;
|
||||||
|
public string AttachmentName;
|
||||||
|
public UUID AttachmentItemID;
|
||||||
|
public string AttachmentOwnerID;
|
||||||
|
|
||||||
|
public GroupNoticeData ToGroupNoticeData()
|
||||||
|
{
|
||||||
|
GroupNoticeData n = new GroupNoticeData();
|
||||||
|
n.FromName = this.FromName;
|
||||||
|
n.AssetType = this.AttachmentType;
|
||||||
|
n.HasAttachment = this.HasAttachment;
|
||||||
|
n.NoticeID = this.NoticeID;
|
||||||
|
n.Subject = this.Subject;
|
||||||
|
n.Timestamp = this.Timestamp;
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class GroupsDataUtils
|
||||||
|
{
|
||||||
|
public static string Sanitize(string s)
|
||||||
|
{
|
||||||
|
return s == null ? string.Empty : s;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Dictionary<string, object> GroupRecord(ExtendedGroupRecord grec)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
if (grec == null)
|
||||||
|
return dict;
|
||||||
|
|
||||||
|
dict["AllowPublish"] = grec.AllowPublish.ToString();
|
||||||
|
dict["Charter"] = Sanitize(grec.Charter);
|
||||||
|
dict["FounderID"] = grec.FounderID.ToString();
|
||||||
|
dict["FounderUUI"] = Sanitize(grec.FounderUUI);
|
||||||
|
dict["GroupID"] = grec.GroupID.ToString();
|
||||||
|
dict["GroupName"] = Sanitize(grec.GroupName);
|
||||||
|
dict["InsigniaID"] = grec.GroupPicture.ToString();
|
||||||
|
dict["MaturePublish"] = grec.MaturePublish.ToString();
|
||||||
|
dict["MembershipFee"] = grec.MembershipFee.ToString();
|
||||||
|
dict["OpenEnrollment"] = grec.OpenEnrollment.ToString();
|
||||||
|
dict["OwnerRoleID"] = grec.OwnerRoleID.ToString();
|
||||||
|
dict["ServiceLocation"] = Sanitize(grec.ServiceLocation);
|
||||||
|
dict["ShownInList"] = grec.ShowInList.ToString();
|
||||||
|
dict["MemberCount"] = grec.MemberCount.ToString();
|
||||||
|
dict["RoleCount"] = grec.RoleCount.ToString();
|
||||||
|
|
||||||
|
return dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ExtendedGroupRecord GroupRecord(Dictionary<string, object> dict)
|
||||||
|
{
|
||||||
|
if (dict == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
ExtendedGroupRecord grec = new ExtendedGroupRecord();
|
||||||
|
if (dict.ContainsKey("AllowPublish") && dict["AllowPublish"] != null)
|
||||||
|
grec.AllowPublish = bool.Parse(dict["AllowPublish"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Charter") && dict["Charter"] != null)
|
||||||
|
grec.Charter = dict["Charter"].ToString();
|
||||||
|
else
|
||||||
|
grec.Charter = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("FounderID") && dict["FounderID"] != null)
|
||||||
|
grec.FounderID = UUID.Parse(dict["FounderID"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("FounderUUI") && dict["FounderUUI"] != null)
|
||||||
|
grec.FounderUUI = dict["FounderUUI"].ToString();
|
||||||
|
else
|
||||||
|
grec.FounderUUI = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("GroupID") && dict["GroupID"] != null)
|
||||||
|
grec.GroupID = UUID.Parse(dict["GroupID"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("GroupName") && dict["GroupName"] != null)
|
||||||
|
grec.GroupName = dict["GroupName"].ToString();
|
||||||
|
else
|
||||||
|
grec.GroupName = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("InsigniaID") && dict["InsigniaID"] != null)
|
||||||
|
grec.GroupPicture = UUID.Parse(dict["InsigniaID"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("MaturePublish") && dict["MaturePublish"] != null)
|
||||||
|
grec.MaturePublish = bool.Parse(dict["MaturePublish"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("MembershipFee") && dict["MembershipFee"] != null)
|
||||||
|
grec.MembershipFee = Int32.Parse(dict["MembershipFee"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("OpenEnrollment") && dict["OpenEnrollment"] != null)
|
||||||
|
grec.OpenEnrollment = bool.Parse(dict["OpenEnrollment"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("OwnerRoleID") && dict["OwnerRoleID"] != null)
|
||||||
|
grec.OwnerRoleID = UUID.Parse(dict["OwnerRoleID"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("ServiceLocation") && dict["ServiceLocation"] != null)
|
||||||
|
grec.ServiceLocation = dict["ServiceLocation"].ToString();
|
||||||
|
else
|
||||||
|
grec.GroupName = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("ShownInList") && dict["ShownInList"] != null)
|
||||||
|
grec.ShowInList = bool.Parse(dict["ShownInList"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("MemberCount") && dict["MemberCount"] != null)
|
||||||
|
grec.MemberCount = Int32.Parse(dict["MemberCount"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("RoleCount") && dict["RoleCount"] != null)
|
||||||
|
grec.RoleCount = Int32.Parse(dict["RoleCount"].ToString());
|
||||||
|
|
||||||
|
return grec;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Dictionary<string, object> GroupMembershipData(ExtendedGroupMembershipData membership)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
if (membership == null)
|
||||||
|
return dict;
|
||||||
|
|
||||||
|
dict["AcceptNotices"] = membership.AcceptNotices.ToString();
|
||||||
|
dict["AccessToken"] = Sanitize(membership.AccessToken);
|
||||||
|
dict["Active"] = membership.Active.ToString();
|
||||||
|
dict["ActiveRole"] = membership.ActiveRole.ToString();
|
||||||
|
dict["AllowPublish"] = membership.AllowPublish.ToString();
|
||||||
|
dict["Charter"] = Sanitize(membership.Charter);
|
||||||
|
dict["Contribution"] = membership.Contribution.ToString();
|
||||||
|
dict["FounderID"] = membership.FounderID.ToString();
|
||||||
|
dict["GroupID"] = membership.GroupID.ToString();
|
||||||
|
dict["GroupName"] = Sanitize(membership.GroupName);
|
||||||
|
dict["GroupPicture"] = membership.GroupPicture.ToString();
|
||||||
|
dict["GroupPowers"] = membership.GroupPowers.ToString();
|
||||||
|
dict["GroupTitle"] = Sanitize(membership.GroupTitle);
|
||||||
|
dict["ListInProfile"] = membership.ListInProfile.ToString();
|
||||||
|
dict["MaturePublish"] = membership.MaturePublish.ToString();
|
||||||
|
dict["MembershipFee"] = membership.MembershipFee.ToString();
|
||||||
|
dict["OpenEnrollment"] = membership.OpenEnrollment.ToString();
|
||||||
|
dict["ShowInList"] = membership.ShowInList.ToString();
|
||||||
|
|
||||||
|
return dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ExtendedGroupMembershipData GroupMembershipData(Dictionary<string, object> dict)
|
||||||
|
{
|
||||||
|
if (dict == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
ExtendedGroupMembershipData membership = new ExtendedGroupMembershipData();
|
||||||
|
|
||||||
|
if (dict.ContainsKey("AcceptNotices") && dict["AcceptNotices"] != null)
|
||||||
|
membership.AcceptNotices = bool.Parse(dict["AcceptNotices"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("AccessToken") && dict["AccessToken"] != null)
|
||||||
|
membership.AccessToken = dict["AccessToken"].ToString();
|
||||||
|
else
|
||||||
|
membership.AccessToken = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Active") && dict["Active"] != null)
|
||||||
|
membership.Active = bool.Parse(dict["Active"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("ActiveRole") && dict["ActiveRole"] != null)
|
||||||
|
membership.ActiveRole = UUID.Parse(dict["ActiveRole"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("AllowPublish") && dict["AllowPublish"] != null)
|
||||||
|
membership.AllowPublish = bool.Parse(dict["AllowPublish"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Charter") && dict["Charter"] != null)
|
||||||
|
membership.Charter = dict["Charter"].ToString();
|
||||||
|
else
|
||||||
|
membership.Charter = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Contribution") && dict["Contribution"] != null)
|
||||||
|
membership.Contribution = Int32.Parse(dict["Contribution"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("FounderID") && dict["FounderID"] != null)
|
||||||
|
membership.FounderID = UUID.Parse(dict["FounderID"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("GroupID") && dict["GroupID"] != null)
|
||||||
|
membership.GroupID = UUID.Parse(dict["GroupID"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("GroupName") && dict["GroupName"] != null)
|
||||||
|
membership.GroupName = dict["GroupName"].ToString();
|
||||||
|
else
|
||||||
|
membership.GroupName = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("GroupPicture") && dict["GroupPicture"] != null)
|
||||||
|
membership.GroupPicture = UUID.Parse(dict["GroupPicture"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("GroupPowers") && dict["GroupPowers"] != null)
|
||||||
|
membership.GroupPowers = UInt64.Parse(dict["GroupPowers"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("GroupTitle") && dict["GroupTitle"] != null)
|
||||||
|
membership.GroupTitle = dict["GroupTitle"].ToString();
|
||||||
|
else
|
||||||
|
membership.GroupTitle = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("ListInProfile") && dict["ListInProfile"] != null)
|
||||||
|
membership.ListInProfile = bool.Parse(dict["ListInProfile"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("MaturePublish") && dict["MaturePublish"] != null)
|
||||||
|
membership.MaturePublish = bool.Parse(dict["MaturePublish"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("MembershipFee") && dict["MembershipFee"] != null)
|
||||||
|
membership.MembershipFee = Int32.Parse(dict["MembershipFee"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("OpenEnrollment") && dict["OpenEnrollment"] != null)
|
||||||
|
membership.OpenEnrollment = bool.Parse(dict["OpenEnrollment"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("ShowInList") && dict["ShowInList"] != null)
|
||||||
|
membership.ShowInList = bool.Parse(dict["ShowInList"].ToString());
|
||||||
|
|
||||||
|
return membership;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Dictionary<string, object> GroupMembersData(ExtendedGroupMembersData member)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
dict["AcceptNotices"] = member.AcceptNotices.ToString();
|
||||||
|
dict["AccessToken"] = Sanitize(member.AccessToken);
|
||||||
|
dict["AgentID"] = Sanitize(member.AgentID);
|
||||||
|
dict["AgentPowers"] = member.AgentPowers.ToString();
|
||||||
|
dict["Contribution"] = member.Contribution.ToString();
|
||||||
|
dict["IsOwner"] = member.IsOwner.ToString();
|
||||||
|
dict["ListInProfile"] = member.ListInProfile.ToString();
|
||||||
|
dict["OnlineStatus"] = Sanitize(member.OnlineStatus);
|
||||||
|
dict["Title"] = Sanitize(member.Title);
|
||||||
|
|
||||||
|
return dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ExtendedGroupMembersData GroupMembersData(Dictionary<string, object> dict)
|
||||||
|
{
|
||||||
|
ExtendedGroupMembersData member = new ExtendedGroupMembersData();
|
||||||
|
|
||||||
|
if (dict == null)
|
||||||
|
return member;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("AcceptNotices") && dict["AcceptNotices"] != null)
|
||||||
|
member.AcceptNotices = bool.Parse(dict["AcceptNotices"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("AccessToken") && dict["AccessToken"] != null)
|
||||||
|
member.AccessToken = Sanitize(dict["AccessToken"].ToString());
|
||||||
|
else
|
||||||
|
member.AccessToken = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("AgentID") && dict["AgentID"] != null)
|
||||||
|
member.AgentID = Sanitize(dict["AgentID"].ToString());
|
||||||
|
else
|
||||||
|
member.AgentID = UUID.Zero.ToString();
|
||||||
|
|
||||||
|
if (dict.ContainsKey("AgentPowers") && dict["AgentPowers"] != null)
|
||||||
|
member.AgentPowers = UInt64.Parse(dict["AgentPowers"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Contribution") && dict["Contribution"] != null)
|
||||||
|
member.Contribution = Int32.Parse(dict["Contribution"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("IsOwner") && dict["IsOwner"] != null)
|
||||||
|
member.IsOwner = bool.Parse(dict["IsOwner"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("ListInProfile") && dict["ListInProfile"] != null)
|
||||||
|
member.ListInProfile = bool.Parse(dict["ListInProfile"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("OnlineStatus") && dict["OnlineStatus"] != null)
|
||||||
|
member.OnlineStatus = Sanitize(dict["OnlineStatus"].ToString());
|
||||||
|
else
|
||||||
|
member.OnlineStatus = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Title") && dict["Title"] != null)
|
||||||
|
member.Title = Sanitize(dict["Title"].ToString());
|
||||||
|
else
|
||||||
|
member.Title = string.Empty;
|
||||||
|
|
||||||
|
return member;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Dictionary<string, object> GroupRolesData(GroupRolesData role)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
dict["Description"] = Sanitize(role.Description);
|
||||||
|
dict["Members"] = role.Members.ToString();
|
||||||
|
dict["Name"] = Sanitize(role.Name);
|
||||||
|
dict["Powers"] = role.Powers.ToString();
|
||||||
|
dict["RoleID"] = role.RoleID.ToString();
|
||||||
|
dict["Title"] = Sanitize(role.Title);
|
||||||
|
|
||||||
|
return dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static GroupRolesData GroupRolesData(Dictionary<string, object> dict)
|
||||||
|
{
|
||||||
|
GroupRolesData role = new GroupRolesData();
|
||||||
|
|
||||||
|
if (dict == null)
|
||||||
|
return role;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Description") && dict["Description"] != null)
|
||||||
|
role.Description = Sanitize(dict["Description"].ToString());
|
||||||
|
else
|
||||||
|
role.Description = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Members") && dict["Members"] != null)
|
||||||
|
role.Members = Int32.Parse(dict["Members"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Name") && dict["Name"] != null)
|
||||||
|
role.Name = Sanitize(dict["Name"].ToString());
|
||||||
|
else
|
||||||
|
role.Name = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Powers") && dict["Powers"] != null)
|
||||||
|
role.Powers = UInt64.Parse(dict["Powers"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("Title") && dict["Title"] != null)
|
||||||
|
role.Title = Sanitize(dict["Title"].ToString());
|
||||||
|
else
|
||||||
|
role.Title = string.Empty;
|
||||||
|
|
||||||
|
if (dict.ContainsKey("RoleID") && dict["RoleID"] != null)
|
||||||
|
role.RoleID = UUID.Parse(dict["RoleID"].ToString());
|
||||||
|
|
||||||
|
return role;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Dictionary<string, object> GroupRoleMembersData(ExtendedGroupRoleMembersData rmember)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
dict["RoleID"] = rmember.RoleID.ToString();
|
||||||
|
dict["MemberID"] = rmember.MemberID;
|
||||||
|
return dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ExtendedGroupRoleMembersData GroupRoleMembersData(Dictionary<string, object> dict)
|
||||||
|
{
|
||||||
|
ExtendedGroupRoleMembersData rmember = new ExtendedGroupRoleMembersData();
|
||||||
|
|
||||||
|
if (dict.ContainsKey("RoleID") && dict["RoleID"] != null)
|
||||||
|
rmember.RoleID = new UUID(dict["RoleID"].ToString());
|
||||||
|
|
||||||
|
if (dict.ContainsKey("MemberID") && dict["MemberID"] != null)
|
||||||
|
rmember.MemberID = dict["MemberID"].ToString();
|
||||||
|
|
||||||
|
return rmember;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Dictionary<string, object> GroupInviteInfo(GroupInviteInfo invite)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
dict["InviteID"] = invite.InviteID.ToString();
|
||||||
|
dict["GroupID"] = invite.GroupID.ToString();
|
||||||
|
dict["RoleID"] = invite.RoleID.ToString();
|
||||||
|
dict["AgentID"] = invite.AgentID;
|
||||||
|
|
||||||
|
return dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static GroupInviteInfo GroupInviteInfo(Dictionary<string, object> dict)
|
||||||
|
{
|
||||||
|
if (dict == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
GroupInviteInfo invite = new GroupInviteInfo();
|
||||||
|
|
||||||
|
invite.InviteID = new UUID(dict["InviteID"].ToString());
|
||||||
|
invite.GroupID = new UUID(dict["GroupID"].ToString());
|
||||||
|
invite.RoleID = new UUID(dict["RoleID"].ToString());
|
||||||
|
invite.AgentID = Sanitize(dict["AgentID"].ToString());
|
||||||
|
|
||||||
|
return invite;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Dictionary<string, object> GroupNoticeData(ExtendedGroupNoticeData notice)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
dict["NoticeID"] = notice.NoticeID.ToString();
|
||||||
|
dict["Timestamp"] = notice.Timestamp.ToString();
|
||||||
|
dict["FromName"] = Sanitize(notice.FromName);
|
||||||
|
dict["Subject"] = Sanitize(notice.Subject);
|
||||||
|
dict["HasAttachment"] = notice.HasAttachment.ToString();
|
||||||
|
dict["AttachmentItemID"] = notice.AttachmentItemID.ToString();
|
||||||
|
dict["AttachmentName"] = Sanitize(notice.AttachmentName);
|
||||||
|
dict["AttachmentType"] = notice.AttachmentType.ToString();
|
||||||
|
dict["AttachmentOwnerID"] = Sanitize(notice.AttachmentOwnerID);
|
||||||
|
|
||||||
|
return dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ExtendedGroupNoticeData GroupNoticeData(Dictionary<string, object> dict)
|
||||||
|
{
|
||||||
|
ExtendedGroupNoticeData notice = new ExtendedGroupNoticeData();
|
||||||
|
|
||||||
|
if (dict == null)
|
||||||
|
return notice;
|
||||||
|
|
||||||
|
notice.NoticeID = new UUID(dict["NoticeID"].ToString());
|
||||||
|
notice.Timestamp = UInt32.Parse(dict["Timestamp"].ToString());
|
||||||
|
notice.FromName = Sanitize(dict["FromName"].ToString());
|
||||||
|
notice.Subject = Sanitize(dict["Subject"].ToString());
|
||||||
|
notice.HasAttachment = bool.Parse(dict["HasAttachment"].ToString());
|
||||||
|
notice.AttachmentItemID = new UUID(dict["AttachmentItemID"].ToString());
|
||||||
|
notice.AttachmentName = dict["AttachmentName"].ToString();
|
||||||
|
notice.AttachmentType = byte.Parse(dict["AttachmentType"].ToString());
|
||||||
|
notice.AttachmentOwnerID = dict["AttachmentOwnerID"].ToString();
|
||||||
|
|
||||||
|
return notice;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Dictionary<string, object> GroupNoticeInfo(GroupNoticeInfo notice)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = GroupNoticeData(notice.noticeData);
|
||||||
|
|
||||||
|
dict["GroupID"] = notice.GroupID.ToString();
|
||||||
|
dict["Message"] = Sanitize(notice.Message);
|
||||||
|
|
||||||
|
return dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static GroupNoticeInfo GroupNoticeInfo(Dictionary<string, object> dict)
|
||||||
|
{
|
||||||
|
GroupNoticeInfo notice = new GroupNoticeInfo();
|
||||||
|
|
||||||
|
notice.noticeData = GroupNoticeData(dict);
|
||||||
|
notice.GroupID = new UUID(dict["GroupID"].ToString());
|
||||||
|
notice.Message = Sanitize(dict["Message"].ToString());
|
||||||
|
|
||||||
|
return notice;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,594 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using log4net;
|
||||||
|
using Mono.Addins;
|
||||||
|
using Nini.Config;
|
||||||
|
using OpenMetaverse;
|
||||||
|
using OpenMetaverse.StructuredData;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Region.Framework.Interfaces;
|
||||||
|
using OpenSim.Region.Framework.Scenes;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
using PresenceInfo = OpenSim.Services.Interfaces.PresenceInfo;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
[Extension(Path = "/OpenSim/RegionModules", NodeName = "RegionModule", Id = "GroupsMessagingModule")]
|
||||||
|
public class GroupsMessagingModule : ISharedRegionModule, IGroupsMessagingModule
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private List<Scene> m_sceneList = new List<Scene>();
|
||||||
|
private IPresenceService m_presenceService;
|
||||||
|
|
||||||
|
private IMessageTransferModule m_msgTransferModule = null;
|
||||||
|
|
||||||
|
private IGroupsServicesConnector m_groupData = null;
|
||||||
|
|
||||||
|
// Config Options
|
||||||
|
private bool m_groupMessagingEnabled = false;
|
||||||
|
private bool m_debugEnabled = true;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// If enabled, module only tries to send group IMs to online users by querying cached presence information.
|
||||||
|
/// </summary>
|
||||||
|
private bool m_messageOnlineAgentsOnly;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Cache for online users.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Group ID is key, presence information for online members is value.
|
||||||
|
/// Will only be non-null if m_messageOnlineAgentsOnly = true
|
||||||
|
/// We cache here so that group messages don't constantly have to re-request the online user list to avoid
|
||||||
|
/// attempted expensive sending of messages to offline users.
|
||||||
|
/// The tradeoff is that a user that comes online will not receive messages consistently from all other users
|
||||||
|
/// until caches have updated.
|
||||||
|
/// Therefore, we set the cache expiry to just 20 seconds.
|
||||||
|
/// </remarks>
|
||||||
|
private ExpiringCache<UUID, PresenceInfo[]> m_usersOnlineCache;
|
||||||
|
|
||||||
|
private int m_usersOnlineCacheExpirySeconds = 20;
|
||||||
|
|
||||||
|
#region Region Module interfaceBase Members
|
||||||
|
|
||||||
|
public void Initialise(IConfigSource config)
|
||||||
|
{
|
||||||
|
IConfig groupsConfig = config.Configs["Groups"];
|
||||||
|
|
||||||
|
if (groupsConfig == null)
|
||||||
|
// Do not run this module by default.
|
||||||
|
return;
|
||||||
|
|
||||||
|
// if groups aren't enabled, we're not needed.
|
||||||
|
// if we're not specified as the connector to use, then we're not wanted
|
||||||
|
if ((groupsConfig.GetBoolean("Enabled", false) == false)
|
||||||
|
|| (groupsConfig.GetString("MessagingModule", "") != Name))
|
||||||
|
{
|
||||||
|
m_groupMessagingEnabled = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_groupMessagingEnabled = groupsConfig.GetBoolean("MessagingEnabled", true);
|
||||||
|
|
||||||
|
if (!m_groupMessagingEnabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_messageOnlineAgentsOnly = groupsConfig.GetBoolean("MessageOnlineUsersOnly", false);
|
||||||
|
|
||||||
|
if (m_messageOnlineAgentsOnly)
|
||||||
|
m_usersOnlineCache = new ExpiringCache<UUID, PresenceInfo[]>();
|
||||||
|
|
||||||
|
m_debugEnabled = groupsConfig.GetBoolean("DebugEnabled", true);
|
||||||
|
|
||||||
|
m_log.InfoFormat(
|
||||||
|
"[Groups.Messaging]: GroupsMessagingModule enabled with MessageOnlineOnly = {0}, DebugEnabled = {1}",
|
||||||
|
m_messageOnlineAgentsOnly, m_debugEnabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_groupMessagingEnabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
scene.RegisterModuleInterface<IGroupsMessagingModule>(this);
|
||||||
|
m_sceneList.Add(scene);
|
||||||
|
|
||||||
|
scene.EventManager.OnNewClient += OnNewClient;
|
||||||
|
scene.EventManager.OnIncomingInstantMessage += OnGridInstantMessage;
|
||||||
|
scene.EventManager.OnClientLogin += OnClientLogin;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RegionLoaded(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_groupMessagingEnabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: {0} called", System.Reflection.MethodBase.GetCurrentMethod().Name);
|
||||||
|
|
||||||
|
m_groupData = scene.RequestModuleInterface<IGroupsServicesConnector>();
|
||||||
|
|
||||||
|
// No groups module, no groups messaging
|
||||||
|
if (m_groupData == null)
|
||||||
|
{
|
||||||
|
m_log.Error("[Groups.Messaging]: Could not get IGroupsServicesConnector, GroupsMessagingModule is now disabled.");
|
||||||
|
RemoveRegion(scene);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_msgTransferModule = scene.RequestModuleInterface<IMessageTransferModule>();
|
||||||
|
|
||||||
|
// No message transfer module, no groups messaging
|
||||||
|
if (m_msgTransferModule == null)
|
||||||
|
{
|
||||||
|
m_log.Error("[Groups.Messaging]: Could not get MessageTransferModule");
|
||||||
|
RemoveRegion(scene);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_presenceService == null)
|
||||||
|
m_presenceService = scene.PresenceService;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_groupMessagingEnabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: {0} called", System.Reflection.MethodBase.GetCurrentMethod().Name);
|
||||||
|
|
||||||
|
m_sceneList.Remove(scene);
|
||||||
|
scene.EventManager.OnNewClient -= OnNewClient;
|
||||||
|
scene.EventManager.OnIncomingInstantMessage -= OnGridInstantMessage;
|
||||||
|
scene.EventManager.OnClientLogin -= OnClientLogin;
|
||||||
|
scene.UnregisterModuleInterface<IGroupsMessagingModule>(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Close()
|
||||||
|
{
|
||||||
|
if (!m_groupMessagingEnabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (m_debugEnabled) m_log.Debug("[Groups.Messaging]: Shutting down GroupsMessagingModule module.");
|
||||||
|
|
||||||
|
m_sceneList.Clear();
|
||||||
|
|
||||||
|
m_groupData = null;
|
||||||
|
m_msgTransferModule = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type ReplaceableInterface
|
||||||
|
{
|
||||||
|
get { return null; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Name
|
||||||
|
{
|
||||||
|
get { return "Groups Messaging Module V2"; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public void PostInitialise()
|
||||||
|
{
|
||||||
|
// NoOp
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Not really needed, but does confirm that the group exists.
|
||||||
|
/// </summary>
|
||||||
|
public bool StartGroupChatSession(UUID agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled)
|
||||||
|
m_log.DebugFormat("[Groups.Messaging]: {0} called", System.Reflection.MethodBase.GetCurrentMethod().Name);
|
||||||
|
|
||||||
|
GroupRecord groupInfo = m_groupData.GetGroupRecord(agentID.ToString(), groupID, null);
|
||||||
|
|
||||||
|
if (groupInfo != null)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SendMessageToGroup(GridInstantMessage im, UUID groupID)
|
||||||
|
{
|
||||||
|
List<GroupMembersData> groupMembers = m_groupData.GetGroupMembers(new UUID(im.fromAgentID).ToString(), groupID);
|
||||||
|
int groupMembersCount = groupMembers.Count;
|
||||||
|
|
||||||
|
if (m_messageOnlineAgentsOnly)
|
||||||
|
{
|
||||||
|
string[] t1 = groupMembers.ConvertAll<string>(gmd => gmd.AgentID.ToString()).ToArray();
|
||||||
|
|
||||||
|
// We cache in order not to overwhlem the presence service on large grids with many groups. This does
|
||||||
|
// mean that members coming online will not see all group members until after m_usersOnlineCacheExpirySeconds has elapsed.
|
||||||
|
// (assuming this is the same across all grid simulators).
|
||||||
|
PresenceInfo[] onlineAgents;
|
||||||
|
if (!m_usersOnlineCache.TryGetValue(groupID, out onlineAgents))
|
||||||
|
{
|
||||||
|
onlineAgents = m_presenceService.GetAgents(t1);
|
||||||
|
m_usersOnlineCache.Add(groupID, onlineAgents, m_usersOnlineCacheExpirySeconds);
|
||||||
|
}
|
||||||
|
|
||||||
|
HashSet<string> onlineAgentsUuidSet = new HashSet<string>();
|
||||||
|
Array.ForEach<PresenceInfo>(onlineAgents, pi => onlineAgentsUuidSet.Add(pi.UserID));
|
||||||
|
|
||||||
|
groupMembers = groupMembers.Where(gmd => onlineAgentsUuidSet.Contains(gmd.AgentID.ToString())).ToList();
|
||||||
|
|
||||||
|
// if (m_debugEnabled)
|
||||||
|
// m_log.DebugFormat(
|
||||||
|
// "[Groups.Messaging]: SendMessageToGroup called for group {0} with {1} visible members, {2} online",
|
||||||
|
// groupID, groupMembersCount, groupMembers.Count());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (m_debugEnabled)
|
||||||
|
m_log.DebugFormat(
|
||||||
|
"[Groups.Messaging]: SendMessageToGroup called for group {0} with {1} visible members",
|
||||||
|
groupID, groupMembers.Count);
|
||||||
|
}
|
||||||
|
|
||||||
|
int requestStartTick = Environment.TickCount;
|
||||||
|
|
||||||
|
foreach (GroupMembersData member in groupMembers)
|
||||||
|
{
|
||||||
|
if (m_groupData.hasAgentDroppedGroupChatSession(member.AgentID.ToString(), groupID))
|
||||||
|
{
|
||||||
|
// Don't deliver messages to people who have dropped this session
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: {0} has dropped session, not delivering to them", member.AgentID);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy Message
|
||||||
|
GridInstantMessage msg = new GridInstantMessage();
|
||||||
|
msg.imSessionID = groupID.Guid;
|
||||||
|
msg.fromAgentName = im.fromAgentName;
|
||||||
|
msg.message = im.message;
|
||||||
|
msg.dialog = im.dialog;
|
||||||
|
msg.offline = im.offline;
|
||||||
|
msg.ParentEstateID = im.ParentEstateID;
|
||||||
|
msg.Position = im.Position;
|
||||||
|
msg.RegionID = im.RegionID;
|
||||||
|
msg.binaryBucket = im.binaryBucket;
|
||||||
|
msg.timestamp = (uint)Util.UnixTimeSinceEpoch();
|
||||||
|
|
||||||
|
msg.fromAgentID = im.fromAgentID;
|
||||||
|
msg.fromGroup = true;
|
||||||
|
|
||||||
|
msg.toAgentID = member.AgentID.Guid;
|
||||||
|
|
||||||
|
IClientAPI client = GetActiveClient(member.AgentID);
|
||||||
|
if (client == null)
|
||||||
|
{
|
||||||
|
// If they're not local, forward across the grid
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: Delivering to {0} via Grid", member.AgentID);
|
||||||
|
m_msgTransferModule.SendInstantMessage(msg, delegate(bool success) { });
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Deliver locally, directly
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: Passing to ProcessMessageFromGroupSession to deliver to {0} locally", client.Name);
|
||||||
|
ProcessMessageFromGroupSession(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Temporary for assessing how long it still takes to send messages to large online groups.
|
||||||
|
if (m_messageOnlineAgentsOnly)
|
||||||
|
m_log.DebugFormat(
|
||||||
|
"[Groups.Messaging]: SendMessageToGroup for group {0} with {1} visible members, {2} online took {3}ms",
|
||||||
|
groupID, groupMembersCount, groupMembers.Count(), Environment.TickCount - requestStartTick);
|
||||||
|
}
|
||||||
|
|
||||||
|
#region SimGridEventHandlers
|
||||||
|
|
||||||
|
void OnClientLogin(IClientAPI client)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: OnInstantMessage registered for {0}", client.Name);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnNewClient(IClientAPI client)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: OnInstantMessage registered for {0}", client.Name);
|
||||||
|
|
||||||
|
client.OnInstantMessage += OnInstantMessage;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnGridInstantMessage(GridInstantMessage msg)
|
||||||
|
{
|
||||||
|
// The instant message module will only deliver messages of dialog types:
|
||||||
|
// MessageFromAgent, StartTyping, StopTyping, MessageFromObject
|
||||||
|
//
|
||||||
|
// Any other message type will not be delivered to a client by the
|
||||||
|
// Instant Message Module
|
||||||
|
|
||||||
|
|
||||||
|
if (m_debugEnabled)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[Groups.Messaging]: {0} called", System.Reflection.MethodBase.GetCurrentMethod().Name);
|
||||||
|
|
||||||
|
DebugGridInstantMessage(msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Incoming message from a group
|
||||||
|
if ((msg.fromGroup == true) &&
|
||||||
|
((msg.dialog == (byte)InstantMessageDialog.SessionSend)
|
||||||
|
|| (msg.dialog == (byte)InstantMessageDialog.SessionAdd)
|
||||||
|
|| (msg.dialog == (byte)InstantMessageDialog.SessionDrop)))
|
||||||
|
{
|
||||||
|
ProcessMessageFromGroupSession(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ProcessMessageFromGroupSession(GridInstantMessage msg)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: Session message from {0} going to agent {1}", msg.fromAgentName, msg.toAgentID);
|
||||||
|
|
||||||
|
UUID AgentID = new UUID(msg.fromAgentID);
|
||||||
|
UUID GroupID = new UUID(msg.imSessionID);
|
||||||
|
|
||||||
|
switch (msg.dialog)
|
||||||
|
{
|
||||||
|
case (byte)InstantMessageDialog.SessionAdd:
|
||||||
|
m_groupData.AgentInvitedToGroupChatSession(AgentID.ToString(), GroupID);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case (byte)InstantMessageDialog.SessionDrop:
|
||||||
|
m_groupData.AgentDroppedFromGroupChatSession(AgentID.ToString(), GroupID);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case (byte)InstantMessageDialog.SessionSend:
|
||||||
|
if (!m_groupData.hasAgentDroppedGroupChatSession(AgentID.ToString(), GroupID)
|
||||||
|
&& !m_groupData.hasAgentBeenInvitedToGroupChatSession(AgentID.ToString(), GroupID)
|
||||||
|
)
|
||||||
|
{
|
||||||
|
// Agent not in session and hasn't dropped from session
|
||||||
|
// Add them to the session for now, and Invite them
|
||||||
|
m_groupData.AgentInvitedToGroupChatSession(AgentID.ToString(), GroupID);
|
||||||
|
|
||||||
|
UUID toAgentID = new UUID(msg.toAgentID);
|
||||||
|
IClientAPI activeClient = GetActiveClient(toAgentID);
|
||||||
|
if (activeClient != null)
|
||||||
|
{
|
||||||
|
GroupRecord groupInfo = m_groupData.GetGroupRecord(UUID.Zero.ToString(), GroupID, null);
|
||||||
|
if (groupInfo != null)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: Sending chatterbox invite instant message");
|
||||||
|
|
||||||
|
// Force? open the group session dialog???
|
||||||
|
// and simultanously deliver the message, so we don't need to do a seperate client.SendInstantMessage(msg);
|
||||||
|
IEventQueue eq = activeClient.Scene.RequestModuleInterface<IEventQueue>();
|
||||||
|
eq.ChatterboxInvitation(
|
||||||
|
GroupID
|
||||||
|
, groupInfo.GroupName
|
||||||
|
, new UUID(msg.fromAgentID)
|
||||||
|
, msg.message
|
||||||
|
, new UUID(msg.toAgentID)
|
||||||
|
, msg.fromAgentName
|
||||||
|
, msg.dialog
|
||||||
|
, msg.timestamp
|
||||||
|
, msg.offline == 1
|
||||||
|
, (int)msg.ParentEstateID
|
||||||
|
, msg.Position
|
||||||
|
, 1
|
||||||
|
, new UUID(msg.imSessionID)
|
||||||
|
, msg.fromGroup
|
||||||
|
, OpenMetaverse.Utils.StringToBytes(groupInfo.GroupName)
|
||||||
|
);
|
||||||
|
|
||||||
|
eq.ChatterBoxSessionAgentListUpdates(
|
||||||
|
new UUID(GroupID)
|
||||||
|
, new UUID(msg.fromAgentID)
|
||||||
|
, new UUID(msg.toAgentID)
|
||||||
|
, false //canVoiceChat
|
||||||
|
, false //isModerator
|
||||||
|
, false //text mute
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (!m_groupData.hasAgentDroppedGroupChatSession(AgentID.ToString(), GroupID))
|
||||||
|
{
|
||||||
|
// User hasn't dropped, so they're in the session,
|
||||||
|
// maybe we should deliver it.
|
||||||
|
IClientAPI client = GetActiveClient(new UUID(msg.toAgentID));
|
||||||
|
if (client != null)
|
||||||
|
{
|
||||||
|
// Deliver locally, directly
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: Delivering to {0} locally", client.Name);
|
||||||
|
client.SendInstantMessage(msg);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: Received a message over the grid for a client that isn't here: {0}", msg.toAgentID);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: I don't know how to proccess a {0} message.", ((InstantMessageDialog)msg.dialog).ToString());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
|
||||||
|
#region ClientEvents
|
||||||
|
private void OnInstantMessage(IClientAPI remoteClient, GridInstantMessage im)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[Groups.Messaging]: {0} called", System.Reflection.MethodBase.GetCurrentMethod().Name);
|
||||||
|
|
||||||
|
DebugGridInstantMessage(im);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start group IM session
|
||||||
|
if ((im.dialog == (byte)InstantMessageDialog.SessionGroupStart))
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.InfoFormat("[Groups.Messaging]: imSessionID({0}) toAgentID({1})", im.imSessionID, im.toAgentID);
|
||||||
|
|
||||||
|
UUID GroupID = new UUID(im.imSessionID);
|
||||||
|
UUID AgentID = new UUID(im.fromAgentID);
|
||||||
|
|
||||||
|
GroupRecord groupInfo = m_groupData.GetGroupRecord(UUID.Zero.ToString(), GroupID, null);
|
||||||
|
|
||||||
|
if (groupInfo != null)
|
||||||
|
{
|
||||||
|
m_groupData.AgentInvitedToGroupChatSession(AgentID.ToString(), GroupID);
|
||||||
|
|
||||||
|
ChatterBoxSessionStartReplyViaCaps(remoteClient, groupInfo.GroupName, GroupID);
|
||||||
|
|
||||||
|
IEventQueue queue = remoteClient.Scene.RequestModuleInterface<IEventQueue>();
|
||||||
|
queue.ChatterBoxSessionAgentListUpdates(
|
||||||
|
GroupID
|
||||||
|
, AgentID
|
||||||
|
, new UUID(im.toAgentID)
|
||||||
|
, false //canVoiceChat
|
||||||
|
, false //isModerator
|
||||||
|
, false //text mute
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send a message from locally connected client to a group
|
||||||
|
if ((im.dialog == (byte)InstantMessageDialog.SessionSend))
|
||||||
|
{
|
||||||
|
UUID GroupID = new UUID(im.imSessionID);
|
||||||
|
UUID AgentID = new UUID(im.fromAgentID);
|
||||||
|
|
||||||
|
if (m_debugEnabled)
|
||||||
|
m_log.DebugFormat("[Groups.Messaging]: Send message to session for group {0} with session ID {1}", GroupID, im.imSessionID.ToString());
|
||||||
|
|
||||||
|
//If this agent is sending a message, then they want to be in the session
|
||||||
|
m_groupData.AgentInvitedToGroupChatSession(AgentID.ToString(), GroupID);
|
||||||
|
|
||||||
|
SendMessageToGroup(im, GroupID);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
void ChatterBoxSessionStartReplyViaCaps(IClientAPI remoteClient, string groupName, UUID groupID)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.DebugFormat("[Groups.Messaging]: {0} called", System.Reflection.MethodBase.GetCurrentMethod().Name);
|
||||||
|
|
||||||
|
OSDMap moderatedMap = new OSDMap(4);
|
||||||
|
moderatedMap.Add("voice", OSD.FromBoolean(false));
|
||||||
|
|
||||||
|
OSDMap sessionMap = new OSDMap(4);
|
||||||
|
sessionMap.Add("moderated_mode", moderatedMap);
|
||||||
|
sessionMap.Add("session_name", OSD.FromString(groupName));
|
||||||
|
sessionMap.Add("type", OSD.FromInteger(0));
|
||||||
|
sessionMap.Add("voice_enabled", OSD.FromBoolean(false));
|
||||||
|
|
||||||
|
OSDMap bodyMap = new OSDMap(4);
|
||||||
|
bodyMap.Add("session_id", OSD.FromUUID(groupID));
|
||||||
|
bodyMap.Add("temp_session_id", OSD.FromUUID(groupID));
|
||||||
|
bodyMap.Add("success", OSD.FromBoolean(true));
|
||||||
|
bodyMap.Add("session_info", sessionMap);
|
||||||
|
|
||||||
|
IEventQueue queue = remoteClient.Scene.RequestModuleInterface<IEventQueue>();
|
||||||
|
|
||||||
|
if (queue != null)
|
||||||
|
{
|
||||||
|
queue.Enqueue(queue.BuildEvent("ChatterBoxSessionStartReply", bodyMap), remoteClient.AgentId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void DebugGridInstantMessage(GridInstantMessage im)
|
||||||
|
{
|
||||||
|
// Don't log any normal IMs (privacy!)
|
||||||
|
if (m_debugEnabled && im.dialog != (byte)InstantMessageDialog.MessageFromAgent)
|
||||||
|
{
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: IM: fromGroup({0})", im.fromGroup ? "True" : "False");
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: IM: Dialog({0})", ((InstantMessageDialog)im.dialog).ToString());
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: IM: fromAgentID({0})", im.fromAgentID.ToString());
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: IM: fromAgentName({0})", im.fromAgentName.ToString());
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: IM: imSessionID({0})", im.imSessionID.ToString());
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: IM: message({0})", im.message.ToString());
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: IM: offline({0})", im.offline.ToString());
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: IM: toAgentID({0})", im.toAgentID.ToString());
|
||||||
|
m_log.WarnFormat("[Groups.Messaging]: IM: binaryBucket({0})", OpenMetaverse.Utils.BytesToHexString(im.binaryBucket, "BinaryBucket"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#region Client Tools
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Try to find an active IClientAPI reference for agentID giving preference to root connections
|
||||||
|
/// </summary>
|
||||||
|
private IClientAPI GetActiveClient(UUID agentID)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.WarnFormat("[Groups.Messaging]: Looking for local client {0}", agentID);
|
||||||
|
|
||||||
|
IClientAPI child = null;
|
||||||
|
|
||||||
|
// Try root avatar first
|
||||||
|
foreach (Scene scene in m_sceneList)
|
||||||
|
{
|
||||||
|
ScenePresence sp = scene.GetScenePresence(agentID);
|
||||||
|
if (sp != null)
|
||||||
|
{
|
||||||
|
if (!sp.IsChildAgent)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.WarnFormat("[Groups.Messaging]: Found root agent for client : {0}", sp.ControllingClient.Name);
|
||||||
|
return sp.ControllingClient;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.WarnFormat("[Groups.Messaging]: Found child agent for client : {0}", sp.ControllingClient.Name);
|
||||||
|
child = sp.ControllingClient;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we didn't find a root, then just return whichever child we found, or null if none
|
||||||
|
if (child == null)
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.WarnFormat("[Groups.Messaging]: Could not find local client for agent : {0}", agentID);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (m_debugEnabled) m_log.WarnFormat("[Groups.Messaging]: Returning child agent for client : {0}", child.Name);
|
||||||
|
}
|
||||||
|
return child;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
}
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,289 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Server.Base;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using log4net;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public class GroupsServiceHGConnector
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private string m_ServerURI;
|
||||||
|
private object m_Lock = new object();
|
||||||
|
|
||||||
|
public GroupsServiceHGConnector(string url)
|
||||||
|
{
|
||||||
|
m_ServerURI = url;
|
||||||
|
if (!m_ServerURI.EndsWith("/"))
|
||||||
|
m_ServerURI += "/";
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups.HGConnector]: Groups server at {0}", m_ServerURI);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool CreateProxy(string RequestingAgentID, string AgentID, string accessToken, UUID groupID, string url, string name, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string,object>();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["AgentID"] = AgentID.ToString();
|
||||||
|
sendData["AccessToken"] = accessToken;
|
||||||
|
sendData["GroupID"] = groupID.ToString();
|
||||||
|
sendData["Location"] = url;
|
||||||
|
sendData["Name"] = name;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("POSTGROUP", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString().ToLower() != "true")
|
||||||
|
{
|
||||||
|
reason = ret["REASON"].ToString();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroup(string AgentID, UUID GroupID, string token)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID;
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["AccessToken"] = GroupsDataUtils.Sanitize(token);
|
||||||
|
MakeRequest("REMOVEAGENTFROMGROUP", sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupRecord GetGroupRecord(string RequestingAgentID, UUID GroupID, string GroupName, string token)
|
||||||
|
{
|
||||||
|
if (GroupID == UUID.Zero && (GroupName == null || (GroupName != null && GroupName == string.Empty)))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
if (GroupID != UUID.Zero)
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
if (GroupName != null && GroupName != string.Empty)
|
||||||
|
sendData["Name"] = GroupsDataUtils.Sanitize(GroupName);
|
||||||
|
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["AccessToken"] = GroupsDataUtils.Sanitize(token);
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETGROUP", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return GroupsDataUtils.GroupRecord((Dictionary<string, object>)ret["RESULT"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupMembersData> GetGroupMembers(string RequestingAgentID, UUID GroupID, string token)
|
||||||
|
{
|
||||||
|
List<ExtendedGroupMembersData> members = new List<ExtendedGroupMembersData>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["AccessToken"] = GroupsDataUtils.Sanitize(token);
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETGROUPMEMBERS", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return members;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return members;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return members;
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
ExtendedGroupMembersData m = GroupsDataUtils.GroupMembersData((Dictionary<string, object>)v);
|
||||||
|
members.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return members;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetGroupRoles(string RequestingAgentID, UUID GroupID, string token)
|
||||||
|
{
|
||||||
|
List<GroupRolesData> roles = new List<GroupRolesData>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["AccessToken"] = GroupsDataUtils.Sanitize(token);
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETGROUPROLES", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return roles;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return roles;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return roles;
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
GroupRolesData m = GroupsDataUtils.GroupRolesData((Dictionary<string, object>)v);
|
||||||
|
roles.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return roles;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupRoleMembersData> GetGroupRoleMembers(string RequestingAgentID, UUID GroupID, string token)
|
||||||
|
{
|
||||||
|
List<ExtendedGroupRoleMembersData> rmembers = new List<ExtendedGroupRoleMembersData>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["AccessToken"] = GroupsDataUtils.Sanitize(token);
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETROLEMEMBERS", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return rmembers;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return rmembers;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return rmembers;
|
||||||
|
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
ExtendedGroupRoleMembersData m = GroupsDataUtils.GroupRoleMembersData((Dictionary<string, object>)v);
|
||||||
|
rmembers.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return rmembers;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddNotice(string RequestingAgentID, UUID groupID, UUID noticeID, string fromName, string subject, string message,
|
||||||
|
bool hasAttachment, byte attType, string attName, UUID attItemID, string attOwnerID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = groupID.ToString();
|
||||||
|
sendData["NoticeID"] = noticeID.ToString();
|
||||||
|
sendData["FromName"] = GroupsDataUtils.Sanitize(fromName);
|
||||||
|
sendData["Subject"] = GroupsDataUtils.Sanitize(subject);
|
||||||
|
sendData["Message"] = GroupsDataUtils.Sanitize(message);
|
||||||
|
sendData["HasAttachment"] = hasAttachment.ToString();
|
||||||
|
if (hasAttachment)
|
||||||
|
{
|
||||||
|
sendData["AttachmentType"] = attType.ToString();
|
||||||
|
sendData["AttachmentName"] = attName.ToString();
|
||||||
|
sendData["AttachmentItemID"] = attItemID.ToString();
|
||||||
|
sendData["AttachmentOwnerID"] = attOwnerID;
|
||||||
|
}
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("ADDNOTICE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString().ToLower() != "true")
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool VerifyNotice(UUID noticeID, UUID groupID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["NoticeID"] = noticeID.ToString();
|
||||||
|
sendData["GroupID"] = groupID.ToString();
|
||||||
|
Dictionary<string, object> ret = MakeRequest("VERIFYNOTICE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString().ToLower() != "true")
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//
|
||||||
|
|
||||||
|
#region Make Request
|
||||||
|
|
||||||
|
private Dictionary<string, object> MakeRequest(string method, Dictionary<string, object> sendData)
|
||||||
|
{
|
||||||
|
sendData["METHOD"] = method;
|
||||||
|
|
||||||
|
string reply = string.Empty;
|
||||||
|
lock (m_Lock)
|
||||||
|
reply = SynchronousRestFormsRequester.MakeRequest("POST",
|
||||||
|
m_ServerURI + "hg-groups",
|
||||||
|
ServerUtils.BuildQueryString(sendData));
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: reply was {0}", reply);
|
||||||
|
|
||||||
|
if (reply == string.Empty || reply == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
Dictionary<string, object> replyData = ServerUtils.ParseXmlResponse(
|
||||||
|
reply);
|
||||||
|
|
||||||
|
return replyData;
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,717 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Framework.Servers;
|
||||||
|
using OpenSim.Region.Framework.Scenes;
|
||||||
|
using OpenSim.Region.Framework.Interfaces;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using Mono.Addins;
|
||||||
|
using log4net;
|
||||||
|
using Nini.Config;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
[Extension(Path = "/OpenSim/RegionModules", NodeName = "RegionModule", Id = "GroupsServiceHGConnectorModule")]
|
||||||
|
public class GroupsServiceHGConnectorModule : ISharedRegionModule, IGroupsServicesConnector
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private bool m_Enabled = false;
|
||||||
|
private IGroupsServicesConnector m_LocalGroupsConnector;
|
||||||
|
private string m_LocalGroupsServiceLocation;
|
||||||
|
private IUserManagement m_UserManagement;
|
||||||
|
private IOfflineIMService m_OfflineIM;
|
||||||
|
private IMessageTransferModule m_Messaging;
|
||||||
|
private List<Scene> m_Scenes;
|
||||||
|
private ForeignImporter m_ForeignImporter;
|
||||||
|
private string m_ServiceLocation;
|
||||||
|
private IConfigSource m_Config;
|
||||||
|
|
||||||
|
private Dictionary<string, GroupsServiceHGConnector> m_NetworkConnectors = new Dictionary<string, GroupsServiceHGConnector>();
|
||||||
|
private RemoteConnectorCacheWrapper m_CacheWrapper; // for caching info of external group services
|
||||||
|
|
||||||
|
#region ISharedRegionModule
|
||||||
|
|
||||||
|
public void Initialise(IConfigSource config)
|
||||||
|
{
|
||||||
|
IConfig groupsConfig = config.Configs["Groups"];
|
||||||
|
if (groupsConfig == null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if ((groupsConfig.GetBoolean("Enabled", false) == false)
|
||||||
|
|| (groupsConfig.GetString("ServicesConnectorModule", string.Empty) != Name))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_Config = config;
|
||||||
|
m_ServiceLocation = groupsConfig.GetString("LocalService", "local"); // local or remote
|
||||||
|
m_LocalGroupsServiceLocation = groupsConfig.GetString("GroupsExternalURI", "http://127.0.0.1");
|
||||||
|
m_Scenes = new List<Scene>();
|
||||||
|
|
||||||
|
m_Enabled = true;
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups]: Initializing {0} with LocalService {1}", this.Name, m_ServiceLocation);
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Name
|
||||||
|
{
|
||||||
|
get { return "Groups HG Service Connector"; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type ReplaceableInterface
|
||||||
|
{
|
||||||
|
get { return null; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups]: Registering {0} with {1}", this.Name, scene.RegionInfo.RegionName);
|
||||||
|
scene.RegisterModuleInterface<IGroupsServicesConnector>(this);
|
||||||
|
m_Scenes.Add(scene);
|
||||||
|
|
||||||
|
scene.EventManager.OnNewClient += OnNewClient;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
scene.UnregisterModuleInterface<IGroupsServicesConnector>(this);
|
||||||
|
m_Scenes.Remove(scene);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RegionLoaded(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (m_UserManagement == null)
|
||||||
|
{
|
||||||
|
m_UserManagement = scene.RequestModuleInterface<IUserManagement>();
|
||||||
|
m_OfflineIM = scene.RequestModuleInterface<IOfflineIMService>();
|
||||||
|
m_Messaging = scene.RequestModuleInterface<IMessageTransferModule>();
|
||||||
|
m_ForeignImporter = new ForeignImporter(m_UserManagement);
|
||||||
|
|
||||||
|
if (m_ServiceLocation.Equals("local"))
|
||||||
|
{
|
||||||
|
m_LocalGroupsConnector = new GroupsServiceLocalConnectorModule(m_Config, m_UserManagement);
|
||||||
|
// Also, if local, create the endpoint for the HGGroupsService
|
||||||
|
new HGGroupsServiceRobustConnector(m_Config, MainServer.Instance, string.Empty,
|
||||||
|
scene.RequestModuleInterface<IOfflineIMService>(), scene.RequestModuleInterface<IUserAccountService>());
|
||||||
|
|
||||||
|
}
|
||||||
|
else
|
||||||
|
m_LocalGroupsConnector = new GroupsServiceRemoteConnectorModule(m_Config, m_UserManagement);
|
||||||
|
|
||||||
|
m_CacheWrapper = new RemoteConnectorCacheWrapper(m_UserManagement);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void PostInitialise()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Close()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
private void OnNewClient(IClientAPI client)
|
||||||
|
{
|
||||||
|
client.OnCompleteMovementToRegion += OnCompleteMovementToRegion;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OnCompleteMovementToRegion(IClientAPI client, bool arg2)
|
||||||
|
{
|
||||||
|
object sp = null;
|
||||||
|
if (client.Scene.TryGetScenePresence(client.AgentId, out sp))
|
||||||
|
{
|
||||||
|
if (sp is ScenePresence && ((ScenePresence)sp).PresenceType != PresenceType.Npc)
|
||||||
|
{
|
||||||
|
AgentCircuitData aCircuit = ((ScenePresence)sp).Scene.AuthenticateHandler.GetAgentCircuitData(client.AgentId);
|
||||||
|
if (aCircuit != null && (aCircuit.teleportFlags & (uint)Constants.TeleportFlags.ViaHGLogin) != 0 &&
|
||||||
|
m_OfflineIM != null && m_Messaging != null)
|
||||||
|
{
|
||||||
|
List<GridInstantMessage> ims = m_OfflineIM.GetMessages(aCircuit.AgentID);
|
||||||
|
if (ims != null && ims.Count > 0)
|
||||||
|
foreach (GridInstantMessage im in ims)
|
||||||
|
m_Messaging.SendInstantMessage(im, delegate(bool success) { });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#region IGroupsServicesConnector
|
||||||
|
|
||||||
|
public UUID CreateGroup(UUID RequestingAgentID, string name, string charter, bool showInList, UUID insigniaID, int membershipFee, bool openEnrollment,
|
||||||
|
bool allowPublish, bool maturePublish, UUID founderID, out string reason)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[Groups]: Creating group {0}", name);
|
||||||
|
reason = string.Empty;
|
||||||
|
if (m_UserManagement.IsLocalGridUser(RequestingAgentID))
|
||||||
|
return m_LocalGroupsConnector.CreateGroup(RequestingAgentID, name, charter, showInList, insigniaID,
|
||||||
|
membershipFee, openEnrollment, allowPublish, maturePublish, founderID, out reason);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
reason = "Only local grid users are allowed to create a new group";
|
||||||
|
return UUID.Zero;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool UpdateGroup(string RequestingAgentID, UUID groupID, string charter, bool showInList, UUID insigniaID, int membershipFee,
|
||||||
|
bool openEnrollment, bool allowPublish, bool maturePublish, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
string url = string.Empty;
|
||||||
|
string name = string.Empty;
|
||||||
|
if (IsLocal(groupID, out url, out name))
|
||||||
|
return m_LocalGroupsConnector.UpdateGroup(AgentUUI(RequestingAgentID), groupID, charter, showInList, insigniaID, membershipFee,
|
||||||
|
openEnrollment, allowPublish, maturePublish, out reason);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
reason = "Changes to remote group not allowed. Please go to the group's original world.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupRecord GetGroupRecord(string RequestingAgentID, UUID GroupID, string GroupName)
|
||||||
|
{
|
||||||
|
string url = string.Empty;
|
||||||
|
string name = string.Empty;
|
||||||
|
if (IsLocal(GroupID, out url, out name))
|
||||||
|
return m_LocalGroupsConnector.GetGroupRecord(AgentUUI(RequestingAgentID), GroupID, GroupName);
|
||||||
|
else if (url != string.Empty)
|
||||||
|
{
|
||||||
|
ExtendedGroupMembershipData membership = m_LocalGroupsConnector.GetAgentGroupMembership(RequestingAgentID, RequestingAgentID, GroupID);
|
||||||
|
string accessToken = string.Empty;
|
||||||
|
if (membership != null)
|
||||||
|
accessToken = membership.AccessToken;
|
||||||
|
else
|
||||||
|
return null;
|
||||||
|
|
||||||
|
GroupsServiceHGConnector c = GetConnector(url);
|
||||||
|
if (c != null)
|
||||||
|
{
|
||||||
|
ExtendedGroupRecord grec = m_CacheWrapper.GetGroupRecord(RequestingAgentID, GroupID, GroupName, delegate
|
||||||
|
{
|
||||||
|
return c.GetGroupRecord(AgentUUIForOutside(RequestingAgentID), GroupID, GroupName, accessToken);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (grec != null)
|
||||||
|
ImportForeigner(grec.FounderUUI);
|
||||||
|
return grec;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<DirGroupsReplyData> FindGroups(string RequestingAgentID, string search)
|
||||||
|
{
|
||||||
|
return m_LocalGroupsConnector.FindGroups(AgentUUI(RequestingAgentID), search);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupMembersData> GetGroupMembers(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
if (IsLocal(GroupID, out url, out gname))
|
||||||
|
return m_LocalGroupsConnector.GetGroupMembers(AgentUUI(RequestingAgentID), GroupID);
|
||||||
|
else if (!string.IsNullOrEmpty(url))
|
||||||
|
{
|
||||||
|
ExtendedGroupMembershipData membership = m_LocalGroupsConnector.GetAgentGroupMembership(RequestingAgentID, RequestingAgentID, GroupID);
|
||||||
|
string accessToken = string.Empty;
|
||||||
|
if (membership != null)
|
||||||
|
accessToken = membership.AccessToken;
|
||||||
|
else
|
||||||
|
return null;
|
||||||
|
|
||||||
|
GroupsServiceHGConnector c = GetConnector(url);
|
||||||
|
if (c != null)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetGroupMembers(RequestingAgentID, GroupID, delegate
|
||||||
|
{
|
||||||
|
return c.GetGroupMembers(AgentUUIForOutside(RequestingAgentID), GroupID, accessToken);
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return new List<GroupMembersData>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(groupID, out url, out gname))
|
||||||
|
return m_LocalGroupsConnector.AddGroupRole(AgentUUI(RequestingAgentID), groupID, roleID, name, description, title, powers, out reason);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
reason = "Operation not allowed outside this group's origin world.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool UpdateGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(groupID, out url, out gname))
|
||||||
|
return m_LocalGroupsConnector.UpdateGroupRole(AgentUUI(RequestingAgentID), groupID, roleID, name, description, title, powers);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveGroupRole(string RequestingAgentID, UUID groupID, UUID roleID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(groupID, out url, out gname))
|
||||||
|
m_LocalGroupsConnector.RemoveGroupRole(AgentUUI(RequestingAgentID), groupID, roleID);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetGroupRoles(string RequestingAgentID, UUID groupID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(groupID, out url, out gname))
|
||||||
|
return m_LocalGroupsConnector.GetGroupRoles(AgentUUI(RequestingAgentID), groupID);
|
||||||
|
else if (!string.IsNullOrEmpty(url))
|
||||||
|
{
|
||||||
|
ExtendedGroupMembershipData membership = m_LocalGroupsConnector.GetAgentGroupMembership(RequestingAgentID, RequestingAgentID, groupID);
|
||||||
|
string accessToken = string.Empty;
|
||||||
|
if (membership != null)
|
||||||
|
accessToken = membership.AccessToken;
|
||||||
|
else
|
||||||
|
return null;
|
||||||
|
|
||||||
|
GroupsServiceHGConnector c = GetConnector(url);
|
||||||
|
if (c != null)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetGroupRoles(RequestingAgentID, groupID, delegate
|
||||||
|
{
|
||||||
|
return c.GetGroupRoles(AgentUUIForOutside(RequestingAgentID), groupID, accessToken);
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return new List<GroupRolesData>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRoleMembersData> GetGroupRoleMembers(string RequestingAgentID, UUID groupID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(groupID, out url, out gname))
|
||||||
|
return m_LocalGroupsConnector.GetGroupRoleMembers(AgentUUI(RequestingAgentID), groupID);
|
||||||
|
else if (!string.IsNullOrEmpty(url))
|
||||||
|
{
|
||||||
|
ExtendedGroupMembershipData membership = m_LocalGroupsConnector.GetAgentGroupMembership(RequestingAgentID, RequestingAgentID, groupID);
|
||||||
|
string accessToken = string.Empty;
|
||||||
|
if (membership != null)
|
||||||
|
accessToken = membership.AccessToken;
|
||||||
|
else
|
||||||
|
return null;
|
||||||
|
|
||||||
|
GroupsServiceHGConnector c = GetConnector(url);
|
||||||
|
if (c != null)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetGroupRoleMembers(RequestingAgentID, groupID, delegate
|
||||||
|
{
|
||||||
|
return c.GetGroupRoleMembers(AgentUUIForOutside(RequestingAgentID), groupID, accessToken);
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return new List<GroupRoleMembersData>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddAgentToGroup(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID, string token, out string reason)
|
||||||
|
{
|
||||||
|
string url = string.Empty;
|
||||||
|
string name = string.Empty;
|
||||||
|
reason = string.Empty;
|
||||||
|
|
||||||
|
UUID uid = new UUID(AgentID);
|
||||||
|
if (IsLocal(GroupID, out url, out name))
|
||||||
|
{
|
||||||
|
if (m_UserManagement.IsLocalGridUser(uid)) // local user
|
||||||
|
{
|
||||||
|
// normal case: local group, local user
|
||||||
|
return m_LocalGroupsConnector.AddAgentToGroup(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID, RoleID, token, out reason);
|
||||||
|
}
|
||||||
|
else // local group, foreign user
|
||||||
|
{
|
||||||
|
// the user is accepting the invitation, or joining, where the group resides
|
||||||
|
token = UUID.Random().ToString();
|
||||||
|
bool success = m_LocalGroupsConnector.AddAgentToGroup(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID, RoleID, token, out reason);
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
url = m_UserManagement.GetUserServerURL(uid, "GroupsServerURI");
|
||||||
|
if (url == string.Empty)
|
||||||
|
{
|
||||||
|
reason = "User doesn't have a groups server";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
GroupsServiceHGConnector c = GetConnector(url);
|
||||||
|
if (c != null)
|
||||||
|
return c.CreateProxy(AgentUUI(RequestingAgentID), AgentID, token, GroupID, m_LocalGroupsServiceLocation, name, out reason);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (m_UserManagement.IsLocalGridUser(uid)) // local user
|
||||||
|
{
|
||||||
|
// foreign group, local user. She's been added already by the HG service.
|
||||||
|
// Let's just check
|
||||||
|
if (m_LocalGroupsConnector.GetAgentGroupMembership(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID) != null)
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
reason = "Operation not allowed outside this group's origin world";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroup(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, name = string.Empty;
|
||||||
|
if (!IsLocal(GroupID, out url, out name) && url != string.Empty)
|
||||||
|
{
|
||||||
|
ExtendedGroupMembershipData membership = m_LocalGroupsConnector.GetAgentGroupMembership(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID);
|
||||||
|
if (membership != null)
|
||||||
|
{
|
||||||
|
GroupsServiceHGConnector c = GetConnector(url);
|
||||||
|
if (c != null)
|
||||||
|
c.RemoveAgentFromGroup(AgentUUIForOutside(AgentID), GroupID, membership.AccessToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove from local service
|
||||||
|
m_LocalGroupsConnector.RemoveAgentFromGroup(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddAgentToGroupInvite(string RequestingAgentID, UUID inviteID, UUID groupID, UUID roleID, string agentID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(groupID, out url, out gname))
|
||||||
|
return m_LocalGroupsConnector.AddAgentToGroupInvite(AgentUUI(RequestingAgentID), inviteID, groupID, roleID, AgentUUI(agentID));
|
||||||
|
else
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupInviteInfo GetAgentToGroupInvite(string RequestingAgentID, UUID inviteID)
|
||||||
|
{
|
||||||
|
return m_LocalGroupsConnector.GetAgentToGroupInvite(AgentUUI(RequestingAgentID), inviteID); ;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentToGroupInvite(string RequestingAgentID, UUID inviteID)
|
||||||
|
{
|
||||||
|
m_LocalGroupsConnector.RemoveAgentToGroupInvite(AgentUUI(RequestingAgentID), inviteID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddAgentToGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(GroupID, out url, out gname))
|
||||||
|
m_LocalGroupsConnector.AddAgentToGroupRole(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID, RoleID);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(GroupID, out url, out gname))
|
||||||
|
m_LocalGroupsConnector.RemoveAgentFromGroupRole(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID, RoleID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetAgentGroupRoles(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(GroupID, out url, out gname))
|
||||||
|
return m_LocalGroupsConnector.GetAgentGroupRoles(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID);
|
||||||
|
else
|
||||||
|
return new List<GroupRolesData>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetAgentActiveGroup(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(GroupID, out url, out gname))
|
||||||
|
m_LocalGroupsConnector.SetAgentActiveGroup(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupMembershipData GetAgentActiveMembership(string RequestingAgentID, string AgentID)
|
||||||
|
{
|
||||||
|
return m_LocalGroupsConnector.GetAgentActiveMembership(AgentUUI(RequestingAgentID), AgentUUI(AgentID));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetAgentActiveGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(GroupID, out url, out gname))
|
||||||
|
m_LocalGroupsConnector.SetAgentActiveGroupRole(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID, RoleID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UpdateMembership(string RequestingAgentID, string AgentID, UUID GroupID, bool AcceptNotices, bool ListInProfile)
|
||||||
|
{
|
||||||
|
m_LocalGroupsConnector.UpdateMembership(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID, AcceptNotices, ListInProfile);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupMembershipData GetAgentGroupMembership(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(GroupID, out url, out gname))
|
||||||
|
return m_LocalGroupsConnector.GetAgentGroupMembership(AgentUUI(RequestingAgentID), AgentUUI(AgentID), GroupID);
|
||||||
|
else
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupMembershipData> GetAgentGroupMemberships(string RequestingAgentID, string AgentID)
|
||||||
|
{
|
||||||
|
return m_LocalGroupsConnector.GetAgentGroupMemberships(AgentUUI(RequestingAgentID), AgentUUI(AgentID));
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupNotice(string RequestingAgentID, UUID groupID, UUID noticeID, string fromName, string subject, string message,
|
||||||
|
bool hasAttachment, byte attType, string attName, UUID attItemID, string attOwnerID)
|
||||||
|
{
|
||||||
|
string url = string.Empty, gname = string.Empty;
|
||||||
|
|
||||||
|
if (IsLocal(groupID, out url, out gname))
|
||||||
|
{
|
||||||
|
if (m_LocalGroupsConnector.AddGroupNotice(AgentUUI(RequestingAgentID), groupID, noticeID, fromName, subject, message,
|
||||||
|
hasAttachment, attType, attName, attItemID, AgentUUI(attOwnerID)))
|
||||||
|
{
|
||||||
|
// then send the notice to every grid for which there are members in this group
|
||||||
|
List<GroupMembersData> members = m_LocalGroupsConnector.GetGroupMembers(AgentUUI(RequestingAgentID), groupID);
|
||||||
|
List<string> urls = new List<string>();
|
||||||
|
foreach (GroupMembersData m in members)
|
||||||
|
{
|
||||||
|
UUID userID = UUID.Zero;
|
||||||
|
if (!m_UserManagement.IsLocalGridUser(m.AgentID))
|
||||||
|
{
|
||||||
|
string gURL = m_UserManagement.GetUserServerURL(m.AgentID, "GroupsServerURI");
|
||||||
|
if (!urls.Contains(gURL))
|
||||||
|
urls.Add(gURL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// so we have the list of urls to send the notice to
|
||||||
|
// this may take a long time...
|
||||||
|
Util.FireAndForget(delegate
|
||||||
|
{
|
||||||
|
foreach (string u in urls)
|
||||||
|
{
|
||||||
|
GroupsServiceHGConnector c = GetConnector(u);
|
||||||
|
if (c != null)
|
||||||
|
{
|
||||||
|
c.AddNotice(AgentUUIForOutside(RequestingAgentID), groupID, noticeID, fromName, subject, message,
|
||||||
|
hasAttachment, attType, attName, attItemID, AgentUUIForOutside(attOwnerID));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupNoticeInfo GetGroupNotice(string RequestingAgentID, UUID noticeID)
|
||||||
|
{
|
||||||
|
GroupNoticeInfo notice = m_LocalGroupsConnector.GetGroupNotice(AgentUUI(RequestingAgentID), noticeID);
|
||||||
|
|
||||||
|
if (notice != null && notice.noticeData.HasAttachment && notice.noticeData.AttachmentOwnerID != null)
|
||||||
|
ImportForeigner(notice.noticeData.AttachmentOwnerID);
|
||||||
|
|
||||||
|
return notice;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupNoticeData> GetGroupNotices(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_LocalGroupsConnector.GetGroupNotices(AgentUUI(RequestingAgentID), GroupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ResetAgentGroupChatSessions(string agentID)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool hasAgentBeenInvitedToGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool hasAgentDroppedGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AgentDroppedFromGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AgentInvitedToGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region hypergrid groups
|
||||||
|
|
||||||
|
private string AgentUUI(string AgentIDStr)
|
||||||
|
{
|
||||||
|
UUID AgentID = UUID.Zero;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
AgentID = new UUID(AgentIDStr);
|
||||||
|
}
|
||||||
|
catch (FormatException)
|
||||||
|
{
|
||||||
|
return AgentID.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_UserManagement.IsLocalGridUser(AgentID))
|
||||||
|
return AgentID.ToString();
|
||||||
|
|
||||||
|
AgentCircuitData agent = null;
|
||||||
|
foreach (Scene scene in m_Scenes)
|
||||||
|
{
|
||||||
|
agent = scene.AuthenticateHandler.GetAgentCircuitData(AgentID);
|
||||||
|
if (agent != null)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (agent == null) // oops
|
||||||
|
return AgentID.ToString();
|
||||||
|
|
||||||
|
return Util.ProduceUserUniversalIdentifier(agent);
|
||||||
|
}
|
||||||
|
|
||||||
|
private string AgentUUIForOutside(string AgentIDStr)
|
||||||
|
{
|
||||||
|
UUID AgentID = UUID.Zero;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
AgentID = new UUID(AgentIDStr);
|
||||||
|
}
|
||||||
|
catch (FormatException)
|
||||||
|
{
|
||||||
|
return AgentID.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
AgentCircuitData agent = null;
|
||||||
|
foreach (Scene scene in m_Scenes)
|
||||||
|
{
|
||||||
|
agent = scene.AuthenticateHandler.GetAgentCircuitData(AgentID);
|
||||||
|
if (agent != null)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (agent == null) // oops
|
||||||
|
return AgentID.ToString();
|
||||||
|
|
||||||
|
return Util.ProduceUserUniversalIdentifier(agent);
|
||||||
|
}
|
||||||
|
|
||||||
|
private UUID ImportForeigner(string uID)
|
||||||
|
{
|
||||||
|
UUID userID = UUID.Zero;
|
||||||
|
string url = string.Empty, first = string.Empty, last = string.Empty, tmp = string.Empty;
|
||||||
|
if (Util.ParseUniversalUserIdentifier(uID, out userID, out url, out first, out last, out tmp))
|
||||||
|
m_UserManagement.AddUser(userID, first, last, url);
|
||||||
|
|
||||||
|
return userID;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool IsLocal(UUID groupID, out string serviceLocation, out string name)
|
||||||
|
{
|
||||||
|
serviceLocation = string.Empty;
|
||||||
|
name = string.Empty;
|
||||||
|
ExtendedGroupRecord group = m_LocalGroupsConnector.GetGroupRecord(UUID.Zero.ToString(), groupID, string.Empty);
|
||||||
|
if (group == null)
|
||||||
|
{
|
||||||
|
//m_log.DebugFormat("[XXX]: IsLocal? group {0} not found -- no.", groupID);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
serviceLocation = group.ServiceLocation;
|
||||||
|
name = group.GroupName;
|
||||||
|
bool isLocal = (group.ServiceLocation == string.Empty);
|
||||||
|
//m_log.DebugFormat("[XXX]: IsLocal? {0}", isLocal);
|
||||||
|
return isLocal;
|
||||||
|
}
|
||||||
|
|
||||||
|
private GroupsServiceHGConnector GetConnector(string url)
|
||||||
|
{
|
||||||
|
lock (m_NetworkConnectors)
|
||||||
|
{
|
||||||
|
if (m_NetworkConnectors.ContainsKey(url))
|
||||||
|
return m_NetworkConnectors[url];
|
||||||
|
|
||||||
|
GroupsServiceHGConnector c = new GroupsServiceHGConnector(url);
|
||||||
|
m_NetworkConnectors[url] = c;
|
||||||
|
}
|
||||||
|
|
||||||
|
return m_NetworkConnectors[url];
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,444 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Text;
|
||||||
|
using System.Xml;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using Nini.Config;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Server.Base;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
using OpenSim.Framework.Servers.HttpServer;
|
||||||
|
using OpenSim.Server.Handlers.Base;
|
||||||
|
using log4net;
|
||||||
|
using OpenMetaverse;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public class HGGroupsServiceRobustConnector : ServiceConnector
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private HGGroupsService m_GroupsService;
|
||||||
|
private string m_HomeURI = string.Empty;
|
||||||
|
private string m_ConfigName = "Groups";
|
||||||
|
|
||||||
|
// Called by Robust shell
|
||||||
|
public HGGroupsServiceRobustConnector(IConfigSource config, IHttpServer server, string configName) :
|
||||||
|
this(config, server, configName, null, null)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called by the sim-bound module
|
||||||
|
public HGGroupsServiceRobustConnector(IConfigSource config, IHttpServer server, string configName, IOfflineIMService im, IUserAccountService users) :
|
||||||
|
base(config, server, configName)
|
||||||
|
{
|
||||||
|
if (configName != String.Empty)
|
||||||
|
m_ConfigName = configName;
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups.RobustHGConnector]: Starting with config name {0}", m_ConfigName);
|
||||||
|
|
||||||
|
string homeURI = Util.GetConfigVarFromSections<string>(config, "HomeURI",
|
||||||
|
new string[] { "Startup", "Hypergrid", m_ConfigName}, string.Empty);
|
||||||
|
if (homeURI == string.Empty)
|
||||||
|
throw new Exception(String.Format("[Groups.RobustHGConnector]: please provide the HomeURI [Startup] or in section {0}", m_ConfigName));
|
||||||
|
|
||||||
|
IConfig cnf = config.Configs[m_ConfigName];
|
||||||
|
if (cnf == null)
|
||||||
|
throw new Exception(String.Format("[Groups.RobustHGConnector]: {0} section does not exist", m_ConfigName));
|
||||||
|
|
||||||
|
if (im == null)
|
||||||
|
{
|
||||||
|
string imDll = cnf.GetString("OfflineIMService", string.Empty);
|
||||||
|
if (imDll == string.Empty)
|
||||||
|
throw new Exception(String.Format("[Groups.RobustHGConnector]: please provide OfflineIMService in section {0}", m_ConfigName));
|
||||||
|
|
||||||
|
Object[] args = new Object[] { config };
|
||||||
|
im = ServerUtils.LoadPlugin<IOfflineIMService>(imDll, args);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (users == null)
|
||||||
|
{
|
||||||
|
string usersDll = cnf.GetString("UserAccountService", string.Empty);
|
||||||
|
if (usersDll == string.Empty)
|
||||||
|
throw new Exception(String.Format("[Groups.RobustHGConnector]: please provide UserAccountService in section {0}", m_ConfigName));
|
||||||
|
|
||||||
|
Object[] args = new Object[] { config };
|
||||||
|
users = ServerUtils.LoadPlugin<IUserAccountService>(usersDll, args);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_GroupsService = new HGGroupsService(config, im, users, homeURI);
|
||||||
|
|
||||||
|
server.AddStreamHandler(new HGGroupsServicePostHandler(m_GroupsService));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class HGGroupsServicePostHandler : BaseStreamHandler
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private HGGroupsService m_GroupsService;
|
||||||
|
|
||||||
|
public HGGroupsServicePostHandler(HGGroupsService service) :
|
||||||
|
base("POST", "/hg-groups")
|
||||||
|
{
|
||||||
|
m_GroupsService = service;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override byte[] Handle(string path, Stream requestData,
|
||||||
|
IOSHttpRequest httpRequest, IOSHttpResponse httpResponse)
|
||||||
|
{
|
||||||
|
StreamReader sr = new StreamReader(requestData);
|
||||||
|
string body = sr.ReadToEnd();
|
||||||
|
sr.Close();
|
||||||
|
body = body.Trim();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: query String: {0}", body);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Dictionary<string, object> request =
|
||||||
|
ServerUtils.ParseQueryString(body);
|
||||||
|
|
||||||
|
if (!request.ContainsKey("METHOD"))
|
||||||
|
return FailureResult();
|
||||||
|
|
||||||
|
string method = request["METHOD"].ToString();
|
||||||
|
request.Remove("METHOD");
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups.RobustHGConnector]: {0}", method);
|
||||||
|
switch (method)
|
||||||
|
{
|
||||||
|
case "POSTGROUP":
|
||||||
|
return HandleAddGroupProxy(request);
|
||||||
|
case "REMOVEAGENTFROMGROUP":
|
||||||
|
return HandleRemoveAgentFromGroup(request);
|
||||||
|
case "GETGROUP":
|
||||||
|
return HandleGetGroup(request);
|
||||||
|
case "ADDNOTICE":
|
||||||
|
return HandleAddNotice(request);
|
||||||
|
case "VERIFYNOTICE":
|
||||||
|
return HandleVerifyNotice(request);
|
||||||
|
case "GETGROUPMEMBERS":
|
||||||
|
return HandleGetGroupMembers(request);
|
||||||
|
case "GETGROUPROLES":
|
||||||
|
return HandleGetGroupRoles(request);
|
||||||
|
case "GETROLEMEMBERS":
|
||||||
|
return HandleGetRoleMembers(request);
|
||||||
|
|
||||||
|
}
|
||||||
|
m_log.DebugFormat("[Groups.RobustHGConnector]: unknown method request: {0}", method);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[Groups.RobustHGConnector]: Exception {0}", e.StackTrace);
|
||||||
|
}
|
||||||
|
|
||||||
|
return FailureResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleAddGroupProxy(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID")
|
||||||
|
|| !request.ContainsKey("AgentID")
|
||||||
|
|| !request.ContainsKey("AccessToken") || !request.ContainsKey("Location"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string RequestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
string agentID = request["AgentID"].ToString();
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string accessToken = request["AccessToken"].ToString();
|
||||||
|
string location = request["Location"].ToString();
|
||||||
|
string name = string.Empty;
|
||||||
|
if (request.ContainsKey("Name"))
|
||||||
|
name = request["Name"].ToString();
|
||||||
|
|
||||||
|
string reason = string.Empty;
|
||||||
|
bool success = m_GroupsService.CreateGroupProxy(RequestingAgentID, agentID, accessToken, groupID, location, name, out reason);
|
||||||
|
result["REASON"] = reason;
|
||||||
|
result["RESULT"] = success.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleRemoveAgentFromGroup(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("AccessToken") || !request.ContainsKey("AgentID") ||
|
||||||
|
!request.ContainsKey("GroupID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string agentID = request["AgentID"].ToString();
|
||||||
|
string token = request["AccessToken"].ToString();
|
||||||
|
string reason = string.Empty;
|
||||||
|
|
||||||
|
m_GroupsService.RemoveAgentFromGroup(agentID, agentID, groupID, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
result["RESULT"] = "true";
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(ServerUtils.BuildXmlResponse(result));
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetGroup(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("AccessToken"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string RequestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
string token = request["AccessToken"].ToString();
|
||||||
|
|
||||||
|
UUID groupID = UUID.Zero;
|
||||||
|
string groupName = string.Empty;
|
||||||
|
|
||||||
|
if (request.ContainsKey("GroupID"))
|
||||||
|
groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
if (request.ContainsKey("Name"))
|
||||||
|
groupName = request["Name"].ToString();
|
||||||
|
|
||||||
|
ExtendedGroupRecord grec = m_GroupsService.GetGroupRecord(RequestingAgentID, groupID, groupName, token);
|
||||||
|
if (grec == null)
|
||||||
|
NullResult(result, "Group not found");
|
||||||
|
else
|
||||||
|
result["RESULT"] = GroupsDataUtils.GroupRecord(grec);
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetGroupMembers(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") || !request.ContainsKey("AccessToken"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
string token = request["AccessToken"].ToString();
|
||||||
|
|
||||||
|
List<ExtendedGroupMembersData> members = m_GroupsService.GetGroupMembers(requestingAgentID, groupID, token);
|
||||||
|
if (members == null || (members != null && members.Count == 0))
|
||||||
|
{
|
||||||
|
NullResult(result, "No members");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (ExtendedGroupMembersData m in members)
|
||||||
|
{
|
||||||
|
dict["m-" + i++] = GroupsDataUtils.GroupMembersData(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetGroupRoles(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") || !request.ContainsKey("AccessToken"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
string token = request["AccessToken"].ToString();
|
||||||
|
|
||||||
|
List<GroupRolesData> roles = m_GroupsService.GetGroupRoles(requestingAgentID, groupID, token);
|
||||||
|
if (roles == null || (roles != null && roles.Count == 0))
|
||||||
|
{
|
||||||
|
NullResult(result, "No members");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (GroupRolesData r in roles)
|
||||||
|
dict["r-" + i++] = GroupsDataUtils.GroupRolesData(r);
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetRoleMembers(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") || !request.ContainsKey("AccessToken"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
string token = request["AccessToken"].ToString();
|
||||||
|
|
||||||
|
List<ExtendedGroupRoleMembersData> rmembers = m_GroupsService.GetGroupRoleMembers(requestingAgentID, groupID, token);
|
||||||
|
if (rmembers == null || (rmembers != null && rmembers.Count == 0))
|
||||||
|
{
|
||||||
|
NullResult(result, "No members");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (ExtendedGroupRoleMembersData rm in rmembers)
|
||||||
|
dict["rm-" + i++] = GroupsDataUtils.GroupRoleMembersData(rm);
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleAddNotice(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") || !request.ContainsKey("NoticeID") ||
|
||||||
|
!request.ContainsKey("FromName") || !request.ContainsKey("Subject") || !request.ContainsKey("Message") ||
|
||||||
|
!request.ContainsKey("HasAttachment"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
|
||||||
|
bool hasAtt = bool.Parse(request["HasAttachment"].ToString());
|
||||||
|
byte attType = 0;
|
||||||
|
string attName = string.Empty;
|
||||||
|
string attOwner = string.Empty;
|
||||||
|
UUID attItem = UUID.Zero;
|
||||||
|
if (request.ContainsKey("AttachmentType"))
|
||||||
|
attType = byte.Parse(request["AttachmentType"].ToString());
|
||||||
|
if (request.ContainsKey("AttachmentName"))
|
||||||
|
attName = request["AttachmentType"].ToString();
|
||||||
|
if (request.ContainsKey("AttachmentItemID"))
|
||||||
|
attItem = new UUID(request["AttachmentItemID"].ToString());
|
||||||
|
if (request.ContainsKey("AttachmentOwnerID"))
|
||||||
|
attOwner = request["AttachmentOwnerID"].ToString();
|
||||||
|
|
||||||
|
bool success = m_GroupsService.AddNotice(request["RequestingAgentID"].ToString(), new UUID(request["GroupID"].ToString()),
|
||||||
|
new UUID(request["NoticeID"].ToString()), request["FromName"].ToString(), request["Subject"].ToString(),
|
||||||
|
request["Message"].ToString(), hasAtt, attType, attName, attItem, attOwner);
|
||||||
|
|
||||||
|
result["RESULT"] = success.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleVerifyNotice(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("NoticeID") || !request.ContainsKey("GroupID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID noticeID = new UUID(request["NoticeID"].ToString());
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
|
||||||
|
bool success = m_GroupsService.VerifyNotice(noticeID, groupID);
|
||||||
|
//m_log.DebugFormat("[XXX]: VerifyNotice returned {0}", success);
|
||||||
|
result["RESULT"] = success.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//
|
||||||
|
|
||||||
|
#region Helpers
|
||||||
|
|
||||||
|
private void NullResult(Dictionary<string, object> result, string reason)
|
||||||
|
{
|
||||||
|
result["RESULT"] = "NULL";
|
||||||
|
result["REASON"] = reason;
|
||||||
|
}
|
||||||
|
|
||||||
|
private byte[] FailureResult()
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
NullResult(result, "Unknown method");
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,118 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using OpenMetaverse;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public interface IGroupsServicesConnector
|
||||||
|
{
|
||||||
|
UUID CreateGroup(UUID RequestingAgentID, string name, string charter, bool showInList, UUID insigniaID, int membershipFee,
|
||||||
|
bool openEnrollment, bool allowPublish, bool maturePublish, UUID founderID, out string reason);
|
||||||
|
bool UpdateGroup(string RequestingAgentID, UUID groupID, string charter, bool showInList, UUID insigniaID, int membershipFee,
|
||||||
|
bool openEnrollment, bool allowPublish, bool maturePublish, out string reason);
|
||||||
|
ExtendedGroupRecord GetGroupRecord(string RequestingAgentID, UUID GroupID, string GroupName);
|
||||||
|
List<DirGroupsReplyData> FindGroups(string RequestingAgentID, string search);
|
||||||
|
List<GroupMembersData> GetGroupMembers(string RequestingAgentID, UUID GroupID);
|
||||||
|
|
||||||
|
bool AddGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers, out string reason);
|
||||||
|
bool UpdateGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers);
|
||||||
|
void RemoveGroupRole(string RequestingAgentID, UUID groupID, UUID roleID);
|
||||||
|
List<GroupRolesData> GetGroupRoles(string RequestingAgentID, UUID GroupID);
|
||||||
|
List<GroupRoleMembersData> GetGroupRoleMembers(string RequestingAgentID, UUID GroupID);
|
||||||
|
|
||||||
|
bool AddAgentToGroup(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID, string token, out string reason);
|
||||||
|
void RemoveAgentFromGroup(string RequestingAgentID, string AgentID, UUID GroupID);
|
||||||
|
|
||||||
|
bool AddAgentToGroupInvite(string RequestingAgentID, UUID inviteID, UUID groupID, UUID roleID, string agentID);
|
||||||
|
GroupInviteInfo GetAgentToGroupInvite(string RequestingAgentID, UUID inviteID);
|
||||||
|
void RemoveAgentToGroupInvite(string RequestingAgentID, UUID inviteID);
|
||||||
|
|
||||||
|
void AddAgentToGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID);
|
||||||
|
void RemoveAgentFromGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID);
|
||||||
|
List<GroupRolesData> GetAgentGroupRoles(string RequestingAgentID, string AgentID, UUID GroupID);
|
||||||
|
|
||||||
|
void SetAgentActiveGroup(string RequestingAgentID, string AgentID, UUID GroupID);
|
||||||
|
ExtendedGroupMembershipData GetAgentActiveMembership(string RequestingAgentID, string AgentID);
|
||||||
|
|
||||||
|
void SetAgentActiveGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID);
|
||||||
|
void UpdateMembership(string RequestingAgentID, string AgentID, UUID GroupID, bool AcceptNotices, bool ListInProfile);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Get information about a specific group to which the user belongs.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="RequestingAgentID">The agent requesting the information.</param>
|
||||||
|
/// <param name="AgentID">The agent requested.</param>
|
||||||
|
/// <param name="GroupID">The group requested.</param>
|
||||||
|
/// <returns>
|
||||||
|
/// If the user is a member of the group then the data structure is returned. If not, then null is returned.
|
||||||
|
/// </returns>
|
||||||
|
ExtendedGroupMembershipData GetAgentGroupMembership(string RequestingAgentID, string AgentID, UUID GroupID);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Get information about the groups to which a user belongs.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="RequestingAgentID">The agent requesting the information.</param>
|
||||||
|
/// <param name="AgentID">The agent requested.</param>
|
||||||
|
/// <returns>
|
||||||
|
/// Information about the groups to which the user belongs. If the user belongs to no groups then an empty
|
||||||
|
/// list is returned.
|
||||||
|
/// </returns>
|
||||||
|
List<GroupMembershipData> GetAgentGroupMemberships(string RequestingAgentID, string AgentID);
|
||||||
|
|
||||||
|
bool AddGroupNotice(string RequestingAgentID, UUID groupID, UUID noticeID, string fromName, string subject, string message,
|
||||||
|
bool hasAttachment, byte attType, string attName, UUID attItemID, string attOwnerID);
|
||||||
|
GroupNoticeInfo GetGroupNotice(string RequestingAgentID, UUID noticeID);
|
||||||
|
List<ExtendedGroupNoticeData> GetGroupNotices(string RequestingAgentID, UUID GroupID);
|
||||||
|
|
||||||
|
void ResetAgentGroupChatSessions(string agentID);
|
||||||
|
bool hasAgentBeenInvitedToGroupChatSession(string agentID, UUID groupID);
|
||||||
|
bool hasAgentDroppedGroupChatSession(string agentID, UUID groupID);
|
||||||
|
void AgentDroppedFromGroupChatSession(string agentID, UUID groupID);
|
||||||
|
void AgentInvitedToGroupChatSession(string agentID, UUID groupID);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class GroupInviteInfo
|
||||||
|
{
|
||||||
|
public UUID GroupID = UUID.Zero;
|
||||||
|
public UUID RoleID = UUID.Zero;
|
||||||
|
public string AgentID = string.Empty;
|
||||||
|
public UUID InviteID = UUID.Zero;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class GroupNoticeInfo
|
||||||
|
{
|
||||||
|
public ExtendedGroupNoticeData noticeData = new ExtendedGroupNoticeData();
|
||||||
|
public UUID GroupID = UUID.Zero;
|
||||||
|
public string Message = string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,347 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Region.Framework.Scenes;
|
||||||
|
using OpenSim.Region.Framework.Interfaces;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using Mono.Addins;
|
||||||
|
using log4net;
|
||||||
|
using Nini.Config;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
[Extension(Path = "/OpenSim/RegionModules", NodeName = "RegionModule", Id = "GroupsServiceLocalConnectorModule")]
|
||||||
|
public class GroupsServiceLocalConnectorModule : ISharedRegionModule, IGroupsServicesConnector
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private bool m_Enabled = false;
|
||||||
|
private GroupsService m_GroupsService;
|
||||||
|
private IUserManagement m_UserManagement;
|
||||||
|
private List<Scene> m_Scenes;
|
||||||
|
private ForeignImporter m_ForeignImporter;
|
||||||
|
|
||||||
|
#region constructors
|
||||||
|
public GroupsServiceLocalConnectorModule()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupsServiceLocalConnectorModule(IConfigSource config, IUserManagement uman)
|
||||||
|
{
|
||||||
|
Init(config);
|
||||||
|
m_UserManagement = uman;
|
||||||
|
m_ForeignImporter = new ForeignImporter(uman);
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
private void Init(IConfigSource config)
|
||||||
|
{
|
||||||
|
m_GroupsService = new GroupsService(config);
|
||||||
|
m_Scenes = new List<Scene>();
|
||||||
|
}
|
||||||
|
|
||||||
|
#region ISharedRegionModule
|
||||||
|
|
||||||
|
public void Initialise(IConfigSource config)
|
||||||
|
{
|
||||||
|
IConfig groupsConfig = config.Configs["Groups"];
|
||||||
|
if (groupsConfig == null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if ((groupsConfig.GetBoolean("Enabled", false) == false)
|
||||||
|
|| (groupsConfig.GetString("ServicesConnectorModule", string.Empty) != Name))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Init(config);
|
||||||
|
m_Enabled = true;
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups]: Initializing {0}", this.Name);
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Name
|
||||||
|
{
|
||||||
|
get { return "Groups Local Service Connector"; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type ReplaceableInterface
|
||||||
|
{
|
||||||
|
get { return null; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups]: Registering {0} with {1}", this.Name, scene.RegionInfo.RegionName);
|
||||||
|
scene.RegisterModuleInterface<IGroupsServicesConnector>(this);
|
||||||
|
m_Scenes.Add(scene);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
scene.UnregisterModuleInterface<IGroupsServicesConnector>(this);
|
||||||
|
m_Scenes.Remove(scene);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RegionLoaded(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (m_UserManagement == null)
|
||||||
|
{
|
||||||
|
m_UserManagement = scene.RequestModuleInterface<IUserManagement>();
|
||||||
|
m_ForeignImporter = new ForeignImporter(m_UserManagement);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void PostInitialise()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Close()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region IGroupsServicesConnector
|
||||||
|
|
||||||
|
public UUID CreateGroup(UUID RequestingAgentID, string name, string charter, bool showInList, UUID insigniaID, int membershipFee, bool openEnrollment,
|
||||||
|
bool allowPublish, bool maturePublish, UUID founderID, out string reason)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[Groups]: Creating group {0}", name);
|
||||||
|
reason = string.Empty;
|
||||||
|
return m_GroupsService.CreateGroup(RequestingAgentID.ToString(), name, charter, showInList, insigniaID,
|
||||||
|
membershipFee, openEnrollment, allowPublish, maturePublish, founderID, out reason);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool UpdateGroup(string RequestingAgentID, UUID groupID, string charter, bool showInList, UUID insigniaID, int membershipFee,
|
||||||
|
bool openEnrollment, bool allowPublish, bool maturePublish, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
m_GroupsService.UpdateGroup(RequestingAgentID, groupID, charter, showInList, insigniaID, membershipFee, openEnrollment, allowPublish, maturePublish);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupRecord GetGroupRecord(string RequestingAgentID, UUID GroupID, string GroupName)
|
||||||
|
{
|
||||||
|
if (GroupID != UUID.Zero)
|
||||||
|
return m_GroupsService.GetGroupRecord(RequestingAgentID, GroupID);
|
||||||
|
else if (GroupName != null)
|
||||||
|
return m_GroupsService.GetGroupRecord(RequestingAgentID, GroupName);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<DirGroupsReplyData> FindGroups(string RequestingAgentID, string search)
|
||||||
|
{
|
||||||
|
return m_GroupsService.FindGroups(RequestingAgentID, search);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupMembersData> GetGroupMembers(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
List<ExtendedGroupMembersData> _members = m_GroupsService.GetGroupMembers(RequestingAgentID, GroupID);
|
||||||
|
if (_members != null && _members.Count > 0)
|
||||||
|
{
|
||||||
|
List<GroupMembersData> members = _members.ConvertAll<GroupMembersData>(new Converter<ExtendedGroupMembersData, GroupMembersData>(m_ForeignImporter.ConvertGroupMembersData));
|
||||||
|
return members;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new List<GroupMembersData>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers, out string reason)
|
||||||
|
{
|
||||||
|
return m_GroupsService.AddGroupRole(RequestingAgentID, groupID, roleID, name, description, title, powers, out reason);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool UpdateGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers)
|
||||||
|
{
|
||||||
|
return m_GroupsService.UpdateGroupRole(RequestingAgentID, groupID, roleID, name, description, title, powers);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveGroupRole(string RequestingAgentID, UUID groupID, UUID roleID)
|
||||||
|
{
|
||||||
|
m_GroupsService.RemoveGroupRole(RequestingAgentID, groupID, roleID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetGroupRoles(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetGroupRoles(RequestingAgentID, GroupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRoleMembersData> GetGroupRoleMembers(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
List<ExtendedGroupRoleMembersData> _rm = m_GroupsService.GetGroupRoleMembers(RequestingAgentID, GroupID);
|
||||||
|
if (_rm != null && _rm.Count > 0)
|
||||||
|
{
|
||||||
|
List<GroupRoleMembersData> rm = _rm.ConvertAll<GroupRoleMembersData>(new Converter<ExtendedGroupRoleMembersData, GroupRoleMembersData>(m_ForeignImporter.ConvertGroupRoleMembersData));
|
||||||
|
return rm;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new List<GroupRoleMembersData>();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddAgentToGroup(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID, string token, out string reason)
|
||||||
|
{
|
||||||
|
return m_GroupsService.AddAgentToGroup(RequestingAgentID, AgentID, GroupID, RoleID, token, out reason);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroup(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
m_GroupsService.RemoveAgentFromGroup(RequestingAgentID, AgentID, GroupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddAgentToGroupInvite(string RequestingAgentID, UUID inviteID, UUID groupID, UUID roleID, string agentID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.AddAgentToGroupInvite(RequestingAgentID, inviteID, groupID, roleID, agentID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupInviteInfo GetAgentToGroupInvite(string RequestingAgentID, UUID inviteID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetAgentToGroupInvite(RequestingAgentID, inviteID); ;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentToGroupInvite(string RequestingAgentID, UUID inviteID)
|
||||||
|
{
|
||||||
|
m_GroupsService.RemoveAgentToGroupInvite(RequestingAgentID, inviteID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddAgentToGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
m_GroupsService.AddAgentToGroupRole(RequestingAgentID, AgentID, GroupID, RoleID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
m_GroupsService.RemoveAgentFromGroupRole(RequestingAgentID, AgentID, GroupID, RoleID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetAgentGroupRoles(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetAgentGroupRoles(RequestingAgentID, AgentID, GroupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetAgentActiveGroup(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
m_GroupsService.SetAgentActiveGroup(RequestingAgentID, AgentID, GroupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupMembershipData GetAgentActiveMembership(string RequestingAgentID, string AgentID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetAgentActiveMembership(RequestingAgentID, AgentID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetAgentActiveGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
m_GroupsService.SetAgentActiveGroupRole(RequestingAgentID, AgentID, GroupID, RoleID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UpdateMembership(string RequestingAgentID, string AgentID, UUID GroupID, bool AcceptNotices, bool ListInProfile)
|
||||||
|
{
|
||||||
|
m_GroupsService.UpdateMembership(RequestingAgentID, AgentID, GroupID, AcceptNotices, ListInProfile);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupMembershipData GetAgentGroupMembership(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetAgentGroupMembership(RequestingAgentID, AgentID, GroupID); ;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupMembershipData> GetAgentGroupMemberships(string RequestingAgentID, string AgentID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetAgentGroupMemberships(RequestingAgentID, AgentID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupNotice(string RequestingAgentID, UUID groupID, UUID noticeID, string fromName, string subject, string message,
|
||||||
|
bool hasAttachment, byte attType, string attName, UUID attItemID, string attOwnerID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.AddGroupNotice(RequestingAgentID, groupID, noticeID, fromName, subject, message,
|
||||||
|
hasAttachment, attType, attName, attItemID, attOwnerID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupNoticeInfo GetGroupNotice(string RequestingAgentID, UUID noticeID)
|
||||||
|
{
|
||||||
|
GroupNoticeInfo notice = m_GroupsService.GetGroupNotice(RequestingAgentID, noticeID);
|
||||||
|
|
||||||
|
//if (notice != null && notice.noticeData.HasAttachment && notice.noticeData.AttachmentOwnerID != null)
|
||||||
|
//{
|
||||||
|
// UUID userID = UUID.Zero;
|
||||||
|
// string url = string.Empty, first = string.Empty, last = string.Empty, tmp = string.Empty;
|
||||||
|
// Util.ParseUniversalUserIdentifier(notice.noticeData.AttachmentOwnerID, out userID, out url, out first, out last, out tmp);
|
||||||
|
// if (url != string.Empty)
|
||||||
|
// m_UserManagement.AddUser(userID, first, last, url);
|
||||||
|
//}
|
||||||
|
|
||||||
|
return notice;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupNoticeData> GetGroupNotices(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetGroupNotices(RequestingAgentID, GroupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ResetAgentGroupChatSessions(string agentID)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool hasAgentBeenInvitedToGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool hasAgentDroppedGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AgentDroppedFromGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AgentInvitedToGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,36 @@
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
using Mono.Addins;
|
||||||
|
|
||||||
|
// General Information about an assembly is controlled through the following
|
||||||
|
// set of attributes. Change these attribute values to modify the information
|
||||||
|
// associated with an assembly.
|
||||||
|
[assembly: AssemblyTitle("OpenSim.Addons.Groups")]
|
||||||
|
[assembly: AssemblyDescription("")]
|
||||||
|
[assembly: AssemblyConfiguration("")]
|
||||||
|
[assembly: AssemblyCompany("http://opensimulator.org")]
|
||||||
|
[assembly: AssemblyProduct("OpenSim.Addons.Groups")]
|
||||||
|
[assembly: AssemblyCopyright("Copyright (c) OpenSimulator.org Developers")]
|
||||||
|
[assembly: AssemblyTrademark("")]
|
||||||
|
[assembly: AssemblyCulture("")]
|
||||||
|
|
||||||
|
// Setting ComVisible to false makes the types in this assembly not visible
|
||||||
|
// to COM components. If you need to access a type in this assembly from
|
||||||
|
// COM, set the ComVisible attribute to true on that type.
|
||||||
|
[assembly: ComVisible(false)]
|
||||||
|
|
||||||
|
// The following GUID is for the ID of the typelib if this project is exposed to COM
|
||||||
|
[assembly: Guid("313d4865-d179-4735-9b5a-fe74885878b2")]
|
||||||
|
|
||||||
|
// Version information for an assembly consists of the following four values:
|
||||||
|
//
|
||||||
|
// Major Version
|
||||||
|
// Minor Version
|
||||||
|
// Build Number
|
||||||
|
// Revision
|
||||||
|
//
|
||||||
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
|
|
||||||
|
[assembly: Addin("OpenSim.Groups", "0.1")]
|
||||||
|
[assembly: AddinDependency("OpenSim", "0.5")]
|
|
@ -0,0 +1,642 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Server.Base;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using log4net;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public class GroupsServiceRemoteConnector
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private string m_ServerURI;
|
||||||
|
private object m_Lock = new object();
|
||||||
|
|
||||||
|
public GroupsServiceRemoteConnector(string url)
|
||||||
|
{
|
||||||
|
m_ServerURI = url;
|
||||||
|
if (!m_ServerURI.EndsWith("/"))
|
||||||
|
m_ServerURI += "/";
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups.RemoteConnector]: Groups server at {0}", m_ServerURI);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupRecord CreateGroup(string RequestingAgentID, string name, string charter, bool showInList, UUID insigniaID, int membershipFee, bool openEnrollment,
|
||||||
|
bool allowPublish, bool maturePublish, UUID founderID, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
|
||||||
|
ExtendedGroupRecord rec = new ExtendedGroupRecord();
|
||||||
|
rec.AllowPublish = allowPublish;
|
||||||
|
rec.Charter = charter;
|
||||||
|
rec.FounderID = founderID;
|
||||||
|
rec.GroupName = name;
|
||||||
|
rec.GroupPicture = insigniaID;
|
||||||
|
rec.MaturePublish = maturePublish;
|
||||||
|
rec.MembershipFee = membershipFee;
|
||||||
|
rec.OpenEnrollment = openEnrollment;
|
||||||
|
rec.ShowInList = showInList;
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = GroupsDataUtils.GroupRecord(rec);
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "ADD";
|
||||||
|
Dictionary<string, object> ret = MakeRequest("PUTGROUP", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
{
|
||||||
|
reason = ret["REASON"].ToString();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return GroupsDataUtils.GroupRecord((Dictionary<string, object>)ret["RESULT"]);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupRecord UpdateGroup(string RequestingAgentID, UUID groupID, string charter, bool showInList, UUID insigniaID, int membershipFee, bool openEnrollment, bool allowPublish, bool maturePublish)
|
||||||
|
{
|
||||||
|
ExtendedGroupRecord rec = new ExtendedGroupRecord();
|
||||||
|
rec.AllowPublish = allowPublish;
|
||||||
|
rec.Charter = charter;
|
||||||
|
rec.GroupPicture = insigniaID;
|
||||||
|
rec.MaturePublish = maturePublish;
|
||||||
|
rec.GroupID = groupID;
|
||||||
|
rec.MembershipFee = membershipFee;
|
||||||
|
rec.OpenEnrollment = openEnrollment;
|
||||||
|
rec.ShowInList = showInList;
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = GroupsDataUtils.GroupRecord(rec);
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "UPDATE";
|
||||||
|
Dictionary<string, object> ret = MakeRequest("PUTGROUP", sendData);
|
||||||
|
|
||||||
|
if (ret == null || (ret != null && ret["RESULT"].ToString() == "NULL"))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return GroupsDataUtils.GroupRecord((Dictionary<string, object>)ret["RESULT"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupRecord GetGroupRecord(string RequestingAgentID, UUID GroupID, string GroupName)
|
||||||
|
{
|
||||||
|
if (GroupID == UUID.Zero && (GroupName == null || (GroupName != null && GroupName == string.Empty)))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
if (GroupID != UUID.Zero)
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
if (GroupName != null && GroupName != string.Empty)
|
||||||
|
sendData["Name"] = GroupsDataUtils.Sanitize(GroupName);
|
||||||
|
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETGROUP", sendData);
|
||||||
|
|
||||||
|
if (ret == null || (ret != null && ret["RESULT"].ToString() == "NULL"))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return GroupsDataUtils.GroupRecord((Dictionary<string, object>)ret["RESULT"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupMembershipData AddAgentToGroup(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID, string token, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string,object>();
|
||||||
|
sendData["AgentID"] = AgentID;
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RoleID"] = RoleID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["AccessToken"] = token;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("ADDAGENTTOGROUP", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
{
|
||||||
|
reason = ret["REASON"].ToString();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return GroupsDataUtils.GroupMembershipData((Dictionary<string, object>)ret["RESULT"]);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroup(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID;
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
MakeRequest("REMOVEAGENTFROMGROUP", sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupMembershipData GetMembership(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID;
|
||||||
|
if (GroupID != UUID.Zero)
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETMEMBERSHIP", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return GroupsDataUtils.GroupMembershipData((Dictionary<string, object>)ret["RESULT"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupMembershipData> GetMemberships(string RequestingAgentID, string AgentID)
|
||||||
|
{
|
||||||
|
List<GroupMembershipData> memberships = new List<GroupMembershipData>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID;
|
||||||
|
sendData["ALL"] = "true";
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETMEMBERSHIP", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return memberships;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return memberships;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return memberships;
|
||||||
|
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
GroupMembershipData m = GroupsDataUtils.GroupMembershipData((Dictionary<string, object>)v);
|
||||||
|
memberships.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return memberships;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupMembersData> GetGroupMembers(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
List<ExtendedGroupMembersData> members = new List<ExtendedGroupMembersData>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETGROUPMEMBERS", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return members;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return members;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return members;
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
ExtendedGroupMembersData m = GroupsDataUtils.GroupMembersData((Dictionary<string, object>)v);
|
||||||
|
members.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return members;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = groupID.ToString();
|
||||||
|
sendData["RoleID"] = roleID.ToString();
|
||||||
|
sendData["Name"] = GroupsDataUtils.Sanitize(name);
|
||||||
|
sendData["Description"] = GroupsDataUtils.Sanitize(description);
|
||||||
|
sendData["Title"] = GroupsDataUtils.Sanitize(title);
|
||||||
|
sendData["Powers"] = powers.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "ADD";
|
||||||
|
Dictionary<string, object> ret = MakeRequest("PUTROLE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString().ToLower() != "true")
|
||||||
|
{
|
||||||
|
reason = ret["REASON"].ToString();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool UpdateGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = groupID.ToString();
|
||||||
|
sendData["RoleID"] = roleID.ToString();
|
||||||
|
sendData["Name"] = GroupsDataUtils.Sanitize(name);
|
||||||
|
sendData["Description"] = GroupsDataUtils.Sanitize(description);
|
||||||
|
sendData["Title"] = GroupsDataUtils.Sanitize(title);
|
||||||
|
sendData["Powers"] = powers.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "UPDATE";
|
||||||
|
Dictionary<string, object> ret = MakeRequest("PUTROLE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString().ToLower() != "true")
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveGroupRole(string RequestingAgentID, UUID groupID, UUID roleID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = groupID.ToString();
|
||||||
|
sendData["RoleID"] = roleID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
MakeRequest("REMOVEROLE", sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetGroupRoles(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
List<GroupRolesData> roles = new List<GroupRolesData>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETGROUPROLES", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return roles;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return roles;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return roles;
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
GroupRolesData m = GroupsDataUtils.GroupRolesData((Dictionary<string, object>)v);
|
||||||
|
roles.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return roles;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupRoleMembersData> GetGroupRoleMembers(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
List<ExtendedGroupRoleMembersData> rmembers = new List<ExtendedGroupRoleMembersData>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETROLEMEMBERS", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return rmembers;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return rmembers;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return rmembers;
|
||||||
|
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
ExtendedGroupRoleMembersData m = GroupsDataUtils.GroupRoleMembersData((Dictionary<string, object>)v);
|
||||||
|
rmembers.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return rmembers;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddAgentToGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID.ToString();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RoleID"] = RoleID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "ADD";
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("AGENTROLE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString().ToLower() != "true")
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool RemoveAgentFromGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID.ToString();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RoleID"] = RoleID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "DELETE";
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("AGENTROLE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString().ToLower() != "true")
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetAgentGroupRoles(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
List<GroupRolesData> roles = new List<GroupRolesData>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID.ToString();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETAGENTROLES", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return roles;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return roles;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return roles;
|
||||||
|
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
GroupRolesData m = GroupsDataUtils.GroupRolesData((Dictionary<string, object>)v);
|
||||||
|
roles.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return roles;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupMembershipData SetAgentActiveGroup(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID.ToString();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "GROUP";
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("SETACTIVE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return GroupsDataUtils.GroupMembershipData((Dictionary<string, object>)ret["RESULT"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetAgentActiveGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID.ToString();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RoleID"] = RoleID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "ROLE";
|
||||||
|
|
||||||
|
MakeRequest("SETACTIVE", sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UpdateMembership(string RequestingAgentID, string AgentID, UUID GroupID, bool AcceptNotices, bool ListInProfile)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["AgentID"] = AgentID.ToString();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["AcceptNotices"] = AcceptNotices.ToString();
|
||||||
|
sendData["ListInProfile"] = ListInProfile.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
MakeRequest("UPDATEMEMBERSHIP", sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddAgentToGroupInvite(string RequestingAgentID, UUID inviteID, UUID groupID, UUID roleID, string agentID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["InviteID"] = inviteID.ToString();
|
||||||
|
sendData["GroupID"] = groupID.ToString();
|
||||||
|
sendData["RoleID"] = roleID.ToString();
|
||||||
|
sendData["AgentID"] = agentID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "ADD";
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("INVITE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString().ToLower() != "true") // it may return "NULL"
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupInviteInfo GetAgentToGroupInvite(string RequestingAgentID, UUID inviteID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["InviteID"] = inviteID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "GET";
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("INVITE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return GroupsDataUtils.GroupInviteInfo((Dictionary<string, object>)ret["RESULT"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentToGroupInvite(string RequestingAgentID, UUID inviteID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["InviteID"] = inviteID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
sendData["OP"] = "DELETE";
|
||||||
|
|
||||||
|
MakeRequest("INVITE", sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupNotice(string RequestingAgentID, UUID groupID, UUID noticeID, string fromName, string subject, string message,
|
||||||
|
bool hasAttachment, byte attType, string attName, UUID attItemID, string attOwnerID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = groupID.ToString();
|
||||||
|
sendData["NoticeID"] = noticeID.ToString();
|
||||||
|
sendData["FromName"] = GroupsDataUtils.Sanitize(fromName);
|
||||||
|
sendData["Subject"] = GroupsDataUtils.Sanitize(subject);
|
||||||
|
sendData["Message"] = GroupsDataUtils.Sanitize(message);
|
||||||
|
sendData["HasAttachment"] = hasAttachment.ToString();
|
||||||
|
if (hasAttachment)
|
||||||
|
{
|
||||||
|
sendData["AttachmentType"] = attType.ToString();
|
||||||
|
sendData["AttachmentName"] = attName.ToString();
|
||||||
|
sendData["AttachmentItemID"] = attItemID.ToString();
|
||||||
|
sendData["AttachmentOwnerID"] = attOwnerID;
|
||||||
|
}
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("ADDNOTICE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString().ToLower() != "true")
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupNoticeInfo GetGroupNotice(string RequestingAgentID, UUID noticeID)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["NoticeID"] = noticeID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETNOTICES", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return GroupsDataUtils.GroupNoticeInfo((Dictionary<string, object>)ret["RESULT"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupNoticeData> GetGroupNotices(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
List<ExtendedGroupNoticeData> notices = new List<ExtendedGroupNoticeData>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["GroupID"] = GroupID.ToString();
|
||||||
|
sendData["RequestingAgentID"] = RequestingAgentID;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GETNOTICES", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return notices;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return notices;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return notices;
|
||||||
|
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
ExtendedGroupNoticeData m = GroupsDataUtils.GroupNoticeData((Dictionary<string, object>)v);
|
||||||
|
notices.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return notices;
|
||||||
|
}
|
||||||
|
|
||||||
|
#region Make Request
|
||||||
|
|
||||||
|
private Dictionary<string, object> MakeRequest(string method, Dictionary<string, object> sendData)
|
||||||
|
{
|
||||||
|
sendData["METHOD"] = method;
|
||||||
|
|
||||||
|
string reply = string.Empty;
|
||||||
|
lock (m_Lock)
|
||||||
|
reply = SynchronousRestFormsRequester.MakeRequest("POST",
|
||||||
|
m_ServerURI + "groups",
|
||||||
|
ServerUtils.BuildQueryString(sendData));
|
||||||
|
|
||||||
|
if (reply == string.Empty)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
Dictionary<string, object> replyData = ServerUtils.ParseXmlResponse(
|
||||||
|
reply);
|
||||||
|
|
||||||
|
return replyData;
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,434 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Region.Framework.Scenes;
|
||||||
|
using OpenSim.Region.Framework.Interfaces;
|
||||||
|
using OpenSim.Server.Base;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using Mono.Addins;
|
||||||
|
using log4net;
|
||||||
|
using Nini.Config;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
[Extension(Path = "/OpenSim/RegionModules", NodeName = "RegionModule", Id = "GroupsServiceRemoteConnectorModule")]
|
||||||
|
public class GroupsServiceRemoteConnectorModule : ISharedRegionModule, IGroupsServicesConnector
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private bool m_Enabled = false;
|
||||||
|
private GroupsServiceRemoteConnector m_GroupsService;
|
||||||
|
private IUserManagement m_UserManagement;
|
||||||
|
private List<Scene> m_Scenes;
|
||||||
|
|
||||||
|
private RemoteConnectorCacheWrapper m_CacheWrapper;
|
||||||
|
|
||||||
|
#region constructors
|
||||||
|
public GroupsServiceRemoteConnectorModule()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupsServiceRemoteConnectorModule(IConfigSource config, IUserManagement uman)
|
||||||
|
{
|
||||||
|
Init(config);
|
||||||
|
m_UserManagement = uman;
|
||||||
|
m_CacheWrapper = new RemoteConnectorCacheWrapper(m_UserManagement);
|
||||||
|
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
private void Init(IConfigSource config)
|
||||||
|
{
|
||||||
|
IConfig groupsConfig = config.Configs["Groups"];
|
||||||
|
string url = groupsConfig.GetString("GroupsServerURI", string.Empty);
|
||||||
|
if (!Uri.IsWellFormedUriString(url, UriKind.Absolute))
|
||||||
|
throw new Exception(string.Format("[Groups.RemoteConnector]: Malformed groups server URL {0}. Fix it or disable the Groups feature.", url));
|
||||||
|
|
||||||
|
m_GroupsService = new GroupsServiceRemoteConnector(url);
|
||||||
|
m_Scenes = new List<Scene>();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#region ISharedRegionModule
|
||||||
|
|
||||||
|
public void Initialise(IConfigSource config)
|
||||||
|
{
|
||||||
|
IConfig groupsConfig = config.Configs["Groups"];
|
||||||
|
if (groupsConfig == null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if ((groupsConfig.GetBoolean("Enabled", false) == false)
|
||||||
|
|| (groupsConfig.GetString("ServicesConnectorModule", string.Empty) != Name))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Init(config);
|
||||||
|
|
||||||
|
m_Enabled = true;
|
||||||
|
m_log.DebugFormat("[Groups.RemoteConnector]: Initializing {0}", this.Name);
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Name
|
||||||
|
{
|
||||||
|
get { return "Groups Remote Service Connector"; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type ReplaceableInterface
|
||||||
|
{
|
||||||
|
get { return null; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups.RemoteConnector]: Registering {0} with {1}", this.Name, scene.RegionInfo.RegionName);
|
||||||
|
scene.RegisterModuleInterface<IGroupsServicesConnector>(this);
|
||||||
|
m_Scenes.Add(scene);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
scene.UnregisterModuleInterface<IGroupsServicesConnector>(this);
|
||||||
|
m_Scenes.Remove(scene);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RegionLoaded(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (m_UserManagement == null)
|
||||||
|
{
|
||||||
|
m_UserManagement = scene.RequestModuleInterface<IUserManagement>();
|
||||||
|
m_CacheWrapper = new RemoteConnectorCacheWrapper(m_UserManagement);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void PostInitialise()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Close()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region IGroupsServicesConnector
|
||||||
|
|
||||||
|
public UUID CreateGroup(UUID RequestingAgentID, string name, string charter, bool showInList, UUID insigniaID, int membershipFee, bool openEnrollment,
|
||||||
|
bool allowPublish, bool maturePublish, UUID founderID, out string reason)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[Groups.RemoteConnector]: Creating group {0}", name);
|
||||||
|
string r = string.Empty;
|
||||||
|
|
||||||
|
UUID groupID = m_CacheWrapper.CreateGroup(RequestingAgentID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.CreateGroup(RequestingAgentID.ToString(), name, charter, showInList, insigniaID,
|
||||||
|
membershipFee, openEnrollment, allowPublish, maturePublish, founderID, out r);
|
||||||
|
});
|
||||||
|
|
||||||
|
reason = r;
|
||||||
|
return groupID;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool UpdateGroup(string RequestingAgentID, UUID groupID, string charter, bool showInList, UUID insigniaID, int membershipFee,
|
||||||
|
bool openEnrollment, bool allowPublish, bool maturePublish, out string reason)
|
||||||
|
{
|
||||||
|
string r = string.Empty;
|
||||||
|
|
||||||
|
bool success = m_CacheWrapper.UpdateGroup(groupID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.UpdateGroup(RequestingAgentID, groupID, charter, showInList, insigniaID, membershipFee, openEnrollment, allowPublish, maturePublish);
|
||||||
|
});
|
||||||
|
|
||||||
|
reason = r;
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupRecord GetGroupRecord(string RequestingAgentID, UUID GroupID, string GroupName)
|
||||||
|
{
|
||||||
|
if (GroupID == UUID.Zero && (GroupName == null || GroupName != null && GroupName == string.Empty))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return m_CacheWrapper.GetGroupRecord(RequestingAgentID,GroupID,GroupName, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetGroupRecord(RequestingAgentID, GroupID, GroupName);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<DirGroupsReplyData> FindGroups(string RequestingAgentID, string search)
|
||||||
|
{
|
||||||
|
// TODO!
|
||||||
|
return new List<DirGroupsReplyData>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddAgentToGroup(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID, string token, out string reason)
|
||||||
|
{
|
||||||
|
string agentFullID = AgentID;
|
||||||
|
m_log.DebugFormat("[Groups.RemoteConnector]: Add agent {0} to group {1}", agentFullID, GroupID);
|
||||||
|
string r = string.Empty;
|
||||||
|
|
||||||
|
bool success = m_CacheWrapper.AddAgentToGroup(RequestingAgentID, AgentID, GroupID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.AddAgentToGroup(RequestingAgentID, agentFullID, GroupID, RoleID, token, out r);
|
||||||
|
});
|
||||||
|
|
||||||
|
reason = r;
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroup(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
m_CacheWrapper.RemoveAgentFromGroup(RequestingAgentID, AgentID, GroupID, delegate
|
||||||
|
{
|
||||||
|
m_GroupsService.RemoveAgentFromGroup(RequestingAgentID, AgentID, GroupID);
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetAgentActiveGroup(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
m_CacheWrapper.SetAgentActiveGroup(AgentID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.SetAgentActiveGroup(RequestingAgentID, AgentID, GroupID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupMembershipData GetAgentActiveMembership(string RequestingAgentID, string AgentID)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetAgentActiveMembership(AgentID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetMembership(RequestingAgentID, AgentID, UUID.Zero);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupMembershipData GetAgentGroupMembership(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetAgentGroupMembership(AgentID, GroupID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetMembership(RequestingAgentID, AgentID, GroupID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupMembershipData> GetAgentGroupMemberships(string RequestingAgentID, string AgentID)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetAgentGroupMemberships(AgentID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetMemberships(RequestingAgentID, AgentID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public List<GroupMembersData> GetGroupMembers(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetGroupMembers(RequestingAgentID, GroupID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetGroupMembers(RequestingAgentID, GroupID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers, out string reason)
|
||||||
|
{
|
||||||
|
string r = string.Empty;
|
||||||
|
bool success = m_CacheWrapper.AddGroupRole(groupID, roleID, description, name, powers, title, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.AddGroupRole(RequestingAgentID, groupID, roleID, name, description, title, powers, out r);
|
||||||
|
});
|
||||||
|
|
||||||
|
reason = r;
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool UpdateGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, string name, string description, string title, ulong powers)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.UpdateGroupRole(groupID, roleID, name, description, title, powers, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.UpdateGroupRole(RequestingAgentID, groupID, roleID, name, description, title, powers);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveGroupRole(string RequestingAgentID, UUID groupID, UUID roleID)
|
||||||
|
{
|
||||||
|
m_CacheWrapper.RemoveGroupRole(RequestingAgentID, groupID, roleID, delegate
|
||||||
|
{
|
||||||
|
m_GroupsService.RemoveGroupRole(RequestingAgentID, groupID, roleID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetGroupRoles(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetGroupRoles(RequestingAgentID, GroupID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetGroupRoles(RequestingAgentID, GroupID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRoleMembersData> GetGroupRoleMembers(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetGroupRoleMembers(RequestingAgentID, GroupID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetGroupRoleMembers(RequestingAgentID, GroupID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddAgentToGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
m_CacheWrapper.AddAgentToGroupRole(RequestingAgentID, AgentID, GroupID, RoleID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.AddAgentToGroupRole(RequestingAgentID, AgentID, GroupID, RoleID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
m_CacheWrapper.RemoveAgentFromGroupRole(RequestingAgentID, AgentID, GroupID, RoleID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.RemoveAgentFromGroupRole(RequestingAgentID, AgentID, GroupID, RoleID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetAgentGroupRoles(string RequestingAgentID, string AgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetAgentGroupRoles(RequestingAgentID, AgentID, GroupID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetAgentGroupRoles(RequestingAgentID, AgentID, GroupID); ;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetAgentActiveGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID)
|
||||||
|
{
|
||||||
|
m_CacheWrapper.SetAgentActiveGroupRole(AgentID, GroupID, delegate
|
||||||
|
{
|
||||||
|
m_GroupsService.SetAgentActiveGroupRole(RequestingAgentID, AgentID, GroupID, RoleID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UpdateMembership(string RequestingAgentID, string AgentID, UUID GroupID, bool AcceptNotices, bool ListInProfile)
|
||||||
|
{
|
||||||
|
m_CacheWrapper.UpdateMembership(AgentID, GroupID, AcceptNotices, ListInProfile, delegate
|
||||||
|
{
|
||||||
|
m_GroupsService.UpdateMembership(RequestingAgentID, AgentID, GroupID, AcceptNotices, ListInProfile);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddAgentToGroupInvite(string RequestingAgentID, UUID inviteID, UUID groupID, UUID roleID, string agentID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.AddAgentToGroupInvite(RequestingAgentID, inviteID, groupID, roleID, agentID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupInviteInfo GetAgentToGroupInvite(string RequestingAgentID, UUID inviteID)
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetAgentToGroupInvite(RequestingAgentID, inviteID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentToGroupInvite(string RequestingAgentID, UUID inviteID)
|
||||||
|
{
|
||||||
|
m_GroupsService.RemoveAgentToGroupInvite(RequestingAgentID, inviteID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupNotice(string RequestingAgentID, UUID groupID, UUID noticeID, string fromName, string subject, string message,
|
||||||
|
bool hasAttachment, byte attType, string attName, UUID attItemID, string attOwnerID)
|
||||||
|
{
|
||||||
|
GroupNoticeInfo notice = new GroupNoticeInfo();
|
||||||
|
notice.GroupID = groupID;
|
||||||
|
notice.Message = message;
|
||||||
|
notice.noticeData = new ExtendedGroupNoticeData();
|
||||||
|
notice.noticeData.AttachmentItemID = attItemID;
|
||||||
|
notice.noticeData.AttachmentName = attName;
|
||||||
|
notice.noticeData.AttachmentOwnerID = attOwnerID.ToString();
|
||||||
|
notice.noticeData.AttachmentType = attType;
|
||||||
|
notice.noticeData.FromName = fromName;
|
||||||
|
notice.noticeData.HasAttachment = hasAttachment;
|
||||||
|
notice.noticeData.NoticeID = noticeID;
|
||||||
|
notice.noticeData.Subject = subject;
|
||||||
|
notice.noticeData.Timestamp = (uint)Util.UnixTimeSinceEpoch();
|
||||||
|
|
||||||
|
return m_CacheWrapper.AddGroupNotice(groupID, noticeID, notice, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.AddGroupNotice(RequestingAgentID, groupID, noticeID, fromName, subject, message,
|
||||||
|
hasAttachment, attType, attName, attItemID, attOwnerID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupNoticeInfo GetGroupNotice(string RequestingAgentID, UUID noticeID)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetGroupNotice(noticeID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetGroupNotice(RequestingAgentID, noticeID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupNoticeData> GetGroupNotices(string RequestingAgentID, UUID GroupID)
|
||||||
|
{
|
||||||
|
return m_CacheWrapper.GetGroupNotices(GroupID, delegate
|
||||||
|
{
|
||||||
|
return m_GroupsService.GetGroupNotices(RequestingAgentID, GroupID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ResetAgentGroupChatSessions(string agentID)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool hasAgentBeenInvitedToGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool hasAgentDroppedGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AgentDroppedFromGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AgentInvitedToGroupChatSession(string agentID, UUID groupID)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,760 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Text;
|
||||||
|
using System.Xml;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using Nini.Config;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Server.Base;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
using OpenSim.Framework.Servers.HttpServer;
|
||||||
|
using OpenSim.Server.Handlers.Base;
|
||||||
|
using log4net;
|
||||||
|
using OpenMetaverse;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public class GroupsServiceRobustConnector : ServiceConnector
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private GroupsService m_GroupsService;
|
||||||
|
private string m_ConfigName = "Groups";
|
||||||
|
|
||||||
|
public GroupsServiceRobustConnector(IConfigSource config, IHttpServer server, string configName) :
|
||||||
|
base(config, server, configName)
|
||||||
|
{
|
||||||
|
if (configName != String.Empty)
|
||||||
|
m_ConfigName = configName;
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups.RobustConnector]: Starting with config name {0}", m_ConfigName);
|
||||||
|
|
||||||
|
m_GroupsService = new GroupsService(config);
|
||||||
|
|
||||||
|
server.AddStreamHandler(new GroupsServicePostHandler(m_GroupsService));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class GroupsServicePostHandler : BaseStreamHandler
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private GroupsService m_GroupsService;
|
||||||
|
|
||||||
|
public GroupsServicePostHandler(GroupsService service) :
|
||||||
|
base("POST", "/groups")
|
||||||
|
{
|
||||||
|
m_GroupsService = service;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override byte[] Handle(string path, Stream requestData,
|
||||||
|
IOSHttpRequest httpRequest, IOSHttpResponse httpResponse)
|
||||||
|
{
|
||||||
|
StreamReader sr = new StreamReader(requestData);
|
||||||
|
string body = sr.ReadToEnd();
|
||||||
|
sr.Close();
|
||||||
|
body = body.Trim();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: query String: {0}", body);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Dictionary<string, object> request =
|
||||||
|
ServerUtils.ParseQueryString(body);
|
||||||
|
|
||||||
|
if (!request.ContainsKey("METHOD"))
|
||||||
|
return FailureResult();
|
||||||
|
|
||||||
|
string method = request["METHOD"].ToString();
|
||||||
|
request.Remove("METHOD");
|
||||||
|
|
||||||
|
m_log.DebugFormat("[Groups.Handler]: {0}", method);
|
||||||
|
switch (method)
|
||||||
|
{
|
||||||
|
case "PUTGROUP":
|
||||||
|
return HandleAddOrUpdateGroup(request);
|
||||||
|
case "GETGROUP":
|
||||||
|
return HandleGetGroup(request);
|
||||||
|
case "ADDAGENTTOGROUP":
|
||||||
|
return HandleAddAgentToGroup(request);
|
||||||
|
case "REMOVEAGENTFROMGROUP":
|
||||||
|
return HandleRemoveAgentFromGroup(request);
|
||||||
|
case "GETMEMBERSHIP":
|
||||||
|
return HandleGetMembership(request);
|
||||||
|
case "GETGROUPMEMBERS":
|
||||||
|
return HandleGetGroupMembers(request);
|
||||||
|
case "PUTROLE":
|
||||||
|
return HandlePutRole(request);
|
||||||
|
case "REMOVEROLE":
|
||||||
|
return HandleRemoveRole(request);
|
||||||
|
case "GETGROUPROLES":
|
||||||
|
return HandleGetGroupRoles(request);
|
||||||
|
case "GETROLEMEMBERS":
|
||||||
|
return HandleGetRoleMembers(request);
|
||||||
|
case "AGENTROLE":
|
||||||
|
return HandleAgentRole(request);
|
||||||
|
case "GETAGENTROLES":
|
||||||
|
return HandleGetAgentRoles(request);
|
||||||
|
case "SETACTIVE":
|
||||||
|
return HandleSetActive(request);
|
||||||
|
case "UPDATEMEMBERSHIP":
|
||||||
|
return HandleUpdateMembership(request);
|
||||||
|
case "INVITE":
|
||||||
|
return HandleInvite(request);
|
||||||
|
case "ADDNOTICE":
|
||||||
|
return HandleAddNotice(request);
|
||||||
|
case "GETNOTICES":
|
||||||
|
return HandleGetNotices(request);
|
||||||
|
}
|
||||||
|
m_log.DebugFormat("[GROUPS HANDLER]: unknown method request: {0}", method);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[GROUPS HANDLER]: Exception {0}", e.StackTrace);
|
||||||
|
}
|
||||||
|
|
||||||
|
return FailureResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleAddOrUpdateGroup(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
ExtendedGroupRecord grec = GroupsDataUtils.GroupRecord(request);
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("OP"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string RequestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
string reason = string.Empty;
|
||||||
|
string op = request["OP"].ToString();
|
||||||
|
if (op == "ADD")
|
||||||
|
{
|
||||||
|
grec.GroupID = m_GroupsService.CreateGroup(RequestingAgentID, grec.GroupName, grec.Charter, grec.ShowInList, grec.GroupPicture, grec.MembershipFee,
|
||||||
|
grec.OpenEnrollment, grec.AllowPublish, grec.MaturePublish, grec.FounderID, out reason);
|
||||||
|
|
||||||
|
}
|
||||||
|
else if (op == "UPDATE")
|
||||||
|
{
|
||||||
|
m_GroupsService.UpdateGroup(RequestingAgentID, grec.GroupID, grec.Charter, grec.ShowInList, grec.GroupPicture, grec.MembershipFee,
|
||||||
|
grec.OpenEnrollment, grec.AllowPublish, grec.MaturePublish);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
grec = m_GroupsService.GetGroupRecord(RequestingAgentID, grec.GroupID);
|
||||||
|
if (grec == null)
|
||||||
|
NullResult(result, "Internal Error");
|
||||||
|
else
|
||||||
|
result["RESULT"] = GroupsDataUtils.GroupRecord(grec);
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetGroup(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string RequestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
ExtendedGroupRecord grec = null;
|
||||||
|
if (request.ContainsKey("GroupID"))
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
grec = m_GroupsService.GetGroupRecord(RequestingAgentID, groupID);
|
||||||
|
}
|
||||||
|
else if (request.ContainsKey("Name"))
|
||||||
|
{
|
||||||
|
string name = request["Name"].ToString();
|
||||||
|
grec = m_GroupsService.GetGroupRecord(RequestingAgentID, name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (grec == null)
|
||||||
|
NullResult(result, "Group not found");
|
||||||
|
else
|
||||||
|
result["RESULT"] = GroupsDataUtils.GroupRecord(grec);
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleAddAgentToGroup(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("AgentID") ||
|
||||||
|
!request.ContainsKey("GroupID") || !request.ContainsKey("RoleID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
UUID roleID = new UUID(request["RoleID"].ToString());
|
||||||
|
string agentID = request["AgentID"].ToString();
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
string token = string.Empty;
|
||||||
|
string reason = string.Empty;
|
||||||
|
|
||||||
|
if (request.ContainsKey("AccessToken"))
|
||||||
|
token = request["AccessToken"].ToString();
|
||||||
|
|
||||||
|
if (!m_GroupsService.AddAgentToGroup(requestingAgentID, agentID, groupID, roleID, token, out reason))
|
||||||
|
NullResult(result, reason);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
GroupMembershipData membership = m_GroupsService.GetAgentGroupMembership(requestingAgentID, agentID, groupID);
|
||||||
|
if (membership == null)
|
||||||
|
NullResult(result, "Internal error");
|
||||||
|
else
|
||||||
|
result["RESULT"] = GroupsDataUtils.GroupMembershipData((ExtendedGroupMembershipData)membership);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleRemoveAgentFromGroup(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("AgentID") || !request.ContainsKey("GroupID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string agentID = request["AgentID"].ToString();
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
string reason = string.Empty;
|
||||||
|
|
||||||
|
m_GroupsService.RemoveAgentFromGroup(requestingAgentID, agentID, groupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
result["RESULT"] = "true";
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(ServerUtils.BuildXmlResponse(result));
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetMembership(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("AgentID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string agentID = request["AgentID"].ToString();
|
||||||
|
UUID groupID = UUID.Zero;
|
||||||
|
if (request.ContainsKey("GroupID"))
|
||||||
|
groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
bool all = request.ContainsKey("ALL");
|
||||||
|
|
||||||
|
if (!all)
|
||||||
|
{
|
||||||
|
ExtendedGroupMembershipData membership = null;
|
||||||
|
if (groupID == UUID.Zero)
|
||||||
|
{
|
||||||
|
membership = m_GroupsService.GetAgentActiveMembership(requestingAgentID, agentID);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
membership = m_GroupsService.GetAgentGroupMembership(requestingAgentID, agentID, groupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (membership == null)
|
||||||
|
NullResult(result, "No such membership");
|
||||||
|
else
|
||||||
|
result["RESULT"] = GroupsDataUtils.GroupMembershipData(membership);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
List<GroupMembershipData> memberships = m_GroupsService.GetAgentGroupMemberships(requestingAgentID, agentID);
|
||||||
|
if (memberships == null || (memberships != null && memberships.Count == 0))
|
||||||
|
{
|
||||||
|
NullResult(result, "No memberships");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (GroupMembershipData m in memberships)
|
||||||
|
dict["m-" + i++] = GroupsDataUtils.GroupMembershipData((ExtendedGroupMembershipData)m);
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetGroupMembers(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
|
||||||
|
List<ExtendedGroupMembersData> members = m_GroupsService.GetGroupMembers(requestingAgentID, groupID);
|
||||||
|
if (members == null || (members != null && members.Count == 0))
|
||||||
|
{
|
||||||
|
NullResult(result, "No members");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (ExtendedGroupMembersData m in members)
|
||||||
|
{
|
||||||
|
dict["m-" + i++] = GroupsDataUtils.GroupMembersData(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandlePutRole(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") || !request.ContainsKey("RoleID") ||
|
||||||
|
!request.ContainsKey("Name") || !request.ContainsKey("Description") || !request.ContainsKey("Title") ||
|
||||||
|
!request.ContainsKey("Powers") || !request.ContainsKey("OP"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string op = request["OP"].ToString();
|
||||||
|
string reason = string.Empty;
|
||||||
|
|
||||||
|
bool success = false;
|
||||||
|
if (op == "ADD")
|
||||||
|
success = m_GroupsService.AddGroupRole(request["RequestingAgentID"].ToString(), new UUID(request["GroupID"].ToString()),
|
||||||
|
new UUID(request["RoleID"].ToString()), request["Name"].ToString(), request["Description"].ToString(),
|
||||||
|
request["Title"].ToString(), UInt64.Parse(request["Powers"].ToString()), out reason);
|
||||||
|
|
||||||
|
else if (op == "UPDATE")
|
||||||
|
success = m_GroupsService.UpdateGroupRole(request["RequestingAgentID"].ToString(), new UUID(request["GroupID"].ToString()),
|
||||||
|
new UUID(request["RoleID"].ToString()), request["Name"].ToString(), request["Description"].ToString(),
|
||||||
|
request["Title"].ToString(), UInt64.Parse(request["Powers"].ToString()));
|
||||||
|
|
||||||
|
result["RESULT"] = success.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleRemoveRole(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") || !request.ContainsKey("RoleID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_GroupsService.RemoveGroupRole(request["RequestingAgentID"].ToString(), new UUID(request["GroupID"].ToString()),
|
||||||
|
new UUID(request["RoleID"].ToString()));
|
||||||
|
result["RESULT"] = "true";
|
||||||
|
}
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(ServerUtils.BuildXmlResponse(result));
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetGroupRoles(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
|
||||||
|
List<GroupRolesData> roles = m_GroupsService.GetGroupRoles(requestingAgentID, groupID);
|
||||||
|
if (roles == null || (roles != null && roles.Count == 0))
|
||||||
|
{
|
||||||
|
NullResult(result, "No members");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (GroupRolesData r in roles)
|
||||||
|
dict["r-" + i++] = GroupsDataUtils.GroupRolesData(r);
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetRoleMembers(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
|
||||||
|
List<ExtendedGroupRoleMembersData> rmembers = m_GroupsService.GetGroupRoleMembers(requestingAgentID, groupID);
|
||||||
|
if (rmembers == null || (rmembers != null && rmembers.Count == 0))
|
||||||
|
{
|
||||||
|
NullResult(result, "No members");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (ExtendedGroupRoleMembersData rm in rmembers)
|
||||||
|
dict["rm-" + i++] = GroupsDataUtils.GroupRoleMembersData(rm);
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleAgentRole(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") || !request.ContainsKey("RoleID") ||
|
||||||
|
!request.ContainsKey("AgentID") || !request.ContainsKey("OP"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string op = request["OP"].ToString();
|
||||||
|
string reason = string.Empty;
|
||||||
|
|
||||||
|
bool success = false;
|
||||||
|
if (op == "ADD")
|
||||||
|
success = m_GroupsService.AddAgentToGroupRole(request["RequestingAgentID"].ToString(), request["AgentID"].ToString(),
|
||||||
|
new UUID(request["GroupID"].ToString()), new UUID(request["RoleID"].ToString()));
|
||||||
|
|
||||||
|
else if (op == "DELETE")
|
||||||
|
success = m_GroupsService.RemoveAgentFromGroupRole(request["RequestingAgentID"].ToString(), request["AgentID"].ToString(),
|
||||||
|
new UUID(request["GroupID"].ToString()), new UUID(request["RoleID"].ToString()));
|
||||||
|
|
||||||
|
result["RESULT"] = success.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetAgentRoles(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") || !request.ContainsKey("AgentID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID groupID = new UUID(request["GroupID"].ToString());
|
||||||
|
string agentID = request["AgentID"].ToString();
|
||||||
|
string requestingAgentID = request["RequestingAgentID"].ToString();
|
||||||
|
|
||||||
|
List<GroupRolesData> roles = m_GroupsService.GetAgentGroupRoles(requestingAgentID, agentID, groupID);
|
||||||
|
if (roles == null || (roles != null && roles.Count == 0))
|
||||||
|
{
|
||||||
|
NullResult(result, "No members");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (GroupRolesData r in roles)
|
||||||
|
dict["r-" + i++] = GroupsDataUtils.GroupRolesData(r);
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleSetActive(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") ||
|
||||||
|
!request.ContainsKey("AgentID") || !request.ContainsKey("OP"))
|
||||||
|
{
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string op = request["OP"].ToString();
|
||||||
|
string reason = string.Empty;
|
||||||
|
|
||||||
|
if (op == "GROUP")
|
||||||
|
{
|
||||||
|
ExtendedGroupMembershipData group = m_GroupsService.SetAgentActiveGroup(request["RequestingAgentID"].ToString(),
|
||||||
|
request["AgentID"].ToString(), new UUID(request["GroupID"].ToString()));
|
||||||
|
|
||||||
|
if (group == null)
|
||||||
|
NullResult(result, "Internal error");
|
||||||
|
else
|
||||||
|
result["RESULT"] = GroupsDataUtils.GroupMembershipData(group);
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
|
||||||
|
}
|
||||||
|
else if (op == "ROLE" && request.ContainsKey("RoleID"))
|
||||||
|
{
|
||||||
|
m_GroupsService.SetAgentActiveGroupRole(request["RequestingAgentID"].ToString(), request["AgentID"].ToString(),
|
||||||
|
new UUID(request["GroupID"].ToString()), new UUID(request["RoleID"].ToString()));
|
||||||
|
result["RESULT"] = "true";
|
||||||
|
}
|
||||||
|
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(ServerUtils.BuildXmlResponse(result));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleUpdateMembership(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("AgentID") || !request.ContainsKey("GroupID") ||
|
||||||
|
!request.ContainsKey("AcceptNotices") || !request.ContainsKey("ListInProfile"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_GroupsService.UpdateMembership(request["RequestingAgentID"].ToString(), request["AgentID"].ToString(), new UUID(request["GroupID"].ToString()),
|
||||||
|
bool.Parse(request["AcceptNotices"].ToString()), bool.Parse(request["ListInProfile"].ToString()));
|
||||||
|
|
||||||
|
result["RESULT"] = "true";
|
||||||
|
}
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(ServerUtils.BuildXmlResponse(result));
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleInvite(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("InviteID"))
|
||||||
|
{
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string op = request["OP"].ToString();
|
||||||
|
string reason = string.Empty;
|
||||||
|
|
||||||
|
if (op == "ADD" && request.ContainsKey("GroupID") && request.ContainsKey("RoleID") && request.ContainsKey("AgentID"))
|
||||||
|
{
|
||||||
|
bool success = m_GroupsService.AddAgentToGroupInvite(request["RequestingAgentID"].ToString(),
|
||||||
|
new UUID(request["InviteID"].ToString()), new UUID(request["GroupID"].ToString()),
|
||||||
|
new UUID(request["RoleID"].ToString()), request["AgentID"].ToString());
|
||||||
|
|
||||||
|
result["RESULT"] = success.ToString();
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(ServerUtils.BuildXmlResponse(result));
|
||||||
|
|
||||||
|
}
|
||||||
|
else if (op == "DELETE")
|
||||||
|
{
|
||||||
|
m_GroupsService.RemoveAgentToGroupInvite(request["RequestingAgentID"].ToString(), new UUID(request["InviteID"].ToString()));
|
||||||
|
result["RESULT"] = "true";
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(ServerUtils.BuildXmlResponse(result));
|
||||||
|
}
|
||||||
|
else if (op == "GET")
|
||||||
|
{
|
||||||
|
GroupInviteInfo invite = m_GroupsService.GetAgentToGroupInvite(request["RequestingAgentID"].ToString(),
|
||||||
|
new UUID(request["InviteID"].ToString()));
|
||||||
|
|
||||||
|
result["RESULT"] = GroupsDataUtils.GroupInviteInfo(invite);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(ServerUtils.BuildXmlResponse(result));
|
||||||
|
}
|
||||||
|
|
||||||
|
NullResult(result, "Bad OP in request");
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(ServerUtils.BuildXmlResponse(result));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleAddNotice(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID") || !request.ContainsKey("GroupID") || !request.ContainsKey("NoticeID") ||
|
||||||
|
!request.ContainsKey("FromName") || !request.ContainsKey("Subject") || !request.ContainsKey("Message") ||
|
||||||
|
!request.ContainsKey("HasAttachment"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
|
||||||
|
bool hasAtt = bool.Parse(request["HasAttachment"].ToString());
|
||||||
|
byte attType = 0;
|
||||||
|
string attName = string.Empty;
|
||||||
|
string attOwner = string.Empty;
|
||||||
|
UUID attItem = UUID.Zero;
|
||||||
|
if (request.ContainsKey("AttachmentType"))
|
||||||
|
attType = byte.Parse(request["AttachmentType"].ToString());
|
||||||
|
if (request.ContainsKey("AttachmentName"))
|
||||||
|
attName = request["AttachmentName"].ToString();
|
||||||
|
if (request.ContainsKey("AttachmentItemID"))
|
||||||
|
attItem = new UUID(request["AttachmentItemID"].ToString());
|
||||||
|
if (request.ContainsKey("AttachmentOwnerID"))
|
||||||
|
attOwner = request["AttachmentOwnerID"].ToString();
|
||||||
|
|
||||||
|
bool success = m_GroupsService.AddGroupNotice(request["RequestingAgentID"].ToString(), new UUID(request["GroupID"].ToString()),
|
||||||
|
new UUID(request["NoticeID"].ToString()), request["FromName"].ToString(), request["Subject"].ToString(),
|
||||||
|
request["Message"].ToString(), hasAtt, attType, attName, attItem, attOwner);
|
||||||
|
|
||||||
|
result["RESULT"] = success.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGetNotices(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("RequestingAgentID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
|
||||||
|
else if (request.ContainsKey("NoticeID")) // just one
|
||||||
|
{
|
||||||
|
GroupNoticeInfo notice = m_GroupsService.GetGroupNotice(request["RequestingAgentID"].ToString(), new UUID(request["NoticeID"].ToString()));
|
||||||
|
|
||||||
|
if (notice == null)
|
||||||
|
NullResult(result, "NO such notice");
|
||||||
|
else
|
||||||
|
result["RESULT"] = GroupsDataUtils.GroupNoticeInfo(notice);
|
||||||
|
|
||||||
|
}
|
||||||
|
else if (request.ContainsKey("GroupID")) // all notices for group
|
||||||
|
{
|
||||||
|
List<ExtendedGroupNoticeData> notices = m_GroupsService.GetGroupNotices(request["RequestingAgentID"].ToString(), new UUID(request["GroupID"].ToString()));
|
||||||
|
|
||||||
|
if (notices == null || (notices != null && notices.Count == 0))
|
||||||
|
NullResult(result, "No notices");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (ExtendedGroupNoticeData n in notices)
|
||||||
|
dict["n-" + i++] = GroupsDataUtils.GroupNoticeData(n);
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
else
|
||||||
|
NullResult(result, "Bad OP in request");
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#region Helpers
|
||||||
|
|
||||||
|
private void NullResult(Dictionary<string, object> result, string reason)
|
||||||
|
{
|
||||||
|
result["RESULT"] = "NULL";
|
||||||
|
result["REASON"] = reason;
|
||||||
|
}
|
||||||
|
|
||||||
|
private byte[] FailureResult()
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
NullResult(result, "Unknown method");
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,831 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Region.Framework.Interfaces;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public delegate ExtendedGroupRecord GroupRecordDelegate();
|
||||||
|
public delegate GroupMembershipData GroupMembershipDelegate();
|
||||||
|
public delegate List<GroupMembershipData> GroupMembershipListDelegate();
|
||||||
|
public delegate List<ExtendedGroupMembersData> GroupMembersListDelegate();
|
||||||
|
public delegate List<GroupRolesData> GroupRolesListDelegate();
|
||||||
|
public delegate List<ExtendedGroupRoleMembersData> RoleMembersListDelegate();
|
||||||
|
public delegate GroupNoticeInfo NoticeDelegate();
|
||||||
|
public delegate List<ExtendedGroupNoticeData> NoticeListDelegate();
|
||||||
|
public delegate void VoidDelegate();
|
||||||
|
public delegate bool BooleanDelegate();
|
||||||
|
|
||||||
|
public class RemoteConnectorCacheWrapper
|
||||||
|
{
|
||||||
|
private ForeignImporter m_ForeignImporter;
|
||||||
|
|
||||||
|
private Dictionary<string, bool> m_ActiveRequests = new Dictionary<string, bool>();
|
||||||
|
private const int GROUPS_CACHE_TIMEOUT = 5 * 60; // 5 minutes
|
||||||
|
|
||||||
|
// This all important cache cahces objects of different types:
|
||||||
|
// group-<GroupID> or group-<Name> => ExtendedGroupRecord
|
||||||
|
// active-<AgentID> => GroupMembershipData
|
||||||
|
// membership-<AgentID>-<GroupID> => GroupMembershipData
|
||||||
|
// memberships-<AgentID> => List<GroupMembershipData>
|
||||||
|
// members-<RequestingAgentID>-<GroupID> => List<ExtendedGroupMembersData>
|
||||||
|
// role-<RoleID> => GroupRolesData
|
||||||
|
// roles-<GroupID> => List<GroupRolesData> ; all roles in the group
|
||||||
|
// roles-<GroupID>-<AgentID> => List<GroupRolesData> ; roles that the agent has
|
||||||
|
// rolemembers-<RequestingAgentID>-<GroupID> => List<ExtendedGroupRoleMembersData>
|
||||||
|
// notice-<noticeID> => GroupNoticeInfo
|
||||||
|
// notices-<GroupID> => List<ExtendedGroupNoticeData>
|
||||||
|
private ExpiringCache<string, object> m_Cache = new ExpiringCache<string, object>();
|
||||||
|
|
||||||
|
public RemoteConnectorCacheWrapper(IUserManagement uman)
|
||||||
|
{
|
||||||
|
m_ForeignImporter = new ForeignImporter(uman);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UUID CreateGroup(UUID RequestingAgentID, GroupRecordDelegate d)
|
||||||
|
{
|
||||||
|
//m_log.DebugFormat("[Groups.RemoteConnector]: Creating group {0}", name);
|
||||||
|
//reason = string.Empty;
|
||||||
|
|
||||||
|
//ExtendedGroupRecord group = m_GroupsService.CreateGroup(RequestingAgentID.ToString(), name, charter, showInList, insigniaID,
|
||||||
|
// membershipFee, openEnrollment, allowPublish, maturePublish, founderID, out reason);
|
||||||
|
ExtendedGroupRecord group = d();
|
||||||
|
|
||||||
|
if (group == null)
|
||||||
|
return UUID.Zero;
|
||||||
|
|
||||||
|
if (group.GroupID != UUID.Zero)
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.Add("group-" + group.GroupID.ToString(), group, GROUPS_CACHE_TIMEOUT);
|
||||||
|
if (m_Cache.Contains("memberships-" + RequestingAgentID.ToString()))
|
||||||
|
m_Cache.Remove("memberships-" + RequestingAgentID.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
return group.GroupID;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool UpdateGroup(UUID groupID, GroupRecordDelegate d)
|
||||||
|
{
|
||||||
|
//reason = string.Empty;
|
||||||
|
//ExtendedGroupRecord group = m_GroupsService.UpdateGroup(RequestingAgentID, groupID, charter, showInList, insigniaID, membershipFee, openEnrollment, allowPublish, maturePublish);
|
||||||
|
ExtendedGroupRecord group = d();
|
||||||
|
|
||||||
|
if (group != null && group.GroupID != UUID.Zero)
|
||||||
|
lock (m_Cache)
|
||||||
|
m_Cache.AddOrUpdate("group-" + group.GroupID.ToString(), group, GROUPS_CACHE_TIMEOUT);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupRecord GetGroupRecord(string RequestingAgentID, UUID GroupID, string GroupName, GroupRecordDelegate d)
|
||||||
|
{
|
||||||
|
//if (GroupID == UUID.Zero && (GroupName == null || GroupName != null && GroupName == string.Empty))
|
||||||
|
// return null;
|
||||||
|
|
||||||
|
object group = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
string cacheKey = "group-";
|
||||||
|
if (GroupID != UUID.Zero)
|
||||||
|
cacheKey += GroupID.ToString();
|
||||||
|
else
|
||||||
|
cacheKey += GroupName;
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: GetGroupRecord {0}", cacheKey);
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out group))
|
||||||
|
{
|
||||||
|
//m_log.DebugFormat("[XXX]: GetGroupRecord {0} cached!", cacheKey);
|
||||||
|
return (ExtendedGroupRecord)group;
|
||||||
|
}
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
//group = m_GroupsService.GetGroupRecord(RequestingAgentID, GroupID, GroupName);
|
||||||
|
group = d();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, group, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
return (ExtendedGroupRecord)group;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddAgentToGroup(string RequestingAgentID, string AgentID, UUID GroupID, GroupMembershipDelegate d)
|
||||||
|
{
|
||||||
|
GroupMembershipData membership = d();
|
||||||
|
if (membership == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
// first, remove everything! add a user is a heavy-duty op
|
||||||
|
m_Cache.Clear();
|
||||||
|
|
||||||
|
m_Cache.AddOrUpdate("active-" + AgentID.ToString(), membership, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_Cache.AddOrUpdate("membership-" + AgentID.ToString() + "-" + GroupID.ToString(), membership, GROUPS_CACHE_TIMEOUT);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroup(string RequestingAgentID, string AgentID, UUID GroupID, VoidDelegate d)
|
||||||
|
{
|
||||||
|
d();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
string cacheKey = "active-" + AgentID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
cacheKey = "memberships-" + AgentID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
cacheKey = "membership-" + AgentID.ToString() + "-" + GroupID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
cacheKey = "members-" + RequestingAgentID.ToString() + "-" + GroupID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
cacheKey = "roles-" + "-" + GroupID.ToString() + "-" + AgentID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetAgentActiveGroup(string AgentID, GroupMembershipDelegate d)
|
||||||
|
{
|
||||||
|
GroupMembershipData activeGroup = d();
|
||||||
|
if (activeGroup != null)
|
||||||
|
{
|
||||||
|
string cacheKey = "active-" + AgentID.ToString();
|
||||||
|
lock (m_Cache)
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, activeGroup, GROUPS_CACHE_TIMEOUT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupMembershipData GetAgentActiveMembership(string AgentID, GroupMembershipDelegate d)
|
||||||
|
{
|
||||||
|
object membership = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
string cacheKey = "active-" + AgentID.ToString();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: GetAgentActiveMembership {0}", cacheKey);
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out membership))
|
||||||
|
{
|
||||||
|
//m_log.DebugFormat("[XXX]: GetAgentActiveMembership {0} cached!", cacheKey);
|
||||||
|
return (ExtendedGroupMembershipData)membership;
|
||||||
|
}
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
membership = d();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, membership, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
return (ExtendedGroupMembershipData)membership;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupMembershipData GetAgentGroupMembership(string AgentID, UUID GroupID, GroupMembershipDelegate d)
|
||||||
|
{
|
||||||
|
object membership = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
string cacheKey = "membership-" + AgentID.ToString() + "-" + GroupID.ToString();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: GetAgentGroupMembership {0}", cacheKey);
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out membership))
|
||||||
|
{
|
||||||
|
//m_log.DebugFormat("[XXX]: GetAgentGroupMembership {0}", cacheKey);
|
||||||
|
return (ExtendedGroupMembershipData)membership;
|
||||||
|
}
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
membership = d();
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, membership, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
return (ExtendedGroupMembershipData)membership;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupMembershipData> GetAgentGroupMemberships(string AgentID, GroupMembershipListDelegate d)
|
||||||
|
{
|
||||||
|
object memberships = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
string cacheKey = "memberships-" + AgentID.ToString();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: GetAgentGroupMemberships {0}", cacheKey);
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out memberships))
|
||||||
|
{
|
||||||
|
//m_log.DebugFormat("[XXX]: GetAgentGroupMemberships {0} cached!", cacheKey);
|
||||||
|
return (List<GroupMembershipData>)memberships;
|
||||||
|
}
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
memberships = d();
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, memberships, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
return (List<GroupMembershipData>)memberships;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupMembersData> GetGroupMembers(string RequestingAgentID, UUID GroupID, GroupMembersListDelegate d)
|
||||||
|
{
|
||||||
|
object members = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
// we need to key in also on the requester, because different ppl have different view privileges
|
||||||
|
string cacheKey = "members-" + RequestingAgentID.ToString() + "-" + GroupID.ToString();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: GetGroupMembers {0}", cacheKey);
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out members))
|
||||||
|
{
|
||||||
|
List<ExtendedGroupMembersData> xx = (List<ExtendedGroupMembersData>)members;
|
||||||
|
return xx.ConvertAll<GroupMembersData>(new Converter<ExtendedGroupMembersData, GroupMembersData>(m_ForeignImporter.ConvertGroupMembersData));
|
||||||
|
}
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
List<ExtendedGroupMembersData> _members = d();
|
||||||
|
|
||||||
|
if (_members != null && _members.Count > 0)
|
||||||
|
members = _members.ConvertAll<GroupMembersData>(new Converter<ExtendedGroupMembersData, GroupMembersData>(m_ForeignImporter.ConvertGroupMembersData));
|
||||||
|
else
|
||||||
|
members = new List<GroupMembersData>();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
//m_Cache.AddOrUpdate(cacheKey, members, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, _members, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
|
||||||
|
return (List<GroupMembersData>)members;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupRole(UUID groupID, UUID roleID, string description, string name, ulong powers, string title, BooleanDelegate d)
|
||||||
|
{
|
||||||
|
if (d())
|
||||||
|
{
|
||||||
|
GroupRolesData role = new GroupRolesData();
|
||||||
|
role.Description = description;
|
||||||
|
role.Members = 0;
|
||||||
|
role.Name = name;
|
||||||
|
role.Powers = powers;
|
||||||
|
role.RoleID = roleID;
|
||||||
|
role.Title = title;
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate("role-" + roleID.ToString(), role, GROUPS_CACHE_TIMEOUT);
|
||||||
|
|
||||||
|
// also remove this list
|
||||||
|
if (m_Cache.Contains("roles-" + groupID.ToString()))
|
||||||
|
m_Cache.Remove("roles-" + groupID.ToString());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool UpdateGroupRole(UUID groupID, UUID roleID, string name, string description, string title, ulong powers, BooleanDelegate d)
|
||||||
|
{
|
||||||
|
if (d())
|
||||||
|
{
|
||||||
|
object role;
|
||||||
|
lock (m_Cache)
|
||||||
|
if (m_Cache.TryGetValue("role-" + roleID.ToString(), out role))
|
||||||
|
{
|
||||||
|
GroupRolesData r = (GroupRolesData)role;
|
||||||
|
r.Description = description;
|
||||||
|
r.Name = name;
|
||||||
|
r.Powers = powers;
|
||||||
|
r.Title = title;
|
||||||
|
|
||||||
|
m_Cache.Update("role-" + roleID.ToString(), r, GROUPS_CACHE_TIMEOUT);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.Contains("role-" + roleID.ToString()))
|
||||||
|
m_Cache.Remove("role-" + roleID.ToString());
|
||||||
|
|
||||||
|
// also remove these lists, because they will have an outdated role
|
||||||
|
if (m_Cache.Contains("roles-" + groupID.ToString()))
|
||||||
|
m_Cache.Remove("roles-" + groupID.ToString());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveGroupRole(string RequestingAgentID, UUID groupID, UUID roleID, VoidDelegate d)
|
||||||
|
{
|
||||||
|
d();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.Contains("role-" + roleID.ToString()))
|
||||||
|
m_Cache.Remove("role-" + roleID.ToString());
|
||||||
|
|
||||||
|
// also remove the list, because it will have an removed role
|
||||||
|
if (m_Cache.Contains("roles-" + groupID.ToString()))
|
||||||
|
m_Cache.Remove("roles-" + groupID.ToString());
|
||||||
|
|
||||||
|
if (m_Cache.Contains("roles-" + groupID.ToString() + "-" + RequestingAgentID.ToString()))
|
||||||
|
m_Cache.Remove("roles-" + groupID.ToString() + "-" + RequestingAgentID.ToString());
|
||||||
|
|
||||||
|
if (m_Cache.Contains("rolemembers-" + RequestingAgentID.ToString() + "-" + groupID.ToString()))
|
||||||
|
m_Cache.Remove("rolemembers-" + RequestingAgentID.ToString() + "-" + groupID.ToString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetGroupRoles(string RequestingAgentID, UUID GroupID, GroupRolesListDelegate d)
|
||||||
|
{
|
||||||
|
object roles = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
string cacheKey = "roles-" + GroupID.ToString();
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out roles))
|
||||||
|
return (List<GroupRolesData>)roles;
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
roles = d();
|
||||||
|
if (roles != null)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, roles, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
return (List<GroupRolesData>)roles;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRoleMembersData> GetGroupRoleMembers(string RequestingAgentID, UUID GroupID, RoleMembersListDelegate d)
|
||||||
|
{
|
||||||
|
object rmembers = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
// we need to key in also on the requester, because different ppl have different view privileges
|
||||||
|
string cacheKey = "rolemembers-" + RequestingAgentID.ToString() + "-" + GroupID.ToString();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: GetGroupRoleMembers {0}", cacheKey);
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out rmembers))
|
||||||
|
{
|
||||||
|
List<ExtendedGroupRoleMembersData> xx = (List<ExtendedGroupRoleMembersData>)rmembers;
|
||||||
|
return xx.ConvertAll<GroupRoleMembersData>(m_ForeignImporter.ConvertGroupRoleMembersData);
|
||||||
|
}
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
List<ExtendedGroupRoleMembersData> _rmembers = d();
|
||||||
|
|
||||||
|
if (_rmembers != null && _rmembers.Count > 0)
|
||||||
|
rmembers = _rmembers.ConvertAll<GroupRoleMembersData>(new Converter<ExtendedGroupRoleMembersData, GroupRoleMembersData>(m_ForeignImporter.ConvertGroupRoleMembersData));
|
||||||
|
else
|
||||||
|
rmembers = new List<GroupRoleMembersData>();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
// For some strange reason, when I cache the list of GroupRoleMembersData,
|
||||||
|
// it gets emptied out. The TryGet gets an empty list...
|
||||||
|
//m_Cache.AddOrUpdate(cacheKey, rmembers, GROUPS_CACHE_TIMEOUT);
|
||||||
|
// Caching the list of ExtendedGroupRoleMembersData doesn't show that issue
|
||||||
|
// I don't get it.
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, _rmembers, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
return (List<GroupRoleMembersData>)rmembers;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddAgentToGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID, BooleanDelegate d)
|
||||||
|
{
|
||||||
|
if (d())
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
// update the cached role
|
||||||
|
string cacheKey = "role-" + RoleID.ToString();
|
||||||
|
object obj;
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out obj))
|
||||||
|
{
|
||||||
|
GroupRolesData r = (GroupRolesData)obj;
|
||||||
|
r.Members++;
|
||||||
|
}
|
||||||
|
|
||||||
|
// add this agent to the list of role members
|
||||||
|
cacheKey = "rolemembers-" + RequestingAgentID.ToString() + "-" + GroupID.ToString();
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out obj))
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// This may throw an exception, in which case the agentID is not a UUID but a full ID
|
||||||
|
// In that case, let's just remove the whoe things from the cache
|
||||||
|
UUID id = new UUID(AgentID);
|
||||||
|
List<ExtendedGroupRoleMembersData> xx = (List<ExtendedGroupRoleMembersData>)obj;
|
||||||
|
List<GroupRoleMembersData> rmlist = xx.ConvertAll<GroupRoleMembersData>(m_ForeignImporter.ConvertGroupRoleMembersData);
|
||||||
|
GroupRoleMembersData rm = new GroupRoleMembersData();
|
||||||
|
rm.MemberID = id;
|
||||||
|
rm.RoleID = RoleID;
|
||||||
|
rmlist.Add(rm);
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove the cached info about this agent's roles
|
||||||
|
// because we don't have enough local info about the new role
|
||||||
|
cacheKey = "roles-" + GroupID.ToString() + "-" + AgentID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroupRole(string RequestingAgentID, string AgentID, UUID GroupID, UUID RoleID, BooleanDelegate d)
|
||||||
|
{
|
||||||
|
if (d())
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
// update the cached role
|
||||||
|
string cacheKey = "role-" + RoleID.ToString();
|
||||||
|
object obj;
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out obj))
|
||||||
|
{
|
||||||
|
GroupRolesData r = (GroupRolesData)obj;
|
||||||
|
r.Members--;
|
||||||
|
}
|
||||||
|
|
||||||
|
cacheKey = "roles-" + GroupID.ToString() + "-" + AgentID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
cacheKey = "rolemembers-" + RequestingAgentID.ToString() + "-" + GroupID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetAgentGroupRoles(string RequestingAgentID, string AgentID, UUID GroupID, GroupRolesListDelegate d)
|
||||||
|
{
|
||||||
|
object roles = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
string cacheKey = "roles-" + GroupID.ToString() + "-" + AgentID.ToString();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: GetAgentGroupRoles {0}", cacheKey);
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out roles))
|
||||||
|
{
|
||||||
|
//m_log.DebugFormat("[XXX]: GetAgentGroupRoles {0} cached!", cacheKey);
|
||||||
|
return (List<GroupRolesData>)roles;
|
||||||
|
}
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
roles = d();
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, roles, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
return (List<GroupRolesData>)roles;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetAgentActiveGroupRole(string AgentID, UUID GroupID, VoidDelegate d)
|
||||||
|
{
|
||||||
|
d();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
// Invalidate cached info, because it has ActiveRoleID and Powers
|
||||||
|
string cacheKey = "membership-" + AgentID.ToString() + "-" + GroupID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
cacheKey = "memberships-" + AgentID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UpdateMembership(string AgentID, UUID GroupID, bool AcceptNotices, bool ListInProfile, VoidDelegate d)
|
||||||
|
{
|
||||||
|
d();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
string cacheKey = "membership-" + AgentID.ToString() + "-" + GroupID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
cacheKey = "memberships-" + AgentID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
cacheKey = "active-" + AgentID.ToString();
|
||||||
|
object m = null;
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out m))
|
||||||
|
{
|
||||||
|
GroupMembershipData membership = (GroupMembershipData)m;
|
||||||
|
membership.ListInProfile = ListInProfile;
|
||||||
|
membership.AcceptNotices = AcceptNotices;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddGroupNotice(UUID groupID, UUID noticeID, GroupNoticeInfo notice, BooleanDelegate d)
|
||||||
|
{
|
||||||
|
if (d())
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate("notice-" + noticeID.ToString(), notice, GROUPS_CACHE_TIMEOUT);
|
||||||
|
string cacheKey = "notices-" + groupID.ToString();
|
||||||
|
if (m_Cache.Contains(cacheKey))
|
||||||
|
m_Cache.Remove(cacheKey);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupNoticeInfo GetGroupNotice(UUID noticeID, NoticeDelegate d)
|
||||||
|
{
|
||||||
|
object notice = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
string cacheKey = "notice-" + noticeID.ToString();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: GetAgentGroupRoles {0}", cacheKey);
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out notice))
|
||||||
|
{
|
||||||
|
return (GroupNoticeInfo)notice;
|
||||||
|
}
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
GroupNoticeInfo _notice = d();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, _notice, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
return _notice;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupNoticeData> GetGroupNotices(UUID GroupID, NoticeListDelegate d)
|
||||||
|
{
|
||||||
|
object notices = null;
|
||||||
|
bool firstCall = false;
|
||||||
|
string cacheKey = "notices-" + GroupID.ToString();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: GetGroupNotices {0}", cacheKey);
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
if (m_Cache.TryGetValue(cacheKey, out notices))
|
||||||
|
{
|
||||||
|
//m_log.DebugFormat("[XXX]: GetGroupNotices {0} cached!", cacheKey);
|
||||||
|
return (List<ExtendedGroupNoticeData>)notices;
|
||||||
|
}
|
||||||
|
|
||||||
|
// not cached
|
||||||
|
if (!m_ActiveRequests.ContainsKey(cacheKey))
|
||||||
|
{
|
||||||
|
m_ActiveRequests.Add(cacheKey, true);
|
||||||
|
firstCall = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
notices = d();
|
||||||
|
|
||||||
|
lock (m_Cache)
|
||||||
|
{
|
||||||
|
m_Cache.AddOrUpdate(cacheKey, notices, GROUPS_CACHE_TIMEOUT);
|
||||||
|
m_ActiveRequests.Remove(cacheKey);
|
||||||
|
return (List<ExtendedGroupNoticeData>)notices;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,84 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Reflection;
|
||||||
|
using Nini.Config;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Data;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
using OpenSim.Services.Base;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public class GroupsServiceBase : ServiceBase
|
||||||
|
{
|
||||||
|
protected IGroupsData m_Database = null;
|
||||||
|
|
||||||
|
public GroupsServiceBase(IConfigSource config, string cName)
|
||||||
|
: base(config)
|
||||||
|
{
|
||||||
|
string dllName = String.Empty;
|
||||||
|
string connString = String.Empty;
|
||||||
|
string realm = "os_groups";
|
||||||
|
string configName = (cName == string.Empty) ? "Groups" : cName;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Try reading the [DatabaseService] section, if it exists
|
||||||
|
//
|
||||||
|
IConfig dbConfig = config.Configs["DatabaseService"];
|
||||||
|
if (dbConfig != null)
|
||||||
|
{
|
||||||
|
if (dllName == String.Empty)
|
||||||
|
dllName = dbConfig.GetString("StorageProvider", String.Empty);
|
||||||
|
if (connString == String.Empty)
|
||||||
|
connString = dbConfig.GetString("ConnectionString", String.Empty);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// [Groups] section overrides [DatabaseService], if it exists
|
||||||
|
//
|
||||||
|
IConfig groupsConfig = config.Configs[configName];
|
||||||
|
if (groupsConfig != null)
|
||||||
|
{
|
||||||
|
dllName = groupsConfig.GetString("StorageProvider", dllName);
|
||||||
|
connString = groupsConfig.GetString("ConnectionString", connString);
|
||||||
|
realm = groupsConfig.GetString("Realm", realm);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// We tried, but this doesn't exist. We can't proceed.
|
||||||
|
//
|
||||||
|
if (dllName.Equals(String.Empty))
|
||||||
|
throw new Exception("No StorageProvider configured");
|
||||||
|
|
||||||
|
m_Database = LoadPlugin<IGroupsData>(dllName, new Object[] { connString, realm });
|
||||||
|
if (m_Database == null)
|
||||||
|
throw new Exception("Could not find a storage interface in the given module " + dllName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,353 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Timers;
|
||||||
|
using log4net;
|
||||||
|
using Nini.Config;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using OpenSim.Data;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
|
||||||
|
namespace OpenSim.Groups
|
||||||
|
{
|
||||||
|
public class HGGroupsService : GroupsService
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private IOfflineIMService m_OfflineIM;
|
||||||
|
private IUserAccountService m_UserAccounts;
|
||||||
|
private string m_HomeURI;
|
||||||
|
|
||||||
|
public HGGroupsService(IConfigSource config, IOfflineIMService im, IUserAccountService users, string homeURI)
|
||||||
|
: base(config, string.Empty)
|
||||||
|
{
|
||||||
|
m_OfflineIM = im;
|
||||||
|
m_UserAccounts = users;
|
||||||
|
m_HomeURI = homeURI;
|
||||||
|
if (!m_HomeURI.EndsWith("/"))
|
||||||
|
m_HomeURI += "/";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#region HG specific operations
|
||||||
|
|
||||||
|
public bool CreateGroupProxy(string RequestingAgentID, string agentID, string accessToken, UUID groupID, string serviceLocation, string name, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
Uri uri = null;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
uri = new Uri(serviceLocation);
|
||||||
|
}
|
||||||
|
catch (UriFormatException)
|
||||||
|
{
|
||||||
|
reason = "Bad location for group proxy";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if it already exists
|
||||||
|
GroupData grec = m_Database.RetrieveGroup(groupID);
|
||||||
|
if (grec == null ||
|
||||||
|
(grec != null && grec.Data["Location"] != string.Empty && grec.Data["Location"].ToLower() != serviceLocation.ToLower()))
|
||||||
|
{
|
||||||
|
// Create the group
|
||||||
|
grec = new GroupData();
|
||||||
|
grec.GroupID = groupID;
|
||||||
|
grec.Data = new Dictionary<string, string>();
|
||||||
|
grec.Data["Name"] = name + " @ " + uri.Authority;
|
||||||
|
grec.Data["Location"] = serviceLocation;
|
||||||
|
grec.Data["Charter"] = string.Empty;
|
||||||
|
grec.Data["InsigniaID"] = UUID.Zero.ToString();
|
||||||
|
grec.Data["FounderID"] = UUID.Zero.ToString();
|
||||||
|
grec.Data["MembershipFee"] = "0";
|
||||||
|
grec.Data["OpenEnrollment"] = "0";
|
||||||
|
grec.Data["ShowInList"] = "0";
|
||||||
|
grec.Data["AllowPublish"] = "0";
|
||||||
|
grec.Data["MaturePublish"] = "0";
|
||||||
|
grec.Data["OwnerRoleID"] = UUID.Zero.ToString();
|
||||||
|
|
||||||
|
|
||||||
|
if (!m_Database.StoreGroup(grec))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (grec.Data["Location"] == string.Empty)
|
||||||
|
{
|
||||||
|
reason = "Cannot add proxy membership to non-proxy group";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
UUID uid = UUID.Zero;
|
||||||
|
string url = string.Empty, first = string.Empty, last = string.Empty, tmp = string.Empty;
|
||||||
|
Util.ParseUniversalUserIdentifier(RequestingAgentID, out uid, out url, out first, out last, out tmp);
|
||||||
|
string fromName = first + "." + last + "@" + url;
|
||||||
|
|
||||||
|
// Invite to group again
|
||||||
|
InviteToGroup(fromName, groupID, new UUID(agentID), grec.Data["Name"]);
|
||||||
|
|
||||||
|
// Stick the proxy membership in the DB already
|
||||||
|
// we'll delete it if the agent declines the invitation
|
||||||
|
MembershipData membership = new MembershipData();
|
||||||
|
membership.PrincipalID = agentID;
|
||||||
|
membership.GroupID = groupID;
|
||||||
|
membership.Data = new Dictionary<string, string>();
|
||||||
|
membership.Data["SelectedRoleID"] = UUID.Zero.ToString();
|
||||||
|
membership.Data["Contribution"] = "0";
|
||||||
|
membership.Data["ListInProfile"] = "1";
|
||||||
|
membership.Data["AcceptNotices"] = "1";
|
||||||
|
membership.Data["AccessToken"] = accessToken;
|
||||||
|
|
||||||
|
m_Database.StoreMember(membership);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveAgentFromGroup(string RequestingAgentID, string AgentID, UUID GroupID, string token)
|
||||||
|
{
|
||||||
|
// check the token
|
||||||
|
MembershipData membership = m_Database.RetrieveMember(GroupID, AgentID);
|
||||||
|
if (membership != null)
|
||||||
|
{
|
||||||
|
if (token != string.Empty && token.Equals(membership.Data["AccessToken"]))
|
||||||
|
RemoveAgentFromGroup(RequestingAgentID, AgentID, GroupID);
|
||||||
|
else
|
||||||
|
m_log.DebugFormat("[Groups.HGGroupsService]: access token {0} did not match stored one {1}", token, membership.Data["AccessToken"]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
m_log.DebugFormat("[Groups.HGGroupsService]: membership not found for {0}", AgentID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExtendedGroupRecord GetGroupRecord(string RequestingAgentID, UUID GroupID, string groupName, string token)
|
||||||
|
{
|
||||||
|
// check the token
|
||||||
|
if (!VerifyToken(GroupID, RequestingAgentID, token))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
ExtendedGroupRecord grec;
|
||||||
|
if (GroupID == UUID.Zero)
|
||||||
|
grec = GetGroupRecord(RequestingAgentID, groupName);
|
||||||
|
else
|
||||||
|
grec = GetGroupRecord(RequestingAgentID, GroupID);
|
||||||
|
|
||||||
|
if (grec != null)
|
||||||
|
FillFounderUUI(grec);
|
||||||
|
|
||||||
|
return grec;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupMembersData> GetGroupMembers(string RequestingAgentID, UUID GroupID, string token)
|
||||||
|
{
|
||||||
|
if (!VerifyToken(GroupID, RequestingAgentID, token))
|
||||||
|
return new List<ExtendedGroupMembersData>();
|
||||||
|
|
||||||
|
List<ExtendedGroupMembersData> members = GetGroupMembers(RequestingAgentID, GroupID);
|
||||||
|
|
||||||
|
// convert UUIDs to UUIs
|
||||||
|
members.ForEach(delegate (ExtendedGroupMembersData m)
|
||||||
|
{
|
||||||
|
if (m.AgentID.ToString().Length == 36) // UUID
|
||||||
|
{
|
||||||
|
UserAccount account = m_UserAccounts.GetUserAccount(UUID.Zero, new UUID(m.AgentID));
|
||||||
|
if (account != null)
|
||||||
|
m.AgentID = Util.UniversalIdentifier(account.PrincipalID, account.FirstName, account.LastName, m_HomeURI);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return members;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GroupRolesData> GetGroupRoles(string RequestingAgentID, UUID GroupID, string token)
|
||||||
|
{
|
||||||
|
if (!VerifyToken(GroupID, RequestingAgentID, token))
|
||||||
|
return new List<GroupRolesData>();
|
||||||
|
|
||||||
|
return GetGroupRoles(RequestingAgentID, GroupID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ExtendedGroupRoleMembersData> GetGroupRoleMembers(string RequestingAgentID, UUID GroupID, string token)
|
||||||
|
{
|
||||||
|
if (!VerifyToken(GroupID, RequestingAgentID, token))
|
||||||
|
return new List<ExtendedGroupRoleMembersData>();
|
||||||
|
|
||||||
|
List<ExtendedGroupRoleMembersData> rolemembers = GetGroupRoleMembers(RequestingAgentID, GroupID);
|
||||||
|
|
||||||
|
// convert UUIDs to UUIs
|
||||||
|
rolemembers.ForEach(delegate(ExtendedGroupRoleMembersData m)
|
||||||
|
{
|
||||||
|
if (m.MemberID.ToString().Length == 36) // UUID
|
||||||
|
{
|
||||||
|
UserAccount account = m_UserAccounts.GetUserAccount(UUID.Zero, new UUID(m.MemberID));
|
||||||
|
if (account != null)
|
||||||
|
m.MemberID = Util.UniversalIdentifier(account.PrincipalID, account.FirstName, account.LastName, m_HomeURI);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return rolemembers;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddNotice(string RequestingAgentID, UUID groupID, UUID noticeID, string fromName, string subject, string message,
|
||||||
|
bool hasAttachment, byte attType, string attName, UUID attItemID, string attOwnerID)
|
||||||
|
{
|
||||||
|
// check that the group proxy exists
|
||||||
|
ExtendedGroupRecord grec = GetGroupRecord(RequestingAgentID, groupID);
|
||||||
|
if (grec == null)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[Groups.HGGroupsService]: attempt at adding notice to non-existent group proxy");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// check that the group is remote
|
||||||
|
if (grec.ServiceLocation == string.Empty)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[Groups.HGGroupsService]: attempt at adding notice to local (non-proxy) group");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// check that there isn't already a notice with the same ID
|
||||||
|
if (GetGroupNotice(RequestingAgentID, noticeID) != null)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[Groups.HGGroupsService]: a notice with the same ID already exists", grec.ServiceLocation);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// This has good intentions (security) but it will potentially DDS the origin...
|
||||||
|
// We'll need to send a proof along with the message. Maybe encrypt the message
|
||||||
|
// using key pairs
|
||||||
|
//
|
||||||
|
//// check that the notice actually exists in the origin
|
||||||
|
//GroupsServiceHGConnector c = new GroupsServiceHGConnector(grec.ServiceLocation);
|
||||||
|
//if (!c.VerifyNotice(noticeID, groupID))
|
||||||
|
//{
|
||||||
|
// m_log.DebugFormat("[Groups.HGGroupsService]: notice does not exist at origin {0}", grec.ServiceLocation);
|
||||||
|
// return false;
|
||||||
|
//}
|
||||||
|
|
||||||
|
// ok, we're good!
|
||||||
|
return _AddNotice(groupID, noticeID, fromName, subject, message, hasAttachment, attType, attName, attItemID, attOwnerID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool VerifyNotice(UUID noticeID, UUID groupID)
|
||||||
|
{
|
||||||
|
GroupNoticeInfo notice = GetGroupNotice(string.Empty, noticeID);
|
||||||
|
|
||||||
|
if (notice == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (notice.GroupID != groupID)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
private void InviteToGroup(string fromName, UUID groupID, UUID invitedAgentID, string groupName)
|
||||||
|
{
|
||||||
|
// Todo: Security check, probably also want to send some kind of notification
|
||||||
|
UUID InviteID = UUID.Random();
|
||||||
|
|
||||||
|
if (AddAgentToGroupInvite(InviteID, groupID, invitedAgentID.ToString()))
|
||||||
|
{
|
||||||
|
Guid inviteUUID = InviteID.Guid;
|
||||||
|
|
||||||
|
GridInstantMessage msg = new GridInstantMessage();
|
||||||
|
|
||||||
|
msg.imSessionID = inviteUUID;
|
||||||
|
|
||||||
|
// msg.fromAgentID = agentID.Guid;
|
||||||
|
msg.fromAgentID = groupID.Guid;
|
||||||
|
msg.toAgentID = invitedAgentID.Guid;
|
||||||
|
//msg.timestamp = (uint)Util.UnixTimeSinceEpoch();
|
||||||
|
msg.timestamp = 0;
|
||||||
|
msg.fromAgentName = fromName;
|
||||||
|
msg.message = string.Format("Please confirm your acceptance to join group {0}.", groupName);
|
||||||
|
msg.dialog = (byte)OpenMetaverse.InstantMessageDialog.GroupInvitation;
|
||||||
|
msg.fromGroup = true;
|
||||||
|
msg.offline = (byte)0;
|
||||||
|
msg.ParentEstateID = 0;
|
||||||
|
msg.Position = Vector3.Zero;
|
||||||
|
msg.RegionID = UUID.Zero.Guid;
|
||||||
|
msg.binaryBucket = new byte[20];
|
||||||
|
|
||||||
|
string reason = string.Empty;
|
||||||
|
m_OfflineIM.StoreMessage(msg, out reason);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool AddAgentToGroupInvite(UUID inviteID, UUID groupID, string agentID)
|
||||||
|
{
|
||||||
|
// Check whether the invitee is already a member of the group
|
||||||
|
MembershipData m = m_Database.RetrieveMember(groupID, agentID);
|
||||||
|
if (m != null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Check whether there are pending invitations and delete them
|
||||||
|
InvitationData invite = m_Database.RetrieveInvitation(groupID, agentID);
|
||||||
|
if (invite != null)
|
||||||
|
m_Database.DeleteInvite(invite.InviteID);
|
||||||
|
|
||||||
|
invite = new InvitationData();
|
||||||
|
invite.InviteID = inviteID;
|
||||||
|
invite.PrincipalID = agentID;
|
||||||
|
invite.GroupID = groupID;
|
||||||
|
invite.RoleID = UUID.Zero;
|
||||||
|
invite.Data = new Dictionary<string, string>();
|
||||||
|
|
||||||
|
return m_Database.StoreInvitation(invite);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void FillFounderUUI(ExtendedGroupRecord grec)
|
||||||
|
{
|
||||||
|
UserAccount account = m_UserAccounts.GetUserAccount(UUID.Zero, grec.FounderID);
|
||||||
|
if (account != null)
|
||||||
|
grec.FounderUUI = Util.UniversalIdentifier(account.PrincipalID, account.FirstName, account.LastName, m_HomeURI);
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool VerifyToken(UUID groupID, string agentID, string token)
|
||||||
|
{
|
||||||
|
// check the token
|
||||||
|
MembershipData membership = m_Database.RetrieveMember(groupID, agentID);
|
||||||
|
if (membership != null)
|
||||||
|
{
|
||||||
|
if (token != string.Empty && token.Equals(membership.Data["AccessToken"]))
|
||||||
|
return true;
|
||||||
|
else
|
||||||
|
m_log.DebugFormat("[Groups.HGGroupsService]: access token {0} did not match stored one {1}", token, membership.Data["AccessToken"]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
m_log.DebugFormat("[Groups.HGGroupsService]: membership not found for {0}", agentID);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,267 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Reflection;
|
||||||
|
using log4net;
|
||||||
|
using Mono.Addins;
|
||||||
|
using Nini.Config;
|
||||||
|
using OpenMetaverse;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Framework.Servers;
|
||||||
|
using OpenSim.Framework.Client;
|
||||||
|
using OpenSim.Region.Framework.Interfaces;
|
||||||
|
using OpenSim.Region.Framework.Scenes;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
|
||||||
|
namespace OpenSim.OfflineIM
|
||||||
|
{
|
||||||
|
[Extension(Path = "/OpenSim/RegionModules", NodeName = "RegionModule", Id = "OfflineIMConnectorModule")]
|
||||||
|
public class OfflineIMRegionModule : ISharedRegionModule, IOfflineIMService
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private bool m_Enabled = false;
|
||||||
|
private List<Scene> m_SceneList = new List<Scene>();
|
||||||
|
IMessageTransferModule m_TransferModule = null;
|
||||||
|
private bool m_ForwardOfflineGroupMessages = true;
|
||||||
|
|
||||||
|
private IOfflineIMService m_OfflineIMService;
|
||||||
|
|
||||||
|
public void Initialise(IConfigSource config)
|
||||||
|
{
|
||||||
|
IConfig cnf = config.Configs["Messaging"];
|
||||||
|
if (cnf == null)
|
||||||
|
return;
|
||||||
|
if (cnf != null && cnf.GetString("OfflineMessageModule", string.Empty) != Name)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_Enabled = true;
|
||||||
|
|
||||||
|
string serviceLocation = cnf.GetString("OfflineMessageURL", string.Empty);
|
||||||
|
if (serviceLocation == string.Empty)
|
||||||
|
m_OfflineIMService = new OfflineIMService(config);
|
||||||
|
else
|
||||||
|
m_OfflineIMService = new OfflineIMServiceRemoteConnector(serviceLocation);
|
||||||
|
|
||||||
|
m_ForwardOfflineGroupMessages = cnf.GetBoolean("ForwardOfflineGroupMessages", m_ForwardOfflineGroupMessages);
|
||||||
|
m_log.DebugFormat("[OfflineIM.V2]: Offline messages enabled by {0}", Name);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
scene.RegisterModuleInterface<IOfflineIMService>(this);
|
||||||
|
m_SceneList.Add(scene);
|
||||||
|
scene.EventManager.OnNewClient += OnNewClient;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RegionLoaded(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (m_TransferModule == null)
|
||||||
|
{
|
||||||
|
m_TransferModule = scene.RequestModuleInterface<IMessageTransferModule>();
|
||||||
|
if (m_TransferModule == null)
|
||||||
|
{
|
||||||
|
scene.EventManager.OnNewClient -= OnNewClient;
|
||||||
|
|
||||||
|
m_SceneList.Clear();
|
||||||
|
|
||||||
|
m_log.Error("[OfflineIM.V2]: No message transfer module is enabled. Disabling offline messages");
|
||||||
|
}
|
||||||
|
m_TransferModule.OnUndeliveredMessage += UndeliveredMessage;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveRegion(Scene scene)
|
||||||
|
{
|
||||||
|
if (!m_Enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_SceneList.Remove(scene);
|
||||||
|
scene.EventManager.OnNewClient -= OnNewClient;
|
||||||
|
m_TransferModule.OnUndeliveredMessage -= UndeliveredMessage;
|
||||||
|
|
||||||
|
scene.ForEachClient(delegate(IClientAPI client)
|
||||||
|
{
|
||||||
|
client.OnRetrieveInstantMessages -= RetrieveInstantMessages;
|
||||||
|
client.OnMuteListRequest -= OnMuteListRequest;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void PostInitialise()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Name
|
||||||
|
{
|
||||||
|
get { return "Offline Message Module V2"; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type ReplaceableInterface
|
||||||
|
{
|
||||||
|
get { return null; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Close()
|
||||||
|
{
|
||||||
|
m_SceneList.Clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
private Scene FindScene(UUID agentID)
|
||||||
|
{
|
||||||
|
foreach (Scene s in m_SceneList)
|
||||||
|
{
|
||||||
|
ScenePresence presence = s.GetScenePresence(agentID);
|
||||||
|
if (presence != null && !presence.IsChildAgent)
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private IClientAPI FindClient(UUID agentID)
|
||||||
|
{
|
||||||
|
foreach (Scene s in m_SceneList)
|
||||||
|
{
|
||||||
|
ScenePresence presence = s.GetScenePresence(agentID);
|
||||||
|
if (presence != null && !presence.IsChildAgent)
|
||||||
|
return presence.ControllingClient;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnNewClient(IClientAPI client)
|
||||||
|
{
|
||||||
|
client.OnRetrieveInstantMessages += RetrieveInstantMessages;
|
||||||
|
client.OnMuteListRequest += OnMuteListRequest;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void RetrieveInstantMessages(IClientAPI client)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[OfflineIM.V2]: Retrieving stored messages for {0}", client.AgentId);
|
||||||
|
|
||||||
|
List<GridInstantMessage> msglist = m_OfflineIMService.GetMessages(client.AgentId);
|
||||||
|
|
||||||
|
if (msglist == null)
|
||||||
|
m_log.DebugFormat("[OfflineIM.V2]: WARNING null message list.");
|
||||||
|
|
||||||
|
foreach (GridInstantMessage im in msglist)
|
||||||
|
{
|
||||||
|
if (im.dialog == (byte)InstantMessageDialog.InventoryOffered)
|
||||||
|
// send it directly or else the item will be given twice
|
||||||
|
client.SendInstantMessage(im);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Send through scene event manager so all modules get a chance
|
||||||
|
// to look at this message before it gets delivered.
|
||||||
|
//
|
||||||
|
// Needed for proper state management for stored group
|
||||||
|
// invitations
|
||||||
|
//
|
||||||
|
Scene s = FindScene(client.AgentId);
|
||||||
|
if (s != null)
|
||||||
|
s.EventManager.TriggerIncomingInstantMessage(im);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apparently this is needed in order for the viewer to request the IMs.
|
||||||
|
private void OnMuteListRequest(IClientAPI client, uint crc)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[OfflineIM.V2] Got mute list request for crc {0}", crc);
|
||||||
|
string filename = "mutes" + client.AgentId.ToString();
|
||||||
|
|
||||||
|
IXfer xfer = client.Scene.RequestModuleInterface<IXfer>();
|
||||||
|
if (xfer != null)
|
||||||
|
{
|
||||||
|
xfer.AddNewFile(filename, new Byte[0]);
|
||||||
|
client.SendMuteListUpdate(filename);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void UndeliveredMessage(GridInstantMessage im)
|
||||||
|
{
|
||||||
|
if (im.dialog != (byte)InstantMessageDialog.MessageFromObject &&
|
||||||
|
im.dialog != (byte)InstantMessageDialog.MessageFromAgent &&
|
||||||
|
im.dialog != (byte)InstantMessageDialog.GroupNotice &&
|
||||||
|
im.dialog != (byte)InstantMessageDialog.GroupInvitation &&
|
||||||
|
im.dialog != (byte)InstantMessageDialog.InventoryOffered)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!m_ForwardOfflineGroupMessages)
|
||||||
|
{
|
||||||
|
if (im.dialog == (byte)InstantMessageDialog.GroupNotice ||
|
||||||
|
im.dialog == (byte)InstantMessageDialog.GroupInvitation)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Scene scene = FindScene(new UUID(im.fromAgentID));
|
||||||
|
if (scene == null)
|
||||||
|
scene = m_SceneList[0];
|
||||||
|
|
||||||
|
string reason = string.Empty;
|
||||||
|
bool success = m_OfflineIMService.StoreMessage(im, out reason);
|
||||||
|
|
||||||
|
if (im.dialog == (byte)InstantMessageDialog.MessageFromAgent)
|
||||||
|
{
|
||||||
|
IClientAPI client = FindClient(new UUID(im.fromAgentID));
|
||||||
|
if (client == null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
client.SendInstantMessage(new GridInstantMessage(
|
||||||
|
null, new UUID(im.toAgentID),
|
||||||
|
"System", new UUID(im.fromAgentID),
|
||||||
|
(byte)InstantMessageDialog.MessageFromAgent,
|
||||||
|
"User is not logged in. " +
|
||||||
|
(success ? "Message saved." : "Message not saved: " + reason),
|
||||||
|
false, new Vector3()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#region IOfflineIM
|
||||||
|
|
||||||
|
public List<GridInstantMessage> GetMessages(UUID principalID)
|
||||||
|
{
|
||||||
|
return m_OfflineIMService.GetMessages(principalID);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool StoreMessage(GridInstantMessage im, out string reason)
|
||||||
|
{
|
||||||
|
return m_OfflineIMService.StoreMessage(im, out reason);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,36 @@
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
using Mono.Addins;
|
||||||
|
|
||||||
|
// General Information about an assembly is controlled through the following
|
||||||
|
// set of attributes. Change these attribute values to modify the information
|
||||||
|
// associated with an assembly.
|
||||||
|
[assembly: AssemblyTitle("OpenSim.Addons.OfflineIM")]
|
||||||
|
[assembly: AssemblyDescription("")]
|
||||||
|
[assembly: AssemblyConfiguration("")]
|
||||||
|
[assembly: AssemblyCompany("http://opensimulator.org")]
|
||||||
|
[assembly: AssemblyProduct("OpenSim.Addons.OfflineIM")]
|
||||||
|
[assembly: AssemblyCopyright("Copyright (c) OpenSimulator.org Developers")]
|
||||||
|
[assembly: AssemblyTrademark("")]
|
||||||
|
[assembly: AssemblyCulture("")]
|
||||||
|
|
||||||
|
// Setting ComVisible to false makes the types in this assembly not visible
|
||||||
|
// to COM components. If you need to access a type in this assembly from
|
||||||
|
// COM, set the ComVisible attribute to true on that type.
|
||||||
|
[assembly: ComVisible(false)]
|
||||||
|
|
||||||
|
// The following GUID is for the ID of the typelib if this project is exposed to COM
|
||||||
|
[assembly: Guid("a16a9905-4393-4872-9fca-4c81bedbd9f2")]
|
||||||
|
|
||||||
|
// Version information for an assembly consists of the following four values:
|
||||||
|
//
|
||||||
|
// Major Version
|
||||||
|
// Minor Version
|
||||||
|
// Build Number
|
||||||
|
// Revision
|
||||||
|
//
|
||||||
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
|
|
||||||
|
[assembly: Addin("OpenSim.OfflineIM", "0.1")]
|
||||||
|
[assembly: AddinDependency("OpenSim", "0.5")]
|
|
@ -0,0 +1,143 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Server.Base;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using log4net;
|
||||||
|
using Nini.Config;
|
||||||
|
|
||||||
|
namespace OpenSim.OfflineIM
|
||||||
|
{
|
||||||
|
public class OfflineIMServiceRemoteConnector : IOfflineIMService
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private string m_ServerURI = string.Empty;
|
||||||
|
private object m_Lock = new object();
|
||||||
|
|
||||||
|
public OfflineIMServiceRemoteConnector(string url)
|
||||||
|
{
|
||||||
|
m_ServerURI = url;
|
||||||
|
m_log.DebugFormat("[OfflineIM.V2.RemoteConnector]: Offline IM server at {0}", m_ServerURI);
|
||||||
|
}
|
||||||
|
|
||||||
|
public OfflineIMServiceRemoteConnector(IConfigSource config)
|
||||||
|
{
|
||||||
|
IConfig cnf = config.Configs["Messaging"];
|
||||||
|
if (cnf == null)
|
||||||
|
{
|
||||||
|
m_log.WarnFormat("[OfflineIM.V2.RemoteConnector]: Missing Messaging configuration");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_ServerURI = cnf.GetString("OfflineMessageURL", string.Empty);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#region IOfflineIMService
|
||||||
|
public List<GridInstantMessage> GetMessages(UUID principalID)
|
||||||
|
{
|
||||||
|
List<GridInstantMessage> ims = new List<GridInstantMessage>();
|
||||||
|
|
||||||
|
Dictionary<string, object> sendData = new Dictionary<string, object>();
|
||||||
|
sendData["PrincipalID"] = principalID;
|
||||||
|
Dictionary<string, object> ret = MakeRequest("GET", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
return ims;
|
||||||
|
|
||||||
|
if (!ret.ContainsKey("RESULT"))
|
||||||
|
return ims;
|
||||||
|
|
||||||
|
if (ret["RESULT"].ToString() == "NULL")
|
||||||
|
return ims;
|
||||||
|
|
||||||
|
foreach (object v in ((Dictionary<string, object>)ret["RESULT"]).Values)
|
||||||
|
{
|
||||||
|
GridInstantMessage m = OfflineIMDataUtils.GridInstantMessage((Dictionary<string, object>)v);
|
||||||
|
ims.Add(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ims;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool StoreMessage(GridInstantMessage im, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
Dictionary<string, object> sendData = OfflineIMDataUtils.GridInstantMessage(im);
|
||||||
|
|
||||||
|
Dictionary<string, object> ret = MakeRequest("STORE", sendData);
|
||||||
|
|
||||||
|
if (ret == null)
|
||||||
|
{
|
||||||
|
reason = "Bad response from server";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
string result = ret["RESULT"].ToString();
|
||||||
|
if (result == "NULL" || result.ToLower() == "false")
|
||||||
|
{
|
||||||
|
reason = ret["REASON"].ToString();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
|
||||||
|
#region Make Request
|
||||||
|
|
||||||
|
private Dictionary<string, object> MakeRequest(string method, Dictionary<string, object> sendData)
|
||||||
|
{
|
||||||
|
sendData["METHOD"] = method;
|
||||||
|
|
||||||
|
string reply = string.Empty;
|
||||||
|
lock (m_Lock)
|
||||||
|
reply = SynchronousRestFormsRequester.MakeRequest("POST",
|
||||||
|
m_ServerURI + "/offlineim",
|
||||||
|
ServerUtils.BuildQueryString(sendData));
|
||||||
|
|
||||||
|
Dictionary<string, object> replyData = ServerUtils.ParseXmlResponse(
|
||||||
|
reply);
|
||||||
|
|
||||||
|
return replyData;
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,215 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Text;
|
||||||
|
using System.Xml;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using Nini.Config;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Server.Base;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
using OpenSim.Framework.Servers.HttpServer;
|
||||||
|
using OpenSim.Server.Handlers.Base;
|
||||||
|
using log4net;
|
||||||
|
using OpenMetaverse;
|
||||||
|
|
||||||
|
namespace OpenSim.OfflineIM
|
||||||
|
{
|
||||||
|
public class OfflineIMServiceRobustConnector : ServiceConnector
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private IOfflineIMService m_OfflineIMService;
|
||||||
|
private string m_ConfigName = "Messaging";
|
||||||
|
|
||||||
|
public OfflineIMServiceRobustConnector(IConfigSource config, IHttpServer server, string configName) :
|
||||||
|
base(config, server, configName)
|
||||||
|
{
|
||||||
|
if (configName != String.Empty)
|
||||||
|
m_ConfigName = configName;
|
||||||
|
|
||||||
|
m_log.DebugFormat("[OfflineIM.V2.RobustConnector]: Starting with config name {0}", m_ConfigName);
|
||||||
|
|
||||||
|
m_OfflineIMService = new OfflineIMService(config);
|
||||||
|
|
||||||
|
server.AddStreamHandler(new OfflineIMServicePostHandler(m_OfflineIMService));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class OfflineIMServicePostHandler : BaseStreamHandler
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
|
private IOfflineIMService m_OfflineIMService;
|
||||||
|
|
||||||
|
public OfflineIMServicePostHandler(IOfflineIMService service) :
|
||||||
|
base("POST", "/offlineim")
|
||||||
|
{
|
||||||
|
m_OfflineIMService = service;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override byte[] Handle(string path, Stream requestData,
|
||||||
|
IOSHttpRequest httpRequest, IOSHttpResponse httpResponse)
|
||||||
|
{
|
||||||
|
StreamReader sr = new StreamReader(requestData);
|
||||||
|
string body = sr.ReadToEnd();
|
||||||
|
sr.Close();
|
||||||
|
body = body.Trim();
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: query String: {0}", body);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Dictionary<string, object> request =
|
||||||
|
ServerUtils.ParseQueryString(body);
|
||||||
|
|
||||||
|
if (!request.ContainsKey("METHOD"))
|
||||||
|
return FailureResult();
|
||||||
|
|
||||||
|
string method = request["METHOD"].ToString();
|
||||||
|
request.Remove("METHOD");
|
||||||
|
|
||||||
|
m_log.DebugFormat("[OfflineIM.V2.Handler]: {0}", method);
|
||||||
|
switch (method)
|
||||||
|
{
|
||||||
|
case "GET":
|
||||||
|
return HandleGet(request);
|
||||||
|
case "STORE":
|
||||||
|
return HandleStore(request);
|
||||||
|
}
|
||||||
|
m_log.DebugFormat("[OFFLINE IM HANDLER]: unknown method request: {0}", method);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
m_log.DebugFormat("[OFFLINE IM HANDLER]: Exception {0}", e.StackTrace);
|
||||||
|
}
|
||||||
|
|
||||||
|
return FailureResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleStore(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
GridInstantMessage im = OfflineIMDataUtils.GridInstantMessage(request);
|
||||||
|
|
||||||
|
string reason = string.Empty;
|
||||||
|
|
||||||
|
bool success = m_OfflineIMService.StoreMessage(im, out reason);
|
||||||
|
|
||||||
|
result["RESULT"] = success.ToString();
|
||||||
|
if (!success)
|
||||||
|
result["REASON"] = reason;
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] HandleGet(Dictionary<string, object> request)
|
||||||
|
{
|
||||||
|
Dictionary<string, object> result = new Dictionary<string, object>();
|
||||||
|
|
||||||
|
if (!request.ContainsKey("PrincipalID"))
|
||||||
|
NullResult(result, "Bad network data");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UUID principalID = new UUID(request["PrincipalID"].ToString());
|
||||||
|
List<GridInstantMessage> ims = m_OfflineIMService.GetMessages(principalID);
|
||||||
|
|
||||||
|
Dictionary<string, object> dict = new Dictionary<string, object>();
|
||||||
|
int i = 0;
|
||||||
|
foreach (GridInstantMessage m in ims)
|
||||||
|
dict["im-" + i++] = OfflineIMDataUtils.GridInstantMessage(m);
|
||||||
|
|
||||||
|
result["RESULT"] = dict;
|
||||||
|
}
|
||||||
|
|
||||||
|
string xmlString = ServerUtils.BuildXmlResponse(result);
|
||||||
|
|
||||||
|
//m_log.DebugFormat("[XXX]: resp string: {0}", xmlString);
|
||||||
|
return Util.UTF8NoBomEncoding.GetBytes(xmlString);
|
||||||
|
}
|
||||||
|
|
||||||
|
#region Helpers
|
||||||
|
|
||||||
|
private void NullResult(Dictionary<string, object> result, string reason)
|
||||||
|
{
|
||||||
|
result["RESULT"] = "NULL";
|
||||||
|
result["REASON"] = reason;
|
||||||
|
}
|
||||||
|
|
||||||
|
private byte[] FailureResult()
|
||||||
|
{
|
||||||
|
return BoolResult(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private byte[] SuccessResult()
|
||||||
|
{
|
||||||
|
return BoolResult(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private byte[] BoolResult(bool value)
|
||||||
|
{
|
||||||
|
XmlDocument doc = new XmlDocument();
|
||||||
|
|
||||||
|
XmlNode xmlnode = doc.CreateNode(XmlNodeType.XmlDeclaration,
|
||||||
|
"", "");
|
||||||
|
|
||||||
|
doc.AppendChild(xmlnode);
|
||||||
|
|
||||||
|
XmlElement rootElement = doc.CreateElement("", "ServerResponse",
|
||||||
|
"");
|
||||||
|
|
||||||
|
doc.AppendChild(rootElement);
|
||||||
|
|
||||||
|
XmlElement result = doc.CreateElement("", "RESULT", "");
|
||||||
|
result.AppendChild(doc.CreateTextNode(value.ToString()));
|
||||||
|
|
||||||
|
rootElement.AppendChild(result);
|
||||||
|
|
||||||
|
return DocToBytes(doc);
|
||||||
|
}
|
||||||
|
|
||||||
|
private byte[] DocToBytes(XmlDocument doc)
|
||||||
|
{
|
||||||
|
MemoryStream ms = new MemoryStream();
|
||||||
|
XmlTextWriter xw = new XmlTextWriter(ms, null);
|
||||||
|
xw.Formatting = Formatting.Indented;
|
||||||
|
doc.WriteTo(xw);
|
||||||
|
xw.Flush();
|
||||||
|
|
||||||
|
return ms.ToArray();
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,131 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Runtime.Serialization;
|
||||||
|
using System.Text;
|
||||||
|
using System.Timers;
|
||||||
|
using System.Xml;
|
||||||
|
using System.Xml.Serialization;
|
||||||
|
using log4net;
|
||||||
|
using Nini.Config;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using OpenSim.Data;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
|
||||||
|
namespace OpenSim.OfflineIM
|
||||||
|
{
|
||||||
|
public class OfflineIMService : OfflineIMServiceBase, IOfflineIMService
|
||||||
|
{
|
||||||
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
private const int MAX_IM = 25;
|
||||||
|
|
||||||
|
private XmlSerializer m_serializer;
|
||||||
|
private static bool m_Initialized = false;
|
||||||
|
|
||||||
|
public OfflineIMService(IConfigSource config)
|
||||||
|
: base(config)
|
||||||
|
{
|
||||||
|
m_serializer = new XmlSerializer(typeof(GridInstantMessage));
|
||||||
|
if (!m_Initialized)
|
||||||
|
{
|
||||||
|
m_Database.DeleteOld();
|
||||||
|
m_Initialized = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<GridInstantMessage> GetMessages(UUID principalID)
|
||||||
|
{
|
||||||
|
List<GridInstantMessage> ims = new List<GridInstantMessage>();
|
||||||
|
|
||||||
|
OfflineIMData[] messages = m_Database.Get("PrincipalID", principalID.ToString());
|
||||||
|
|
||||||
|
if (messages == null || (messages != null && messages.Length == 0))
|
||||||
|
return ims;
|
||||||
|
|
||||||
|
foreach (OfflineIMData m in messages)
|
||||||
|
{
|
||||||
|
using (MemoryStream mstream = new MemoryStream(Encoding.UTF8.GetBytes(m.Data["Message"])))
|
||||||
|
{
|
||||||
|
GridInstantMessage im = (GridInstantMessage)m_serializer.Deserialize(mstream);
|
||||||
|
ims.Add(im);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Then, delete them
|
||||||
|
m_Database.Delete("PrincipalID", principalID.ToString());
|
||||||
|
|
||||||
|
return ims;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool StoreMessage(GridInstantMessage im, out string reason)
|
||||||
|
{
|
||||||
|
reason = string.Empty;
|
||||||
|
|
||||||
|
// TODO Check limits
|
||||||
|
UUID principalID = new UUID(im.toAgentID);
|
||||||
|
long count = m_Database.GetCount("PrincipalID", principalID.ToString());
|
||||||
|
if (count >= MAX_IM)
|
||||||
|
{
|
||||||
|
reason = "Number of offline IMs has maxed out";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
string imXml = string.Empty;
|
||||||
|
using (MemoryStream mstream = new MemoryStream())
|
||||||
|
{
|
||||||
|
XmlWriterSettings settings = new XmlWriterSettings();
|
||||||
|
settings.Encoding = Encoding.UTF8;
|
||||||
|
|
||||||
|
using (XmlWriter writer = XmlWriter.Create(mstream, settings))
|
||||||
|
{
|
||||||
|
m_serializer.Serialize(writer, im);
|
||||||
|
writer.Flush();
|
||||||
|
|
||||||
|
mstream.Position = 0;
|
||||||
|
using (StreamReader sreader = new StreamReader(mstream))
|
||||||
|
{
|
||||||
|
imXml = sreader.ReadToEnd();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
OfflineIMData data = new OfflineIMData();
|
||||||
|
data.PrincipalID = principalID;
|
||||||
|
data.Data = new Dictionary<string, string>();
|
||||||
|
data.Data["Message"] = imXml;
|
||||||
|
|
||||||
|
return m_Database.Store(data);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,83 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Reflection;
|
||||||
|
using Nini.Config;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Data;
|
||||||
|
using OpenSim.Services.Interfaces;
|
||||||
|
using OpenSim.Services.Base;
|
||||||
|
|
||||||
|
namespace OpenSim.OfflineIM
|
||||||
|
{
|
||||||
|
public class OfflineIMServiceBase : ServiceBase
|
||||||
|
{
|
||||||
|
protected IOfflineIMData m_Database = null;
|
||||||
|
|
||||||
|
public OfflineIMServiceBase(IConfigSource config)
|
||||||
|
: base(config)
|
||||||
|
{
|
||||||
|
string dllName = String.Empty;
|
||||||
|
string connString = String.Empty;
|
||||||
|
string realm = "im_offline";
|
||||||
|
|
||||||
|
//
|
||||||
|
// Try reading the [DatabaseService] section, if it exists
|
||||||
|
//
|
||||||
|
IConfig dbConfig = config.Configs["DatabaseService"];
|
||||||
|
if (dbConfig != null)
|
||||||
|
{
|
||||||
|
if (dllName == String.Empty)
|
||||||
|
dllName = dbConfig.GetString("StorageProvider", String.Empty);
|
||||||
|
if (connString == String.Empty)
|
||||||
|
connString = dbConfig.GetString("ConnectionString", String.Empty);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// [Messaging] section overrides [DatabaseService], if it exists
|
||||||
|
//
|
||||||
|
IConfig imConfig = config.Configs["Messaging"];
|
||||||
|
if (imConfig != null)
|
||||||
|
{
|
||||||
|
dllName = imConfig.GetString("StorageProvider", dllName);
|
||||||
|
connString = imConfig.GetString("ConnectionString", connString);
|
||||||
|
realm = imConfig.GetString("Realm", realm);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// We tried, but this doesn't exist. We can't proceed.
|
||||||
|
//
|
||||||
|
if (dllName.Equals(String.Empty))
|
||||||
|
throw new Exception("No StorageProvider configured");
|
||||||
|
|
||||||
|
m_Database = LoadPlugin<IOfflineIMData>(dllName, new Object[] { connString, realm });
|
||||||
|
if (m_Database == null)
|
||||||
|
throw new Exception("Could not find a storage interface in the given module " + dllName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -60,7 +60,6 @@ using System.Runtime.InteropServices;
|
||||||
//
|
//
|
||||||
// You can specify all the values or you can default the Build and Revision Numbers
|
// You can specify all the values or you can default the Build and Revision Numbers
|
||||||
// by using the '*' as shown below:
|
// by using the '*' as shown below:
|
||||||
// [assembly: AssemblyVersion("0.7.5.*")]
|
// [assembly: AssemblyVersion("0.7.6.*")]
|
||||||
|
|
||||||
[assembly : AssemblyVersion("0.7.5.*")]
|
[assembly : AssemblyVersion("0.7.6.*")]
|
||||||
[assembly : AssemblyFileVersion("0.6.5.0")]
|
|
||||||
|
|
|
@ -29,5 +29,5 @@ using System.Runtime.InteropServices;
|
||||||
// Build Number
|
// Build Number
|
||||||
// Revision
|
// Revision
|
||||||
//
|
//
|
||||||
[assembly: AssemblyVersion("0.7.5.*")]
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
|
||||||
|
|
|
@ -85,16 +85,26 @@ namespace OpenSim.ApplicationPlugins.RegionModulesController
|
||||||
if (modulesConfig == null)
|
if (modulesConfig == null)
|
||||||
modulesConfig = m_openSim.ConfigSource.Source.AddConfig("Modules");
|
modulesConfig = m_openSim.ConfigSource.Source.AddConfig("Modules");
|
||||||
|
|
||||||
|
Dictionary<RuntimeAddin, IList<int>> loadedModules = new Dictionary<RuntimeAddin, IList<int>>();
|
||||||
|
|
||||||
// Scan modules and load all that aren't disabled
|
// Scan modules and load all that aren't disabled
|
||||||
foreach (TypeExtensionNode node in
|
foreach (TypeExtensionNode node in
|
||||||
AddinManager.GetExtensionNodes("/OpenSim/RegionModules"))
|
AddinManager.GetExtensionNodes("/OpenSim/RegionModules"))
|
||||||
{
|
{
|
||||||
|
IList<int> loadedModuleData;
|
||||||
|
|
||||||
|
if (!loadedModules.ContainsKey(node.Addin))
|
||||||
|
loadedModules.Add(node.Addin, new List<int> { 0, 0, 0 });
|
||||||
|
|
||||||
|
loadedModuleData = loadedModules[node.Addin];
|
||||||
|
|
||||||
if (node.Type.GetInterface(typeof(ISharedRegionModule).ToString()) != null)
|
if (node.Type.GetInterface(typeof(ISharedRegionModule).ToString()) != null)
|
||||||
{
|
{
|
||||||
if (CheckModuleEnabled(node, modulesConfig))
|
if (CheckModuleEnabled(node, modulesConfig))
|
||||||
{
|
{
|
||||||
m_log.DebugFormat("[REGIONMODULES]: Found shared region module {0}, class {1}", node.Id, node.Type);
|
m_log.DebugFormat("[REGIONMODULES]: Found shared region module {0}, class {1}", node.Id, node.Type);
|
||||||
m_sharedModules.Add(node);
|
m_sharedModules.Add(node);
|
||||||
|
loadedModuleData[0]++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (node.Type.GetInterface(typeof(INonSharedRegionModule).ToString()) != null)
|
else if (node.Type.GetInterface(typeof(INonSharedRegionModule).ToString()) != null)
|
||||||
|
@ -103,14 +113,26 @@ namespace OpenSim.ApplicationPlugins.RegionModulesController
|
||||||
{
|
{
|
||||||
m_log.DebugFormat("[REGIONMODULES]: Found non-shared region module {0}, class {1}", node.Id, node.Type);
|
m_log.DebugFormat("[REGIONMODULES]: Found non-shared region module {0}, class {1}", node.Id, node.Type);
|
||||||
m_nonSharedModules.Add(node);
|
m_nonSharedModules.Add(node);
|
||||||
|
loadedModuleData[1]++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
m_log.DebugFormat("[REGIONMODULES]: Found unknown type of module {0}, class {1}", node.Id, node.Type);
|
m_log.WarnFormat("[REGIONMODULES]: Found unknown type of module {0}, class {1}", node.Id, node.Type);
|
||||||
|
loadedModuleData[2]++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
foreach (KeyValuePair<RuntimeAddin, IList<int>> loadedModuleData in loadedModules)
|
||||||
|
{
|
||||||
|
m_log.InfoFormat(
|
||||||
|
"[REGIONMODULES]: From plugin {0}, (version {1}), loaded {2} modules, {3} shared, {4} non-shared {5} unknown",
|
||||||
|
loadedModuleData.Key.Id,
|
||||||
|
loadedModuleData.Key.Version,
|
||||||
|
loadedModuleData.Value[0] + loadedModuleData.Value[1] + loadedModuleData.Value[2],
|
||||||
|
loadedModuleData.Value[0], loadedModuleData.Value[1], loadedModuleData.Value[2]);
|
||||||
|
}
|
||||||
|
|
||||||
// Load and init the module. We try a constructor with a port
|
// Load and init the module. We try a constructor with a port
|
||||||
// if a port was given, fall back to one without if there is
|
// if a port was given, fall back to one without if there is
|
||||||
// no port or the more specific constructor fails.
|
// no port or the more specific constructor fails.
|
||||||
|
|
|
@ -29,5 +29,5 @@ using System.Runtime.InteropServices;
|
||||||
// Build Number
|
// Build Number
|
||||||
// Revision
|
// Revision
|
||||||
//
|
//
|
||||||
[assembly: AssemblyVersion("0.7.5.*")]
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
|
||||||
|
|
|
@ -50,6 +50,7 @@ using OpenSim.Region.Framework.Scenes;
|
||||||
using OpenSim.Services.Interfaces;
|
using OpenSim.Services.Interfaces;
|
||||||
using PresenceInfo = OpenSim.Services.Interfaces.PresenceInfo;
|
using PresenceInfo = OpenSim.Services.Interfaces.PresenceInfo;
|
||||||
using GridRegion = OpenSim.Services.Interfaces.GridRegion;
|
using GridRegion = OpenSim.Services.Interfaces.GridRegion;
|
||||||
|
using PermissionMask = OpenSim.Framework.PermissionMask;
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.RemoteController
|
namespace OpenSim.ApplicationPlugins.RemoteController
|
||||||
{
|
{
|
||||||
|
@ -140,6 +141,7 @@ namespace OpenSim.ApplicationPlugins.RemoteController
|
||||||
availableMethods["admin_save_heightmap"] = (req, ep) => InvokeXmlRpcMethod(req, ep, XmlRpcSaveHeightmapMethod);
|
availableMethods["admin_save_heightmap"] = (req, ep) => InvokeXmlRpcMethod(req, ep, XmlRpcSaveHeightmapMethod);
|
||||||
|
|
||||||
// Agent management
|
// Agent management
|
||||||
|
availableMethods["admin_get_agents"] = (req, ep) => InvokeXmlRpcMethod(req, ep, XmlRpcGetAgentsMethod);
|
||||||
availableMethods["admin_teleport_agent"] = (req, ep) => InvokeXmlRpcMethod(req, ep, XmlRpcTeleportAgentMethod);
|
availableMethods["admin_teleport_agent"] = (req, ep) => InvokeXmlRpcMethod(req, ep, XmlRpcTeleportAgentMethod);
|
||||||
|
|
||||||
// User management
|
// User management
|
||||||
|
@ -1900,6 +1902,71 @@ namespace OpenSim.ApplicationPlugins.RemoteController
|
||||||
m_log.Info("[RADMIN]: Access List List Request complete");
|
m_log.Info("[RADMIN]: Access List List Request complete");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void XmlRpcGetAgentsMethod(XmlRpcRequest request, XmlRpcResponse response, IPEndPoint remoteClient)
|
||||||
|
{
|
||||||
|
Hashtable responseData = (Hashtable)response.Value;
|
||||||
|
Hashtable requestData = (Hashtable)request.Params[0];
|
||||||
|
|
||||||
|
bool includeChildren = false;
|
||||||
|
|
||||||
|
if (requestData.Contains("include_children"))
|
||||||
|
bool.TryParse((string)requestData["include_children"], out includeChildren);
|
||||||
|
|
||||||
|
Scene scene;
|
||||||
|
GetSceneFromRegionParams(requestData, responseData, out scene);
|
||||||
|
|
||||||
|
ArrayList xmlRpcRegions = new ArrayList();
|
||||||
|
responseData["regions"] = xmlRpcRegions;
|
||||||
|
|
||||||
|
Hashtable xmlRpcRegion = new Hashtable();
|
||||||
|
xmlRpcRegions.Add(xmlRpcRegion);
|
||||||
|
|
||||||
|
xmlRpcRegion["name"] = scene.Name;
|
||||||
|
xmlRpcRegion["id"] = scene.RegionInfo.RegionID.ToString();
|
||||||
|
|
||||||
|
List<ScenePresence> agents = scene.GetScenePresences();
|
||||||
|
ArrayList xmlrpcAgents = new ArrayList();
|
||||||
|
|
||||||
|
foreach (ScenePresence agent in agents)
|
||||||
|
{
|
||||||
|
if (agent.IsChildAgent && !includeChildren)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
Hashtable xmlRpcAgent = new Hashtable();
|
||||||
|
xmlRpcAgent.Add("name", agent.Name);
|
||||||
|
xmlRpcAgent.Add("id", agent.UUID.ToString());
|
||||||
|
xmlRpcAgent.Add("type", agent.PresenceType.ToString());
|
||||||
|
xmlRpcAgent.Add("current_parcel_id", agent.currentParcelUUID.ToString());
|
||||||
|
|
||||||
|
Vector3 pos = agent.AbsolutePosition;
|
||||||
|
xmlRpcAgent.Add("pos_x", pos.X.ToString());
|
||||||
|
xmlRpcAgent.Add("pos_y", pos.Y.ToString());
|
||||||
|
xmlRpcAgent.Add("pos_z", pos.Z.ToString());
|
||||||
|
|
||||||
|
Vector3 lookAt = agent.Lookat;
|
||||||
|
xmlRpcAgent.Add("lookat_x", lookAt.X.ToString());
|
||||||
|
xmlRpcAgent.Add("lookat_y", lookAt.Y.ToString());
|
||||||
|
xmlRpcAgent.Add("lookat_z", lookAt.Z.ToString());
|
||||||
|
|
||||||
|
Vector3 vel = agent.Velocity;
|
||||||
|
xmlRpcAgent.Add("vel_x", vel.X.ToString());
|
||||||
|
xmlRpcAgent.Add("vel_y", vel.Y.ToString());
|
||||||
|
xmlRpcAgent.Add("vel_z", vel.Z.ToString());
|
||||||
|
|
||||||
|
xmlRpcAgent.Add("is_flying", agent.Flying.ToString());
|
||||||
|
xmlRpcAgent.Add("is_sat_on_ground", agent.SitGround.ToString());
|
||||||
|
xmlRpcAgent.Add("is_sat_on_object", agent.IsSatOnObject.ToString());
|
||||||
|
|
||||||
|
xmlrpcAgents.Add(xmlRpcAgent);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_log.DebugFormat(
|
||||||
|
"[REMOTE ADMIN]: XmlRpcGetAgents found {0} agents in {1}", xmlrpcAgents.Count, scene.Name);
|
||||||
|
|
||||||
|
xmlRpcRegion["agents"] = xmlrpcAgents;
|
||||||
|
responseData["success"] = true;
|
||||||
|
}
|
||||||
|
|
||||||
private void XmlRpcTeleportAgentMethod(XmlRpcRequest request, XmlRpcResponse response, IPEndPoint remoteClient)
|
private void XmlRpcTeleportAgentMethod(XmlRpcRequest request, XmlRpcResponse response, IPEndPoint remoteClient)
|
||||||
{
|
{
|
||||||
Hashtable responseData = (Hashtable)response.Value;
|
Hashtable responseData = (Hashtable)response.Value;
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,11 +0,0 @@
|
||||||
<Addin id="OpenSim.ApplicationPlugins.Rest.Inventory" version="0.1">
|
|
||||||
<Runtime>
|
|
||||||
<Import assembly="OpenSim.ApplicationPlugins.Rest.Inventory.dll"/>
|
|
||||||
</Runtime>
|
|
||||||
<Dependencies>
|
|
||||||
<Addin id="OpenSim" version="0.5" />
|
|
||||||
</Dependencies>
|
|
||||||
<Extension path = "/OpenSim/Startup">
|
|
||||||
<Plugin id="RestInventory" type="OpenSim.ApplicationPlugins.Rest.Inventory.RestHandler" />
|
|
||||||
</Extension>
|
|
||||||
</Addin>
|
|
|
@ -1,551 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Text;
|
|
||||||
using log4net;
|
|
||||||
using Nini.Config;
|
|
||||||
using OpenSim.Framework;
|
|
||||||
using OpenSim.Framework.Communications;
|
|
||||||
using OpenSim.Services.Interfaces;
|
|
||||||
using IAvatarService = OpenSim.Services.Interfaces.IAvatarService;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Inventory
|
|
||||||
{
|
|
||||||
public class Rest
|
|
||||||
{
|
|
||||||
internal static readonly ILog Log =
|
|
||||||
LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
|
||||||
|
|
||||||
internal static bool DEBUG = Log.IsDebugEnabled;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Supported authentication schemes
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
public const string AS_BASIC = "Basic"; // simple user/password verification
|
|
||||||
public const string AS_DIGEST = "Digest"; // password safe authentication
|
|
||||||
|
|
||||||
/// Supported Digest algorithms
|
|
||||||
|
|
||||||
public const string Digest_MD5 = "MD5"; // assumed default if omitted
|
|
||||||
public const string Digest_MD5Sess = "MD5-sess"; // session-span - not good for REST?
|
|
||||||
|
|
||||||
public const string Qop_Auth = "auth"; // authentication only
|
|
||||||
public const string Qop_Int = "auth-int"; // TODO
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// These values have a single value for the whole
|
|
||||||
/// domain and lifetime of the plugin handler. We
|
|
||||||
/// make them static for ease of reference within
|
|
||||||
/// the assembly. These are initialized by the
|
|
||||||
/// RestHandler class during start-up.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal static IRestHandler Plugin = null;
|
|
||||||
internal static OpenSimBase main = null;
|
|
||||||
internal static string Prefix = null;
|
|
||||||
internal static IConfig Config = null;
|
|
||||||
internal static string GodKey = null;
|
|
||||||
internal static bool Authenticate = true;
|
|
||||||
internal static bool Secure = true;
|
|
||||||
internal static bool ExtendedEscape = true;
|
|
||||||
internal static bool DumpAsset = false;
|
|
||||||
internal static bool Fill = true;
|
|
||||||
internal static bool FlushEnabled = true;
|
|
||||||
internal static string Realm = "OpenSim REST";
|
|
||||||
internal static string Scheme = AS_BASIC;
|
|
||||||
internal static int DumpLineSize = 32; // Should be a multiple of 16 or (possibly) 4
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// These are all dependent upon the Comms manager
|
|
||||||
/// being initialized. So they have to be properties
|
|
||||||
/// because the comms manager is now a module and is
|
|
||||||
/// not guaranteed to be there when the rest handler
|
|
||||||
/// initializes.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal static IInventoryService InventoryServices
|
|
||||||
{
|
|
||||||
get { return main.SceneManager.CurrentOrFirstScene.InventoryService; }
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static IUserAccountService UserServices
|
|
||||||
{
|
|
||||||
get { return main.SceneManager.CurrentOrFirstScene.UserAccountService; }
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static IAuthenticationService AuthServices
|
|
||||||
{
|
|
||||||
get { return main.SceneManager.CurrentOrFirstScene.AuthenticationService; }
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static IAvatarService AvatarServices
|
|
||||||
{
|
|
||||||
get { return main.SceneManager.CurrentOrFirstScene.AvatarService; }
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static IAssetService AssetServices
|
|
||||||
{
|
|
||||||
get { return main.SceneManager.CurrentOrFirstScene.AssetService; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// HTTP requires that status information be generated for PUT
|
|
||||||
/// and POST opertaions. This is in support of that. The
|
|
||||||
/// operation verb gets substituted into the first string,
|
|
||||||
/// and the completion code is inserted into the tail. The
|
|
||||||
/// strings are put here to encourage consistency.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal static string statusHead = "<html><body><title>{0} status</title><break>";
|
|
||||||
internal static string statusTail = "</body></html>";
|
|
||||||
|
|
||||||
internal static Dictionary<int,string> HttpStatusDesc;
|
|
||||||
|
|
||||||
static Rest()
|
|
||||||
{
|
|
||||||
HttpStatusDesc = new Dictionary<int,string>();
|
|
||||||
if (HttpStatusCodeArray.Length != HttpStatusDescArray.Length)
|
|
||||||
{
|
|
||||||
Log.ErrorFormat("{0} HTTP Status Code and Description arrays do not match");
|
|
||||||
throw new Exception("HTTP Status array discrepancy");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Repackage the data into something more tractable. The sparse
|
|
||||||
// nature of HTTP return codes makes an array a bad choice.
|
|
||||||
|
|
||||||
for (int i=0; i<HttpStatusCodeArray.Length; i++)
|
|
||||||
{
|
|
||||||
HttpStatusDesc.Add(HttpStatusCodeArray[i], HttpStatusDescArray[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static int CreationDate
|
|
||||||
{
|
|
||||||
get { return (int) (DateTime.UtcNow - new DateTime(1970, 1, 1)).TotalSeconds; }
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static string MsgId
|
|
||||||
{
|
|
||||||
get { return Plugin.MsgId; }
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static string RequestId
|
|
||||||
{
|
|
||||||
get { return Plugin.RequestId; }
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static Encoding Encoding = Util.UTF8;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Version control for REST implementation. This
|
|
||||||
/// refers to the overall infrastructure represented
|
|
||||||
/// by the following classes
|
|
||||||
/// RequestData
|
|
||||||
/// RequestInventoryPlugin
|
|
||||||
/// Rest
|
|
||||||
/// It does no describe implementation classes such as
|
|
||||||
/// RestInventoryServices, which may morph much more
|
|
||||||
/// often. Such classes ARE dependent upon this however
|
|
||||||
/// and should check it in their Initialize method.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
public static readonly float Version = 1.0F;
|
|
||||||
public const string Name = "REST 1.0";
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Currently defined HTTP methods.
|
|
||||||
/// Only GET and HEAD are required to be
|
|
||||||
/// supported by all servers. See Respond
|
|
||||||
/// to see how these are handled.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
// REST AGENT 1.0 interpretations
|
|
||||||
public const string GET = "get"; // information retrieval - server state unchanged
|
|
||||||
public const string HEAD = "head"; // same as get except only the headers are returned.
|
|
||||||
public const string POST = "post"; // Replace the URI designated resource with the entity.
|
|
||||||
public const string PUT = "put"; // Add the entity to the context represented by the URI
|
|
||||||
public const string DELETE = "delete"; // Remove the URI designated resource from the server.
|
|
||||||
|
|
||||||
public const string OPTIONS = "options"; //
|
|
||||||
public const string TRACE = "trace"; //
|
|
||||||
public const string CONNECT = "connect"; //
|
|
||||||
|
|
||||||
// Define this in one place...
|
|
||||||
|
|
||||||
public const string UrlPathSeparator = "/";
|
|
||||||
public const string UrlMethodSeparator = ":";
|
|
||||||
|
|
||||||
// Redirection qualifications
|
|
||||||
|
|
||||||
public const bool PERMANENT = false;
|
|
||||||
public const bool TEMPORARY = true;
|
|
||||||
|
|
||||||
// Constant arrays used by String.Split
|
|
||||||
|
|
||||||
public static readonly char C_SPACE = ' ';
|
|
||||||
public static readonly char C_SLASH = '/';
|
|
||||||
public static readonly char C_PATHSEP = '/';
|
|
||||||
public static readonly char C_COLON = ':';
|
|
||||||
public static readonly char C_PLUS = '+';
|
|
||||||
public static readonly char C_PERIOD = '.';
|
|
||||||
public static readonly char C_COMMA = ',';
|
|
||||||
public static readonly char C_DQUOTE = '"';
|
|
||||||
|
|
||||||
public static readonly string CS_SPACE = " ";
|
|
||||||
public static readonly string CS_SLASH = "/";
|
|
||||||
public static readonly string CS_PATHSEP = "/";
|
|
||||||
public static readonly string CS_COLON = ":";
|
|
||||||
public static readonly string CS_PLUS = "+";
|
|
||||||
public static readonly string CS_PERIOD = ".";
|
|
||||||
public static readonly string CS_COMMA = ",";
|
|
||||||
public static readonly string CS_DQUOTE = "\"";
|
|
||||||
|
|
||||||
public static readonly char[] CA_SPACE = { C_SPACE };
|
|
||||||
public static readonly char[] CA_SLASH = { C_SLASH };
|
|
||||||
public static readonly char[] CA_PATHSEP = { C_PATHSEP };
|
|
||||||
public static readonly char[] CA_COLON = { C_COLON };
|
|
||||||
public static readonly char[] CA_PERIOD = { C_PERIOD };
|
|
||||||
public static readonly char[] CA_PLUS = { C_PLUS };
|
|
||||||
public static readonly char[] CA_COMMA = { C_COMMA };
|
|
||||||
public static readonly char[] CA_DQUOTE = { C_DQUOTE };
|
|
||||||
|
|
||||||
// HTTP Code Values (in value order)
|
|
||||||
|
|
||||||
public const int HttpStatusCodeContinue = 100;
|
|
||||||
public const int HttpStatusCodeSwitchingProtocols = 101;
|
|
||||||
|
|
||||||
public const int HttpStatusCodeOK = 200;
|
|
||||||
public const int HttpStatusCodeCreated = 201;
|
|
||||||
public const int HttpStatusCodeAccepted = 202;
|
|
||||||
public const int HttpStatusCodeNonAuthoritative = 203;
|
|
||||||
public const int HttpStatusCodeNoContent = 204;
|
|
||||||
public const int HttpStatusCodeResetContent = 205;
|
|
||||||
public const int HttpStatusCodePartialContent = 206;
|
|
||||||
|
|
||||||
public const int HttpStatusCodeMultipleChoices = 300;
|
|
||||||
public const int HttpStatusCodePermanentRedirect = 301;
|
|
||||||
public const int HttpStatusCodeFound = 302;
|
|
||||||
public const int HttpStatusCodeSeeOther = 303;
|
|
||||||
public const int HttpStatusCodeNotModified = 304;
|
|
||||||
public const int HttpStatusCodeUseProxy = 305;
|
|
||||||
public const int HttpStatusCodeReserved306 = 306;
|
|
||||||
public const int HttpStatusCodeTemporaryRedirect = 307;
|
|
||||||
|
|
||||||
public const int HttpStatusCodeBadRequest = 400;
|
|
||||||
public const int HttpStatusCodeNotAuthorized = 401;
|
|
||||||
public const int HttpStatusCodePaymentRequired = 402;
|
|
||||||
public const int HttpStatusCodeForbidden = 403;
|
|
||||||
public const int HttpStatusCodeNotFound = 404;
|
|
||||||
public const int HttpStatusCodeMethodNotAllowed = 405;
|
|
||||||
public const int HttpStatusCodeNotAcceptable = 406;
|
|
||||||
public const int HttpStatusCodeProxyAuthenticate = 407;
|
|
||||||
public const int HttpStatusCodeTimeOut = 408;
|
|
||||||
public const int HttpStatusCodeConflict = 409;
|
|
||||||
public const int HttpStatusCodeGone = 410;
|
|
||||||
public const int HttpStatusCodeLengthRequired = 411;
|
|
||||||
public const int HttpStatusCodePreconditionFailed = 412;
|
|
||||||
public const int HttpStatusCodeEntityTooLarge = 413;
|
|
||||||
public const int HttpStatusCodeUriTooLarge = 414;
|
|
||||||
public const int HttpStatusCodeUnsupportedMedia = 415;
|
|
||||||
public const int HttpStatusCodeRangeNotSatsified = 416;
|
|
||||||
public const int HttpStatusCodeExpectationFailed = 417;
|
|
||||||
|
|
||||||
public const int HttpStatusCodeServerError = 500;
|
|
||||||
public const int HttpStatusCodeNotImplemented = 501;
|
|
||||||
public const int HttpStatusCodeBadGateway = 502;
|
|
||||||
public const int HttpStatusCodeServiceUnavailable = 503;
|
|
||||||
public const int HttpStatusCodeGatewayTimeout = 504;
|
|
||||||
public const int HttpStatusCodeHttpVersionError = 505;
|
|
||||||
|
|
||||||
public static readonly int[] HttpStatusCodeArray = {
|
|
||||||
HttpStatusCodeContinue,
|
|
||||||
HttpStatusCodeSwitchingProtocols,
|
|
||||||
HttpStatusCodeOK,
|
|
||||||
HttpStatusCodeCreated,
|
|
||||||
HttpStatusCodeAccepted,
|
|
||||||
HttpStatusCodeNonAuthoritative,
|
|
||||||
HttpStatusCodeNoContent,
|
|
||||||
HttpStatusCodeResetContent,
|
|
||||||
HttpStatusCodePartialContent,
|
|
||||||
HttpStatusCodeMultipleChoices,
|
|
||||||
HttpStatusCodePermanentRedirect,
|
|
||||||
HttpStatusCodeFound,
|
|
||||||
HttpStatusCodeSeeOther,
|
|
||||||
HttpStatusCodeNotModified,
|
|
||||||
HttpStatusCodeUseProxy,
|
|
||||||
HttpStatusCodeReserved306,
|
|
||||||
HttpStatusCodeTemporaryRedirect,
|
|
||||||
HttpStatusCodeBadRequest,
|
|
||||||
HttpStatusCodeNotAuthorized,
|
|
||||||
HttpStatusCodePaymentRequired,
|
|
||||||
HttpStatusCodeForbidden,
|
|
||||||
HttpStatusCodeNotFound,
|
|
||||||
HttpStatusCodeMethodNotAllowed,
|
|
||||||
HttpStatusCodeNotAcceptable,
|
|
||||||
HttpStatusCodeProxyAuthenticate,
|
|
||||||
HttpStatusCodeTimeOut,
|
|
||||||
HttpStatusCodeConflict,
|
|
||||||
HttpStatusCodeGone,
|
|
||||||
HttpStatusCodeLengthRequired,
|
|
||||||
HttpStatusCodePreconditionFailed,
|
|
||||||
HttpStatusCodeEntityTooLarge,
|
|
||||||
HttpStatusCodeUriTooLarge,
|
|
||||||
HttpStatusCodeUnsupportedMedia,
|
|
||||||
HttpStatusCodeRangeNotSatsified,
|
|
||||||
HttpStatusCodeExpectationFailed,
|
|
||||||
HttpStatusCodeServerError,
|
|
||||||
HttpStatusCodeNotImplemented,
|
|
||||||
HttpStatusCodeBadGateway,
|
|
||||||
HttpStatusCodeServiceUnavailable,
|
|
||||||
HttpStatusCodeGatewayTimeout,
|
|
||||||
HttpStatusCodeHttpVersionError
|
|
||||||
};
|
|
||||||
|
|
||||||
// HTTP Status Descriptions (in status code order)
|
|
||||||
// This array must be kept strictly consistent with respect
|
|
||||||
// to the status code array above.
|
|
||||||
|
|
||||||
public static readonly string[] HttpStatusDescArray = {
|
|
||||||
"Continue Request",
|
|
||||||
"Switching Protocols",
|
|
||||||
"OK",
|
|
||||||
"CREATED",
|
|
||||||
"ACCEPTED",
|
|
||||||
"NON-AUTHORITATIVE INFORMATION",
|
|
||||||
"NO CONTENT",
|
|
||||||
"RESET CONTENT",
|
|
||||||
"PARTIAL CONTENT",
|
|
||||||
"MULTIPLE CHOICES",
|
|
||||||
"PERMANENT REDIRECT",
|
|
||||||
"FOUND",
|
|
||||||
"SEE OTHER",
|
|
||||||
"NOT MODIFIED",
|
|
||||||
"USE PROXY",
|
|
||||||
"RESERVED CODE 306",
|
|
||||||
"TEMPORARY REDIRECT",
|
|
||||||
"BAD REQUEST",
|
|
||||||
"NOT AUTHORIZED",
|
|
||||||
"PAYMENT REQUIRED",
|
|
||||||
"FORBIDDEN",
|
|
||||||
"NOT FOUND",
|
|
||||||
"METHOD NOT ALLOWED",
|
|
||||||
"NOT ACCEPTABLE",
|
|
||||||
"PROXY AUTHENTICATION REQUIRED",
|
|
||||||
"TIMEOUT",
|
|
||||||
"CONFLICT",
|
|
||||||
"GONE",
|
|
||||||
"LENGTH REQUIRED",
|
|
||||||
"PRECONDITION FAILED",
|
|
||||||
"ENTITY TOO LARGE",
|
|
||||||
"URI TOO LARGE",
|
|
||||||
"UNSUPPORTED MEDIA",
|
|
||||||
"RANGE NOT SATISFIED",
|
|
||||||
"EXPECTATION FAILED",
|
|
||||||
"SERVER ERROR",
|
|
||||||
"NOT IMPLEMENTED",
|
|
||||||
"BAD GATEWAY",
|
|
||||||
"SERVICE UNAVAILABLE",
|
|
||||||
"GATEWAY TIMEOUT",
|
|
||||||
"HTTP VERSION NOT SUPPORTED"
|
|
||||||
};
|
|
||||||
|
|
||||||
// HTTP Headers
|
|
||||||
|
|
||||||
public const string HttpHeaderAccept = "Accept";
|
|
||||||
public const string HttpHeaderAcceptCharset = "Accept-Charset";
|
|
||||||
public const string HttpHeaderAcceptEncoding = "Accept-Encoding";
|
|
||||||
public const string HttpHeaderAcceptLanguage = "Accept-Language";
|
|
||||||
public const string HttpHeaderAcceptRanges = "Accept-Ranges";
|
|
||||||
public const string HttpHeaderAge = "Age";
|
|
||||||
public const string HttpHeaderAllow = "Allow";
|
|
||||||
public const string HttpHeaderAuthorization = "Authorization";
|
|
||||||
public const string HttpHeaderCacheControl = "Cache-Control";
|
|
||||||
public const string HttpHeaderConnection = "Connection";
|
|
||||||
public const string HttpHeaderContentEncoding = "Content-Encoding";
|
|
||||||
public const string HttpHeaderContentLanguage = "Content-Language";
|
|
||||||
public const string HttpHeaderContentLength = "Content-Length";
|
|
||||||
public const string HttpHeaderContentLocation = "Content-Location";
|
|
||||||
public const string HttpHeaderContentMD5 = "Content-MD5";
|
|
||||||
public const string HttpHeaderContentRange = "Content-Range";
|
|
||||||
public const string HttpHeaderContentType = "Content-Type";
|
|
||||||
public const string HttpHeaderDate = "Date";
|
|
||||||
public const string HttpHeaderETag = "ETag";
|
|
||||||
public const string HttpHeaderExpect = "Expect";
|
|
||||||
public const string HttpHeaderExpires = "Expires";
|
|
||||||
public const string HttpHeaderFrom = "From";
|
|
||||||
public const string HttpHeaderHost = "Host";
|
|
||||||
public const string HttpHeaderIfMatch = "If-Match";
|
|
||||||
public const string HttpHeaderIfModifiedSince = "If-Modified-Since";
|
|
||||||
public const string HttpHeaderIfNoneMatch = "If-None-Match";
|
|
||||||
public const string HttpHeaderIfRange = "If-Range";
|
|
||||||
public const string HttpHeaderIfUnmodifiedSince = "If-Unmodified-Since";
|
|
||||||
public const string HttpHeaderLastModified = "Last-Modified";
|
|
||||||
public const string HttpHeaderLocation = "Location";
|
|
||||||
public const string HttpHeaderMaxForwards = "Max-Forwards";
|
|
||||||
public const string HttpHeaderPragma = "Pragma";
|
|
||||||
public const string HttpHeaderProxyAuthenticate = "Proxy-Authenticate";
|
|
||||||
public const string HttpHeaderProxyAuthorization = "Proxy-Authorization";
|
|
||||||
public const string HttpHeaderRange = "Range";
|
|
||||||
public const string HttpHeaderReferer = "Referer";
|
|
||||||
public const string HttpHeaderRetryAfter = "Retry-After";
|
|
||||||
public const string HttpHeaderServer = "Server";
|
|
||||||
public const string HttpHeaderTE = "TE";
|
|
||||||
public const string HttpHeaderTrailer = "Trailer";
|
|
||||||
public const string HttpHeaderTransferEncoding = "Transfer-Encoding";
|
|
||||||
public const string HttpHeaderUpgrade = "Upgrade";
|
|
||||||
public const string HttpHeaderUserAgent = "User-Agent";
|
|
||||||
public const string HttpHeaderVary = "Vary";
|
|
||||||
public const string HttpHeaderVia = "Via";
|
|
||||||
public const string HttpHeaderWarning = "Warning";
|
|
||||||
public const string HttpHeaderWWWAuthenticate = "WWW-Authenticate";
|
|
||||||
|
|
||||||
/// Utility routines
|
|
||||||
|
|
||||||
public static string StringToBase64(string str)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
byte[] encData_byte = new byte[str.Length];
|
|
||||||
encData_byte = Util.UTF8.GetBytes(str);
|
|
||||||
return Convert.ToBase64String(encData_byte);
|
|
||||||
}
|
|
||||||
catch
|
|
||||||
{
|
|
||||||
return String.Empty;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static string Base64ToString(string str)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
return Util.Base64ToString(str);
|
|
||||||
}
|
|
||||||
catch
|
|
||||||
{
|
|
||||||
return String.Empty;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private const string hvals = "0123456789abcdef";
|
|
||||||
|
|
||||||
public static int Hex2Int(string hex)
|
|
||||||
{
|
|
||||||
int val = 0;
|
|
||||||
int sum = 0;
|
|
||||||
string tmp = null;
|
|
||||||
|
|
||||||
if (hex != null)
|
|
||||||
{
|
|
||||||
tmp = hex.ToLower();
|
|
||||||
for (int i = 0; i < tmp.Length; i++)
|
|
||||||
{
|
|
||||||
val = hvals.IndexOf(tmp[i]);
|
|
||||||
if (val == -1)
|
|
||||||
break;
|
|
||||||
sum *= 16;
|
|
||||||
sum += val;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return sum;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Nonce management
|
|
||||||
|
|
||||||
public static string NonceGenerator()
|
|
||||||
{
|
|
||||||
return StringToBase64(CreationDate + Guid.NewGuid().ToString());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Dump the specified data stream
|
|
||||||
|
|
||||||
public static void Dump(byte[] data)
|
|
||||||
{
|
|
||||||
char[] buffer = new char[DumpLineSize];
|
|
||||||
int cc = 0;
|
|
||||||
|
|
||||||
for (int i = 0; i < data.Length; i++)
|
|
||||||
{
|
|
||||||
if (i % DumpLineSize == 0) Console.Write("\n{0}: ",i.ToString("d8"));
|
|
||||||
|
|
||||||
if (i % 4 == 0) Console.Write(" ");
|
|
||||||
|
|
||||||
Console.Write("{0}",data[i].ToString("x2"));
|
|
||||||
|
|
||||||
if (data[i] < 127 && data[i] > 31)
|
|
||||||
buffer[i % DumpLineSize] = (char) data[i];
|
|
||||||
else
|
|
||||||
buffer[i % DumpLineSize] = '.';
|
|
||||||
|
|
||||||
cc++;
|
|
||||||
|
|
||||||
if (i != 0 && (i + 1) % DumpLineSize == 0)
|
|
||||||
{
|
|
||||||
Console.Write(" |"+(new String(buffer))+"|");
|
|
||||||
cc = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Finish off any incomplete line
|
|
||||||
|
|
||||||
if (cc != 0)
|
|
||||||
{
|
|
||||||
for (int i = cc ; i < DumpLineSize; i++)
|
|
||||||
{
|
|
||||||
if (i % 4 == 0) Console.Write(" ");
|
|
||||||
Console.Write(" ");
|
|
||||||
buffer[i % DumpLineSize] = ' ';
|
|
||||||
}
|
|
||||||
Console.WriteLine(" |"+(new String(buffer))+"|");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Console.Write("\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Local exception type
|
|
||||||
|
|
||||||
public class RestException : Exception
|
|
||||||
{
|
|
||||||
internal int statusCode;
|
|
||||||
internal string statusDesc;
|
|
||||||
internal string httpmethod;
|
|
||||||
internal string httppath;
|
|
||||||
|
|
||||||
public RestException(string msg) : base(msg)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,860 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Collections;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Xml;
|
|
||||||
using OpenMetaverse;
|
|
||||||
using OpenSim.Framework;
|
|
||||||
using OpenSim.Framework.Servers;
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
|
||||||
using OpenSim.Services.Interfaces;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Inventory
|
|
||||||
{
|
|
||||||
|
|
||||||
public class RestAppearanceServices : IRest
|
|
||||||
{
|
|
||||||
// private static readonly int PARM_USERID = 0;
|
|
||||||
|
|
||||||
// private static readonly int PARM_PATH = 1;
|
|
||||||
|
|
||||||
// private bool enabled = false;
|
|
||||||
private string qPrefix = "appearance";
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The constructor makes sure that the service prefix is absolute
|
|
||||||
/// and the registers the service handler and the allocator.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
public RestAppearanceServices()
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} User appearance services initializing", MsgId);
|
|
||||||
Rest.Log.InfoFormat("{0} Using REST Implementation Version {1}", MsgId, Rest.Version);
|
|
||||||
|
|
||||||
// If a relative path was specified for the handler's domain,
|
|
||||||
// add the standard prefix to make it absolute, e.g. /admin
|
|
||||||
|
|
||||||
if (!qPrefix.StartsWith(Rest.UrlPathSeparator))
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} Domain is relative, adding absolute prefix", MsgId);
|
|
||||||
qPrefix = String.Format("{0}{1}{2}", Rest.Prefix, Rest.UrlPathSeparator, qPrefix);
|
|
||||||
qPrefix = String.Format("{0}{1}{2}", Rest.Prefix, Rest.UrlPathSeparator, qPrefix);
|
|
||||||
Rest.Log.InfoFormat("{0} Domain is now <{1}>", MsgId, qPrefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Register interface using the absolute URI.
|
|
||||||
|
|
||||||
Rest.Plugin.AddPathHandler(DoAppearance,qPrefix,Allocate);
|
|
||||||
|
|
||||||
// Activate if everything went OK
|
|
||||||
|
|
||||||
// enabled = true;
|
|
||||||
|
|
||||||
Rest.Log.InfoFormat("{0} User appearance services initialization complete", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Post-construction, pre-enabled initialization opportunity
|
|
||||||
/// Not currently exploited.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
public void Initialize()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Called by the plug-in to halt service processing. Local processing is
|
|
||||||
/// disabled.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
public void Close()
|
|
||||||
{
|
|
||||||
// enabled = false;
|
|
||||||
Rest.Log.InfoFormat("{0} User appearance services closing down", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This property is declared locally because it is used a lot and
|
|
||||||
/// brevity is nice.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal string MsgId
|
|
||||||
{
|
|
||||||
get { return Rest.MsgId; }
|
|
||||||
}
|
|
||||||
|
|
||||||
#region Interface
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The plugin (RestHandler) calls this method to allocate the request
|
|
||||||
/// state carrier for a new request. It is destroyed when the request
|
|
||||||
/// completes. All request-instance specific state is kept here. This
|
|
||||||
/// is registered when this service provider is registered.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name=request>Inbound HTTP request information</param>
|
|
||||||
/// <param name=response>Outbound HTTP request information</param>
|
|
||||||
/// <param name=qPrefix>REST service domain prefix</param>
|
|
||||||
/// <returns>A RequestData instance suitable for this service</returns>
|
|
||||||
|
|
||||||
private RequestData Allocate(OSHttpRequest request, OSHttpResponse response, string prefix)
|
|
||||||
{
|
|
||||||
return (RequestData) new AppearanceRequestData(request, response, prefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This method is registered with the handler when this service provider
|
|
||||||
/// is initialized. It is called whenever the plug-in identifies this service
|
|
||||||
/// provider as the best match for a given request.
|
|
||||||
/// It handles all aspects of inventory REST processing, i.e. /admin/inventory
|
|
||||||
/// </summary>
|
|
||||||
/// <param name=hdata>A consolidated HTTP request work area</param>
|
|
||||||
|
|
||||||
private void DoAppearance(RequestData hdata)
|
|
||||||
{
|
|
||||||
// !!! REFACTORIMG PROBLEM. This needs rewriting for 0.7
|
|
||||||
|
|
||||||
//AppearanceRequestData rdata = (AppearanceRequestData) hdata;
|
|
||||||
|
|
||||||
//Rest.Log.DebugFormat("{0} DoAppearance ENTRY", MsgId);
|
|
||||||
|
|
||||||
//// If we're disabled, do nothing.
|
|
||||||
|
|
||||||
//if (!enabled)
|
|
||||||
//{
|
|
||||||
// return;
|
|
||||||
//}
|
|
||||||
|
|
||||||
//// Now that we know this is a serious attempt to
|
|
||||||
//// access inventory data, we should find out who
|
|
||||||
//// is asking, and make sure they are authorized
|
|
||||||
//// to do so. We need to validate the caller's
|
|
||||||
//// identity before revealing anything about the
|
|
||||||
//// status quo. Authenticate throws an exception
|
|
||||||
//// via Fail if no identity information is present.
|
|
||||||
////
|
|
||||||
//// With the present HTTP server we can't use the
|
|
||||||
//// builtin authentication mechanisms because they
|
|
||||||
//// would be enforced for all in-bound requests.
|
|
||||||
//// Instead we look at the headers ourselves and
|
|
||||||
//// handle authentication directly.
|
|
||||||
|
|
||||||
//try
|
|
||||||
//{
|
|
||||||
// if (!rdata.IsAuthenticated)
|
|
||||||
// {
|
|
||||||
// rdata.Fail(Rest.HttpStatusCodeNotAuthorized,String.Format("user \"{0}\" could not be authenticated", rdata.userName));
|
|
||||||
// }
|
|
||||||
//}
|
|
||||||
//catch (RestException e)
|
|
||||||
//{
|
|
||||||
// if (e.statusCode == Rest.HttpStatusCodeNotAuthorized)
|
|
||||||
// {
|
|
||||||
// Rest.Log.WarnFormat("{0} User not authenticated", MsgId);
|
|
||||||
// Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId, rdata.request.Headers.Get("Authorization"));
|
|
||||||
// }
|
|
||||||
// else
|
|
||||||
// {
|
|
||||||
// Rest.Log.ErrorFormat("{0} User authentication failed", MsgId);
|
|
||||||
// Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId, rdata.request.Headers.Get("Authorization"));
|
|
||||||
// }
|
|
||||||
// throw (e);
|
|
||||||
//}
|
|
||||||
|
|
||||||
//Rest.Log.DebugFormat("{0} Authenticated {1}", MsgId, rdata.userName);
|
|
||||||
|
|
||||||
//// We can only get here if we are authorized
|
|
||||||
////
|
|
||||||
//// The requestor may have specified an UUID or
|
|
||||||
//// a conjoined FirstName LastName string. We'll
|
|
||||||
//// try both. If we fail with the first, UUID,
|
|
||||||
//// attempt, we try the other. As an example, the
|
|
||||||
//// URI for a valid inventory request might be:
|
|
||||||
////
|
|
||||||
//// http://<host>:<port>/admin/inventory/Arthur Dent
|
|
||||||
////
|
|
||||||
//// Indicating that this is an inventory request for
|
|
||||||
//// an avatar named Arthur Dent. This is ALL that is
|
|
||||||
//// required to designate a GET for an entire
|
|
||||||
//// inventory.
|
|
||||||
////
|
|
||||||
|
|
||||||
//// Do we have at least a user agent name?
|
|
||||||
|
|
||||||
//if (rdata.Parameters.Length < 1)
|
|
||||||
//{
|
|
||||||
// Rest.Log.WarnFormat("{0} Appearance: No user agent identifier specified", MsgId);
|
|
||||||
// rdata.Fail(Rest.HttpStatusCodeBadRequest, "no user identity specified");
|
|
||||||
//}
|
|
||||||
|
|
||||||
//// The first parameter MUST be the agent identification, either an UUID
|
|
||||||
//// or a space-separated First-name Last-Name specification. We check for
|
|
||||||
//// an UUID first, if anyone names their character using a valid UUID
|
|
||||||
//// that identifies another existing avatar will cause this a problem...
|
|
||||||
|
|
||||||
//try
|
|
||||||
//{
|
|
||||||
// rdata.uuid = new UUID(rdata.Parameters[PARM_USERID]);
|
|
||||||
// Rest.Log.DebugFormat("{0} UUID supplied", MsgId);
|
|
||||||
// rdata.userProfile = Rest.UserServices.GetUserProfile(rdata.uuid);
|
|
||||||
//}
|
|
||||||
//catch
|
|
||||||
//{
|
|
||||||
// string[] names = rdata.Parameters[PARM_USERID].Split(Rest.CA_SPACE);
|
|
||||||
// if (names.Length == 2)
|
|
||||||
// {
|
|
||||||
// Rest.Log.DebugFormat("{0} Agent Name supplied [2]", MsgId);
|
|
||||||
// rdata.userProfile = Rest.UserServices.GetUserProfile(names[0],names[1]);
|
|
||||||
// }
|
|
||||||
// else
|
|
||||||
// {
|
|
||||||
// Rest.Log.WarnFormat("{0} A Valid UUID or both first and last names must be specified", MsgId);
|
|
||||||
// rdata.Fail(Rest.HttpStatusCodeBadRequest, "invalid user identity");
|
|
||||||
// }
|
|
||||||
//}
|
|
||||||
|
|
||||||
//// If the user profile is null then either the server is broken, or the
|
|
||||||
//// user is not known. We always assume the latter case.
|
|
||||||
|
|
||||||
//if (rdata.userProfile != null)
|
|
||||||
//{
|
|
||||||
// Rest.Log.DebugFormat("{0} User profile obtained for agent {1} {2}",
|
|
||||||
// MsgId, rdata.userProfile.FirstName, rdata.userProfile.SurName);
|
|
||||||
//}
|
|
||||||
//else
|
|
||||||
//{
|
|
||||||
// Rest.Log.WarnFormat("{0} No user profile for {1}", MsgId, rdata.path);
|
|
||||||
// rdata.Fail(Rest.HttpStatusCodeNotFound, "unrecognized user identity");
|
|
||||||
//}
|
|
||||||
|
|
||||||
//// If we get to here, then we have effectively validated the user's
|
|
||||||
|
|
||||||
//switch (rdata.method)
|
|
||||||
//{
|
|
||||||
// case Rest.HEAD : // Do the processing, set the status code, suppress entity
|
|
||||||
// DoGet(rdata);
|
|
||||||
// rdata.buffer = null;
|
|
||||||
// break;
|
|
||||||
|
|
||||||
// case Rest.GET : // Do the processing, set the status code, return entity
|
|
||||||
// DoGet(rdata);
|
|
||||||
// break;
|
|
||||||
|
|
||||||
// case Rest.PUT : // Update named element
|
|
||||||
// DoUpdate(rdata);
|
|
||||||
// break;
|
|
||||||
|
|
||||||
// case Rest.POST : // Add new information to identified context.
|
|
||||||
// DoExtend(rdata);
|
|
||||||
// break;
|
|
||||||
|
|
||||||
// case Rest.DELETE : // Delete information
|
|
||||||
// DoDelete(rdata);
|
|
||||||
// break;
|
|
||||||
|
|
||||||
// default :
|
|
||||||
// Rest.Log.WarnFormat("{0} Method {1} not supported for {2}",
|
|
||||||
// MsgId, rdata.method, rdata.path);
|
|
||||||
// rdata.Fail(Rest.HttpStatusCodeMethodNotAllowed,
|
|
||||||
// String.Format("{0} not supported", rdata.method));
|
|
||||||
// break;
|
|
||||||
//}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion Interface
|
|
||||||
|
|
||||||
#region method-specific processing
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This method implements GET processing for user's appearance.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name=rdata>HTTP service request work area</param>
|
|
||||||
|
|
||||||
// private void DoGet(AppearanceRequestData rdata)
|
|
||||||
// {
|
|
||||||
// AvatarData adata = Rest.AvatarServices.GetAvatar(rdata.userProfile.ID);
|
|
||||||
//
|
|
||||||
// if (adata == null)
|
|
||||||
// {
|
|
||||||
// rdata.Fail(Rest.HttpStatusCodeNoContent,
|
|
||||||
// String.Format("appearance data not found for user {0} {1}",
|
|
||||||
// rdata.userProfile.FirstName, rdata.userProfile.SurName));
|
|
||||||
// }
|
|
||||||
// rdata.userAppearance = adata.ToAvatarAppearance(rdata.userProfile.ID);
|
|
||||||
//
|
|
||||||
// rdata.initXmlWriter();
|
|
||||||
//
|
|
||||||
// FormatUserAppearance(rdata);
|
|
||||||
//
|
|
||||||
// // Indicate a successful request
|
|
||||||
//
|
|
||||||
// rdata.Complete();
|
|
||||||
//
|
|
||||||
// // Send the response to the user. The body will be implicitly
|
|
||||||
// // constructed from the result of the XML writer.
|
|
||||||
//
|
|
||||||
// rdata.Respond(String.Format("Appearance {0} Normal completion", rdata.method));
|
|
||||||
// }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// POST adds NEW information to the user profile database.
|
|
||||||
/// This effectively resets the appearance before applying those
|
|
||||||
/// characteristics supplied in the request.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
// private void DoExtend(AppearanceRequestData rdata)
|
|
||||||
// {
|
|
||||||
//
|
|
||||||
// bool created = false;
|
|
||||||
// bool modified = false;
|
|
||||||
// string newnode = String.Empty;
|
|
||||||
//
|
|
||||||
// Rest.Log.DebugFormat("{0} POST ENTRY", MsgId);
|
|
||||||
//
|
|
||||||
// //AvatarAppearance old = Rest.AvatarServices.GetUserAppearance(rdata.userProfile.ID);
|
|
||||||
//
|
|
||||||
// rdata.userAppearance = new AvatarAppearance();
|
|
||||||
//
|
|
||||||
// // Although the following behavior is admitted by HTTP I am becoming
|
|
||||||
// // increasingly doubtful that it is appropriate for REST. If I attempt to
|
|
||||||
// // add a new record, and it already exists, then it seems to me that the
|
|
||||||
// // attempt should fail, rather than update the existing record.
|
|
||||||
// AvatarData adata = null;
|
|
||||||
// if (GetUserAppearance(rdata))
|
|
||||||
// {
|
|
||||||
// modified = rdata.userAppearance != null;
|
|
||||||
// created = !modified;
|
|
||||||
// adata = new AvatarData(rdata.userAppearance);
|
|
||||||
// Rest.AvatarServices.SetAvatar(rdata.userProfile.ID, adata);
|
|
||||||
// // Rest.UserServices.UpdateUserProfile(rdata.userProfile);
|
|
||||||
// }
|
|
||||||
// else
|
|
||||||
// {
|
|
||||||
// created = true;
|
|
||||||
// adata = new AvatarData(rdata.userAppearance);
|
|
||||||
// Rest.AvatarServices.SetAvatar(rdata.userProfile.ID, adata);
|
|
||||||
// // Rest.UserServices.UpdateUserProfile(rdata.userProfile);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// if (created)
|
|
||||||
// {
|
|
||||||
// newnode = String.Format("{0} {1}", rdata.userProfile.FirstName,
|
|
||||||
// rdata.userProfile.SurName);
|
|
||||||
// // Must include a location header with a URI that identifies the new resource.
|
|
||||||
//
|
|
||||||
// rdata.AddHeader(Rest.HttpHeaderLocation,String.Format("http://{0}{1}:{2}{3}{4}",
|
|
||||||
// rdata.hostname,rdata.port,rdata.path,Rest.UrlPathSeparator, newnode));
|
|
||||||
// rdata.Complete(Rest.HttpStatusCodeCreated);
|
|
||||||
//
|
|
||||||
// }
|
|
||||||
// else
|
|
||||||
// {
|
|
||||||
// if (modified)
|
|
||||||
// {
|
|
||||||
// rdata.Complete(Rest.HttpStatusCodeOK);
|
|
||||||
// }
|
|
||||||
// else
|
|
||||||
// {
|
|
||||||
// rdata.Complete(Rest.HttpStatusCodeNoContent);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// rdata.Respond(String.Format("Appearance {0} : Normal completion", rdata.method));
|
|
||||||
//
|
|
||||||
// }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This updates the user's appearance. not all aspects need to be provided,
|
|
||||||
/// only those supplied will be changed.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
// private void DoUpdate(AppearanceRequestData rdata)
|
|
||||||
// {
|
|
||||||
//
|
|
||||||
// // REFACTORING PROBLEM This was commented out. It doesn't work for 0.7
|
|
||||||
//
|
|
||||||
// //bool created = false;
|
|
||||||
// //bool modified = false;
|
|
||||||
//
|
|
||||||
//
|
|
||||||
// //rdata.userAppearance = Rest.AvatarServices.GetUserAppearance(rdata.userProfile.ID);
|
|
||||||
//
|
|
||||||
// //// If the user exists then this is considered a modification regardless
|
|
||||||
// //// of what may, or may not be, specified in the payload.
|
|
||||||
//
|
|
||||||
// //if (rdata.userAppearance != null)
|
|
||||||
// //{
|
|
||||||
// // modified = true;
|
|
||||||
// // Rest.AvatarServices.UpdateUserAppearance(rdata.userProfile.ID, rdata.userAppearance);
|
|
||||||
// // Rest.UserServices.UpdateUserProfile(rdata.userProfile);
|
|
||||||
// //}
|
|
||||||
//
|
|
||||||
// //if (created)
|
|
||||||
// //{
|
|
||||||
// // rdata.Complete(Rest.HttpStatusCodeCreated);
|
|
||||||
// //}
|
|
||||||
// //else
|
|
||||||
// //{
|
|
||||||
// // if (modified)
|
|
||||||
// // {
|
|
||||||
// // rdata.Complete(Rest.HttpStatusCodeOK);
|
|
||||||
// // }
|
|
||||||
// // else
|
|
||||||
// // {
|
|
||||||
// // rdata.Complete(Rest.HttpStatusCodeNoContent);
|
|
||||||
// // }
|
|
||||||
// //}
|
|
||||||
//
|
|
||||||
// rdata.Respond(String.Format("Appearance {0} : Normal completion", rdata.method));
|
|
||||||
//
|
|
||||||
// }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Delete the specified user's appearance. This actually performs a reset
|
|
||||||
/// to the default avatar appearance, if the info is already there.
|
|
||||||
/// Existing ownership is preserved. All prior updates are lost and can not
|
|
||||||
/// be recovered.
|
|
||||||
/// </summary>
|
|
||||||
// private void DoDelete(AppearanceRequestData rdata)
|
|
||||||
// {
|
|
||||||
// AvatarData adata = Rest.AvatarServices.GetAvatar(rdata.userProfile.ID);
|
|
||||||
//
|
|
||||||
// if (adata != null)
|
|
||||||
// {
|
|
||||||
// AvatarAppearance old = adata.ToAvatarAppearance(rdata.userProfile.ID);
|
|
||||||
// rdata.userAppearance = new AvatarAppearance();
|
|
||||||
// rdata.userAppearance.Owner = old.Owner;
|
|
||||||
// adata = new AvatarData(rdata.userAppearance);
|
|
||||||
//
|
|
||||||
// Rest.AvatarServices.SetAvatar(rdata.userProfile.ID, adata);
|
|
||||||
//
|
|
||||||
// rdata.Complete();
|
|
||||||
// }
|
|
||||||
// else
|
|
||||||
// {
|
|
||||||
//
|
|
||||||
// rdata.Complete(Rest.HttpStatusCodeNoContent);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// rdata.Respond(String.Format("Appearance {0} : Normal completion", rdata.method));
|
|
||||||
// }
|
|
||||||
|
|
||||||
#endregion method-specific processing
|
|
||||||
|
|
||||||
private bool GetUserAppearance(AppearanceRequestData rdata)
|
|
||||||
{
|
|
||||||
|
|
||||||
XmlReader xml;
|
|
||||||
bool indata = false;
|
|
||||||
|
|
||||||
rdata.initXmlReader();
|
|
||||||
xml = rdata.reader;
|
|
||||||
|
|
||||||
while (xml.Read())
|
|
||||||
{
|
|
||||||
switch (xml.NodeType)
|
|
||||||
{
|
|
||||||
case XmlNodeType.Element :
|
|
||||||
switch (xml.Name)
|
|
||||||
{
|
|
||||||
case "Appearance" :
|
|
||||||
if (xml.MoveToAttribute("Height"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.AvatarHeight = (float) Convert.ToDouble(xml.Value);
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
// if (xml.MoveToAttribute("Owner"))
|
|
||||||
// {
|
|
||||||
// rdata.userAppearance.Owner = (UUID)xml.Value;
|
|
||||||
// indata = true;
|
|
||||||
// }
|
|
||||||
if (xml.MoveToAttribute("Serial"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.Serial = Convert.ToInt32(xml.Value);
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
/*
|
|
||||||
case "Body" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.BodyItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.BodyAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Skin" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.SkinItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.SkinAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Hair" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.HairItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.HairAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Eyes" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.EyesItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.EyesAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Shirt" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.ShirtItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.ShirtAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Pants" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.PantsItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.PantsAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Shoes" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.ShoesItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.ShoesAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Socks" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.SocksItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.SocksAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Jacket" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.JacketItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.JacketAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Gloves" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.GlovesItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.GlovesAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "UnderShirt" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.UnderShirtItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.UnderShirtAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "UnderPants" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.UnderPantsItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.UnderPantsAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Skirt" :
|
|
||||||
if (xml.MoveToAttribute("Item"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.SkirtItem = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.SkirtAsset = (UUID)xml.Value;
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
*/
|
|
||||||
case "Attachment" :
|
|
||||||
{
|
|
||||||
|
|
||||||
int ap;
|
|
||||||
UUID asset;
|
|
||||||
UUID item;
|
|
||||||
|
|
||||||
if (xml.MoveToAttribute("AtPoint"))
|
|
||||||
{
|
|
||||||
ap = Convert.ToInt32(xml.Value);
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
asset = new UUID(xml.Value);
|
|
||||||
if (xml.MoveToAttribute("Asset"))
|
|
||||||
{
|
|
||||||
item = new UUID(xml.Value);
|
|
||||||
rdata.userAppearance.SetAttachment(ap, item, asset);
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Texture" :
|
|
||||||
if (xml.MoveToAttribute("Default"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.Texture = new Primitive.TextureEntry(new UUID(xml.Value));
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "Face" :
|
|
||||||
{
|
|
||||||
uint index;
|
|
||||||
if (xml.MoveToAttribute("Index"))
|
|
||||||
{
|
|
||||||
index = Convert.ToUInt32(xml.Value);
|
|
||||||
if (xml.MoveToAttribute("Id"))
|
|
||||||
{
|
|
||||||
rdata.userAppearance.Texture.CreateFace(index).TextureID = new UUID(xml.Value);
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case "VisualParameters" :
|
|
||||||
{
|
|
||||||
xml.ReadContentAsBase64(rdata.userAppearance.VisualParams,
|
|
||||||
0, rdata.userAppearance.VisualParams.Length);
|
|
||||||
indata = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return indata;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private void FormatPart(AppearanceRequestData rdata, string part, UUID item, UUID asset)
|
|
||||||
{
|
|
||||||
if (item != UUID.Zero || asset != UUID.Zero)
|
|
||||||
{
|
|
||||||
rdata.writer.WriteStartElement(part);
|
|
||||||
if (item != UUID.Zero)
|
|
||||||
{
|
|
||||||
rdata.writer.WriteAttributeString("Item",item.ToString());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (asset != UUID.Zero)
|
|
||||||
{
|
|
||||||
rdata.writer.WriteAttributeString("Asset",asset.ToString());
|
|
||||||
}
|
|
||||||
rdata.writer.WriteEndElement();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void FormatUserAppearance(AppearanceRequestData rdata)
|
|
||||||
{
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} FormatUserAppearance", MsgId);
|
|
||||||
|
|
||||||
if (rdata.userAppearance != null)
|
|
||||||
{
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} FormatUserAppearance: appearance object exists", MsgId);
|
|
||||||
rdata.writer.WriteStartElement("Appearance");
|
|
||||||
|
|
||||||
rdata.writer.WriteAttributeString("Height", rdata.userAppearance.AvatarHeight.ToString());
|
|
||||||
// if (rdata.userAppearance.Owner != UUID.Zero)
|
|
||||||
// rdata.writer.WriteAttributeString("Owner", rdata.userAppearance.Owner.ToString());
|
|
||||||
rdata.writer.WriteAttributeString("Serial", rdata.userAppearance.Serial.ToString());
|
|
||||||
|
|
||||||
/*
|
|
||||||
FormatPart(rdata, "Body", rdata.userAppearance.BodyItem, rdata.userAppearance.BodyAsset);
|
|
||||||
FormatPart(rdata, "Skin", rdata.userAppearance.SkinItem, rdata.userAppearance.SkinAsset);
|
|
||||||
FormatPart(rdata, "Hair", rdata.userAppearance.HairItem, rdata.userAppearance.HairAsset);
|
|
||||||
FormatPart(rdata, "Eyes", rdata.userAppearance.EyesItem, rdata.userAppearance.EyesAsset);
|
|
||||||
|
|
||||||
FormatPart(rdata, "Shirt", rdata.userAppearance.ShirtItem, rdata.userAppearance.ShirtAsset);
|
|
||||||
FormatPart(rdata, "Pants", rdata.userAppearance.PantsItem, rdata.userAppearance.PantsAsset);
|
|
||||||
FormatPart(rdata, "Skirt", rdata.userAppearance.SkirtItem, rdata.userAppearance.SkirtAsset);
|
|
||||||
FormatPart(rdata, "Shoes", rdata.userAppearance.ShoesItem, rdata.userAppearance.ShoesAsset);
|
|
||||||
FormatPart(rdata, "Socks", rdata.userAppearance.SocksItem, rdata.userAppearance.SocksAsset);
|
|
||||||
|
|
||||||
FormatPart(rdata, "Jacket", rdata.userAppearance.JacketItem, rdata.userAppearance.JacketAsset);
|
|
||||||
FormatPart(rdata, "Gloves", rdata.userAppearance.GlovesItem, rdata.userAppearance.GlovesAsset);
|
|
||||||
|
|
||||||
FormatPart(rdata, "UnderShirt", rdata.userAppearance.UnderShirtItem, rdata.userAppearance.UnderShirtAsset);
|
|
||||||
FormatPart(rdata, "UnderPants", rdata.userAppearance.UnderPantsItem, rdata.userAppearance.UnderPantsAsset);
|
|
||||||
*/
|
|
||||||
Rest.Log.DebugFormat("{0} FormatUserAppearance: Formatting attachments", MsgId);
|
|
||||||
|
|
||||||
rdata.writer.WriteStartElement("Attachments");
|
|
||||||
List<AvatarAttachment> attachments = rdata.userAppearance.GetAttachments();
|
|
||||||
foreach (AvatarAttachment attach in attachments)
|
|
||||||
{
|
|
||||||
rdata.writer.WriteStartElement("Attachment");
|
|
||||||
rdata.writer.WriteAttributeString("AtPoint", attach.AttachPoint.ToString());
|
|
||||||
rdata.writer.WriteAttributeString("Item", attach.ItemID.ToString());
|
|
||||||
rdata.writer.WriteAttributeString("Asset", attach.AssetID.ToString());
|
|
||||||
rdata.writer.WriteEndElement();
|
|
||||||
}
|
|
||||||
rdata.writer.WriteEndElement();
|
|
||||||
|
|
||||||
Primitive.TextureEntry texture = rdata.userAppearance.Texture;
|
|
||||||
|
|
||||||
if (texture != null && (texture.DefaultTexture != null || texture.FaceTextures != null))
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} FormatUserAppearance: Formatting textures", MsgId);
|
|
||||||
|
|
||||||
rdata.writer.WriteStartElement("Texture");
|
|
||||||
|
|
||||||
if (texture.DefaultTexture != null)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} FormatUserAppearance: Formatting default texture", MsgId);
|
|
||||||
rdata.writer.WriteAttributeString("Default",
|
|
||||||
texture.DefaultTexture.TextureID.ToString());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (texture.FaceTextures != null)
|
|
||||||
{
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} FormatUserAppearance: Formatting face textures", MsgId);
|
|
||||||
|
|
||||||
for (int i=0; i<texture.FaceTextures.Length;i++)
|
|
||||||
{
|
|
||||||
if (texture.FaceTextures[i] != null)
|
|
||||||
{
|
|
||||||
rdata.writer.WriteStartElement("Face");
|
|
||||||
rdata.writer.WriteAttributeString("Index", i.ToString());
|
|
||||||
rdata.writer.WriteAttributeString("Id",
|
|
||||||
texture.FaceTextures[i].TextureID.ToString());
|
|
||||||
rdata.writer.WriteEndElement();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.writer.WriteEndElement();
|
|
||||||
}
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} FormatUserAppearance: Formatting visual parameters", MsgId);
|
|
||||||
|
|
||||||
rdata.writer.WriteStartElement("VisualParameters");
|
|
||||||
rdata.writer.WriteBase64(rdata.userAppearance.VisualParams,0,
|
|
||||||
rdata.userAppearance.VisualParams.Length);
|
|
||||||
rdata.writer.WriteEndElement();
|
|
||||||
rdata.writer.WriteFullEndElement();
|
|
||||||
}
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} FormatUserAppearance: completed", MsgId);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
#region appearance RequestData extension
|
|
||||||
|
|
||||||
internal class AppearanceRequestData : RequestData
|
|
||||||
{
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// These are the inventory specific request/response state
|
|
||||||
/// extensions.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal UUID uuid = UUID.Zero;
|
|
||||||
internal UserProfileData userProfile = null;
|
|
||||||
internal AvatarAppearance userAppearance = null;
|
|
||||||
|
|
||||||
internal AppearanceRequestData(OSHttpRequest request, OSHttpResponse response, string prefix)
|
|
||||||
: base(request, response, prefix)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion Appearance RequestData extension
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,383 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Xml;
|
|
||||||
using OpenMetaverse;
|
|
||||||
using OpenSim.Framework;
|
|
||||||
using OpenSim.Framework.Servers;
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Inventory
|
|
||||||
{
|
|
||||||
public class RestAssetServices : IRest
|
|
||||||
{
|
|
||||||
private bool enabled = false;
|
|
||||||
private string qPrefix = "assets";
|
|
||||||
|
|
||||||
// A simple constructor is used to handle any once-only
|
|
||||||
// initialization of working classes.
|
|
||||||
|
|
||||||
public RestAssetServices()
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} Asset services initializing", MsgId);
|
|
||||||
Rest.Log.InfoFormat("{0} Using REST Implementation Version {1}", MsgId, Rest.Version);
|
|
||||||
|
|
||||||
// If the handler specifies a relative path for its domain
|
|
||||||
// then we must add the standard absolute prefix, e.g. /admin
|
|
||||||
|
|
||||||
if (!qPrefix.StartsWith(Rest.UrlPathSeparator))
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} Prefixing domain name ({1})", MsgId, qPrefix);
|
|
||||||
qPrefix = String.Format("{0}{1}{2}", Rest.Prefix, Rest.UrlPathSeparator, qPrefix);
|
|
||||||
Rest.Log.InfoFormat("{0} Fully qualified domain name is <{1}>", MsgId, qPrefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Register interface using the fully-qualified prefix
|
|
||||||
|
|
||||||
Rest.Plugin.AddPathHandler(DoAsset, qPrefix, Allocate);
|
|
||||||
|
|
||||||
// Activate if all went OK
|
|
||||||
|
|
||||||
enabled = true;
|
|
||||||
|
|
||||||
Rest.Log.InfoFormat("{0} Asset services initialization complete", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Post-construction, pre-enabled initialization opportunity
|
|
||||||
// Not currently exploited.
|
|
||||||
|
|
||||||
public void Initialize()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// Called by the plug-in to halt REST processing. Local processing is
|
|
||||||
// disabled, and control blocks until all current processing has
|
|
||||||
// completed. No new processing will be started
|
|
||||||
|
|
||||||
public void Close()
|
|
||||||
{
|
|
||||||
enabled = false;
|
|
||||||
Rest.Log.InfoFormat("{0} Asset services ({1}) closing down", MsgId, qPrefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Properties
|
|
||||||
|
|
||||||
internal string MsgId
|
|
||||||
{
|
|
||||||
get { return Rest.MsgId; }
|
|
||||||
}
|
|
||||||
|
|
||||||
#region Interface
|
|
||||||
|
|
||||||
private RequestData Allocate(OSHttpRequest request, OSHttpResponse response, string prefix)
|
|
||||||
{
|
|
||||||
return (RequestData) new AssetRequestData(request, response, prefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Asset Handler
|
|
||||||
|
|
||||||
private void DoAsset(RequestData rparm)
|
|
||||||
{
|
|
||||||
if (!enabled) return;
|
|
||||||
|
|
||||||
AssetRequestData rdata = (AssetRequestData) rparm;
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST Asset handler ({1}) ENTRY", MsgId, qPrefix);
|
|
||||||
|
|
||||||
// Now that we know this is a serious attempt to
|
|
||||||
// access inventory data, we should find out who
|
|
||||||
// is asking, and make sure they are authorized
|
|
||||||
// to do so. We need to validate the caller's
|
|
||||||
// identity before revealing anything about the
|
|
||||||
// status quo. Authenticate throws an exception
|
|
||||||
// via Fail if no identity information is present.
|
|
||||||
//
|
|
||||||
// With the present HTTP server we can't use the
|
|
||||||
// builtin authentication mechanisms because they
|
|
||||||
// would be enforced for all in-bound requests.
|
|
||||||
// Instead we look at the headers ourselves and
|
|
||||||
// handle authentication directly.
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (!rdata.IsAuthenticated)
|
|
||||||
{
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotAuthorized, String.Format("user \"{0}\" could not be authenticated"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (RestException e)
|
|
||||||
{
|
|
||||||
if (e.statusCode == Rest.HttpStatusCodeNotAuthorized)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} User not authenticated", MsgId);
|
|
||||||
Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId,
|
|
||||||
rdata.request.Headers.Get("Authorization"));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.ErrorFormat("{0} User authentication failed", MsgId);
|
|
||||||
Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId,
|
|
||||||
rdata.request.Headers.Get("Authorization"));
|
|
||||||
}
|
|
||||||
throw (e);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove the prefix and what's left are the parameters. If we don't have
|
|
||||||
// the parameters we need, fail the request. Parameters do NOT include
|
|
||||||
// any supplied query values.
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length > 0)
|
|
||||||
{
|
|
||||||
switch (rdata.method)
|
|
||||||
{
|
|
||||||
case "get" :
|
|
||||||
DoGet(rdata);
|
|
||||||
break;
|
|
||||||
case "put" :
|
|
||||||
DoPut(rdata);
|
|
||||||
break;
|
|
||||||
case "post" :
|
|
||||||
DoPost(rdata);
|
|
||||||
break;
|
|
||||||
case "delete" :
|
|
||||||
default :
|
|
||||||
Rest.Log.WarnFormat("{0} Asset: Method not supported: {1}",
|
|
||||||
MsgId, rdata.method);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest,String.Format("method <{0}> not supported", rdata.method));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} Asset: No agent information provided", MsgId);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest, "no agent information provided");
|
|
||||||
}
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST Asset handler EXIT", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion Interface
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The only parameter we recognize is a UUID.If an asset with this identification is
|
|
||||||
/// found, it's content, base-64 encoded, is returned to the client.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
private void DoGet(AssetRequestData rdata)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} REST Asset handler, Method = <{1}> ENTRY", MsgId, rdata.method);
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length == 1)
|
|
||||||
{
|
|
||||||
UUID uuid = new UUID(rdata.Parameters[0]);
|
|
||||||
AssetBase asset = Rest.AssetServices.Get(uuid.ToString());
|
|
||||||
|
|
||||||
if (asset != null)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Asset located <{1}>", MsgId, rdata.Parameters[0]);
|
|
||||||
|
|
||||||
rdata.initXmlWriter();
|
|
||||||
|
|
||||||
rdata.writer.WriteStartElement(String.Empty,"Asset",String.Empty);
|
|
||||||
|
|
||||||
rdata.writer.WriteAttributeString("id", asset.ID);
|
|
||||||
rdata.writer.WriteAttributeString("name", asset.Name);
|
|
||||||
rdata.writer.WriteAttributeString("desc", asset.Description);
|
|
||||||
rdata.writer.WriteAttributeString("type", asset.Type.ToString());
|
|
||||||
rdata.writer.WriteAttributeString("local", asset.Local.ToString());
|
|
||||||
rdata.writer.WriteAttributeString("temporary", asset.Temporary.ToString());
|
|
||||||
|
|
||||||
rdata.writer.WriteBase64(asset.Data,0,asset.Data.Length);
|
|
||||||
|
|
||||||
rdata.writer.WriteFullEndElement();
|
|
||||||
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid parameters: <{1}>", MsgId, rdata.path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotFound, "invalid parameters");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.Complete();
|
|
||||||
rdata.Respond(String.Format("Asset <{0}> : Normal completion", rdata.method));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// UPDATE existing item, if it exists. URI identifies the item in question.
|
|
||||||
/// The only parameter we recognize is a UUID. The enclosed asset data (base-64 encoded)
|
|
||||||
/// is decoded and stored in the database, identified by the supplied UUID.
|
|
||||||
/// </summary>
|
|
||||||
private void DoPut(AssetRequestData rdata)
|
|
||||||
{
|
|
||||||
bool modified = false;
|
|
||||||
bool created = false;
|
|
||||||
|
|
||||||
AssetBase asset = null;
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST Asset handler, Method = <{1}> ENTRY", MsgId, rdata.method);
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length == 1)
|
|
||||||
{
|
|
||||||
|
|
||||||
rdata.initXmlReader();
|
|
||||||
XmlReader xml = rdata.reader;
|
|
||||||
|
|
||||||
if (!xml.ReadToFollowing("Asset"))
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid request data: <{1}>", MsgId, rdata.path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest,"invalid request data");
|
|
||||||
}
|
|
||||||
|
|
||||||
UUID uuid = new UUID(rdata.Parameters[0]);
|
|
||||||
asset = Rest.AssetServices.Get(uuid.ToString());
|
|
||||||
|
|
||||||
modified = (asset != null);
|
|
||||||
created = !modified;
|
|
||||||
|
|
||||||
asset = new AssetBase(uuid, xml.GetAttribute("name"), SByte.Parse(xml.GetAttribute("type")), UUID.Zero.ToString());
|
|
||||||
asset.Description = xml.GetAttribute("desc");
|
|
||||||
asset.Local = Int32.Parse(xml.GetAttribute("local")) != 0;
|
|
||||||
asset.Temporary = Int32.Parse(xml.GetAttribute("temporary")) != 0;
|
|
||||||
asset.Data = Convert.FromBase64String(xml.ReadElementContentAsString("Asset", ""));
|
|
||||||
|
|
||||||
if (asset.ID != rdata.Parameters[0])
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} URI and payload disagree on UUID U:{1} vs P:{2}",
|
|
||||||
MsgId, rdata.Parameters[0], asset.ID);
|
|
||||||
}
|
|
||||||
|
|
||||||
Rest.AssetServices.Store(asset);
|
|
||||||
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid parameters: <{1}>", MsgId, rdata.path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotFound, "invalid parameters");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (created)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Created asset {0}, UUID {1} <p>", asset.Name, asset.FullID));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeCreated);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (modified)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Modified asset {0}, UUID {1} <p>", asset.Name, asset.FullID));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeOK);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeNoContent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.Respond(String.Format("Asset {0} : Normal completion", rdata.method));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// CREATE new item, replace if it exists. URI identifies the context for the item in question.
|
|
||||||
/// No parameters are required for POST, just thepayload.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
private void DoPost(AssetRequestData rdata)
|
|
||||||
{
|
|
||||||
|
|
||||||
bool modified = false;
|
|
||||||
bool created = false;
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST Asset handler, Method = <{1}> ENTRY", MsgId, rdata.method);
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length != 0)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} Parameters ignored <{1}>", MsgId, rdata.path);
|
|
||||||
Rest.Log.InfoFormat("{0} POST of an asset has no parameters", MsgId, rdata.path);
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.initXmlReader();
|
|
||||||
XmlReader xml = rdata.reader;
|
|
||||||
|
|
||||||
if (!xml.ReadToFollowing("Asset"))
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid request data: <{1}>", MsgId, rdata.path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest,"invalid request data");
|
|
||||||
}
|
|
||||||
|
|
||||||
UUID uuid = new UUID(xml.GetAttribute("id"));
|
|
||||||
AssetBase asset = Rest.AssetServices.Get(uuid.ToString());
|
|
||||||
|
|
||||||
modified = (asset != null);
|
|
||||||
created = !modified;
|
|
||||||
|
|
||||||
asset = new AssetBase(uuid, xml.GetAttribute("name"), SByte.Parse(xml.GetAttribute("type")), UUID.Zero.ToString());
|
|
||||||
asset.Description = xml.GetAttribute("desc");
|
|
||||||
asset.Local = Int32.Parse(xml.GetAttribute("local")) != 0;
|
|
||||||
asset.Temporary = Int32.Parse(xml.GetAttribute("temporary")) != 0;
|
|
||||||
asset.Data = Convert.FromBase64String(xml.ReadElementContentAsString("Asset", ""));
|
|
||||||
|
|
||||||
Rest.AssetServices.Store(asset);
|
|
||||||
|
|
||||||
if (created)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Created asset {0}, UUID {1} <p>", asset.Name, asset.FullID));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeCreated);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (modified)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Modified asset {0}, UUID {1} <p>", asset.Name, asset.FullID));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeOK);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeNoContent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.Respond(String.Format("Asset {0} : Normal completion", rdata.method));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Asset processing has no special data area requirements.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal class AssetRequestData : RequestData
|
|
||||||
{
|
|
||||||
internal AssetRequestData(OSHttpRequest request, OSHttpResponse response, string prefix)
|
|
||||||
: base(request, response, prefix)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,448 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Xml;
|
|
||||||
using System.IO;
|
|
||||||
using OpenMetaverse;
|
|
||||||
using OpenSim.Framework;
|
|
||||||
using OpenSim.Framework.Servers;
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Inventory
|
|
||||||
{
|
|
||||||
public class RestFileServices : IRest
|
|
||||||
{
|
|
||||||
private bool enabled = false;
|
|
||||||
private string qPrefix = "files";
|
|
||||||
|
|
||||||
// A simple constructor is used to handle any once-only
|
|
||||||
// initialization of working classes.
|
|
||||||
|
|
||||||
public RestFileServices()
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} File services initializing", MsgId);
|
|
||||||
Rest.Log.InfoFormat("{0} Using REST Implementation Version {1}", MsgId, Rest.Version);
|
|
||||||
|
|
||||||
// If the handler specifies a relative path for its domain
|
|
||||||
// then we must add the standard absolute prefix, e.g. /admin
|
|
||||||
|
|
||||||
if (!qPrefix.StartsWith(Rest.UrlPathSeparator))
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} Prefixing domain name ({1})", MsgId, qPrefix);
|
|
||||||
qPrefix = String.Format("{0}{1}{2}", Rest.Prefix, Rest.UrlPathSeparator, qPrefix);
|
|
||||||
Rest.Log.InfoFormat("{0} Fully qualified domain name is <{1}>", MsgId, qPrefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Register interface using the fully-qualified prefix
|
|
||||||
|
|
||||||
Rest.Plugin.AddPathHandler(DoFile, qPrefix, Allocate);
|
|
||||||
|
|
||||||
// Activate if all went OK
|
|
||||||
|
|
||||||
enabled = true;
|
|
||||||
|
|
||||||
Rest.Log.InfoFormat("{0} File services initialization complete", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Post-construction, pre-enabled initialization opportunity
|
|
||||||
// Not currently exploited.
|
|
||||||
|
|
||||||
public void Initialize()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// Called by the plug-in to halt REST processing. Local processing is
|
|
||||||
// disabled, and control blocks until all current processing has
|
|
||||||
// completed. No new processing will be started
|
|
||||||
|
|
||||||
public void Close()
|
|
||||||
{
|
|
||||||
enabled = false;
|
|
||||||
Rest.Log.InfoFormat("{0} File services ({1}) closing down", MsgId, qPrefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Properties
|
|
||||||
|
|
||||||
internal string MsgId
|
|
||||||
{
|
|
||||||
get { return Rest.MsgId; }
|
|
||||||
}
|
|
||||||
|
|
||||||
#region Interface
|
|
||||||
|
|
||||||
private RequestData Allocate(OSHttpRequest request, OSHttpResponse response, string prefix)
|
|
||||||
{
|
|
||||||
return (RequestData) new FileRequestData(request, response, prefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Asset Handler
|
|
||||||
|
|
||||||
private void DoFile(RequestData rparm)
|
|
||||||
{
|
|
||||||
if (!enabled) return;
|
|
||||||
|
|
||||||
FileRequestData rdata = (FileRequestData) rparm;
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST File handler ({1}) ENTRY", MsgId, qPrefix);
|
|
||||||
|
|
||||||
// Now that we know this is a serious attempt to
|
|
||||||
// access file data, we should find out who
|
|
||||||
// is asking, and make sure they are authorized
|
|
||||||
// to do so. We need to validate the caller's
|
|
||||||
// identity before revealing anything about the
|
|
||||||
// status quo. Authenticate throws an exception
|
|
||||||
// via Fail if no identity information is present.
|
|
||||||
//
|
|
||||||
// With the present HTTP server we can't use the
|
|
||||||
// builtin authentication mechanisms because they
|
|
||||||
// would be enforced for all in-bound requests.
|
|
||||||
// Instead we look at the headers ourselves and
|
|
||||||
// handle authentication directly.
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (!rdata.IsAuthenticated)
|
|
||||||
{
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotAuthorized, String.Format("user \"{0}\" could not be authenticated"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (RestException e)
|
|
||||||
{
|
|
||||||
if (e.statusCode == Rest.HttpStatusCodeNotAuthorized)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} User not authenticated", MsgId);
|
|
||||||
Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId,
|
|
||||||
rdata.request.Headers.Get("Authorization"));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.ErrorFormat("{0} User authentication failed", MsgId);
|
|
||||||
Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId,
|
|
||||||
rdata.request.Headers.Get("Authorization"));
|
|
||||||
}
|
|
||||||
throw (e);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove the prefix and what's left are the parameters. If we don't have
|
|
||||||
// the parameters we need, fail the request. Parameters do NOT include
|
|
||||||
// any supplied query values.
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length > 0)
|
|
||||||
{
|
|
||||||
switch (rdata.method)
|
|
||||||
{
|
|
||||||
case "get" :
|
|
||||||
DoGet(rdata);
|
|
||||||
break;
|
|
||||||
case "put" :
|
|
||||||
DoPut(rdata);
|
|
||||||
break;
|
|
||||||
case "post" :
|
|
||||||
DoPost(rdata);
|
|
||||||
break;
|
|
||||||
case "delete" :
|
|
||||||
DoDelete(rdata);
|
|
||||||
break;
|
|
||||||
default :
|
|
||||||
Rest.Log.WarnFormat("{0} File: Method not supported: {1}",
|
|
||||||
MsgId, rdata.method);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest,String.Format("method <{0}> not supported", rdata.method));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} File: No agent information provided", MsgId);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest, "no agent information provided");
|
|
||||||
}
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST File handler EXIT", MsgId);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion Interface
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The only parameter we recognize is a UUID.If an asset with this identification is
|
|
||||||
/// found, it's content, base-64 encoded, is returned to the client.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
private void DoGet(FileRequestData rdata)
|
|
||||||
{
|
|
||||||
|
|
||||||
string path = String.Empty;
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST File handler, Method = <{1}> ENTRY", MsgId, rdata.method);
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length > 1)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
path = rdata.path.Substring(rdata.Parameters[0].Length+qPrefix.Length+2);
|
|
||||||
if (File.Exists(path))
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} File located <{1}>", MsgId, path);
|
|
||||||
Byte[] data = File.ReadAllBytes(path);
|
|
||||||
rdata.initXmlWriter();
|
|
||||||
rdata.writer.WriteStartElement(String.Empty,"File",String.Empty);
|
|
||||||
rdata.writer.WriteAttributeString("name", path);
|
|
||||||
rdata.writer.WriteBase64(data,0,data.Length);
|
|
||||||
rdata.writer.WriteFullEndElement();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid parameters: <{1}>", MsgId, path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotFound, String.Format("invalid parameters : {0}", path));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid parameters: <{1}>", MsgId, e.Message);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotFound, String.Format("invalid parameters : {0} {1}",
|
|
||||||
path, e.Message));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.Complete();
|
|
||||||
rdata.Respond(String.Format("File <{0}> : Normal completion", rdata.method));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// UPDATE existing item, if it exists. URI identifies the item in question.
|
|
||||||
/// The only parameter we recognize is a UUID. The enclosed asset data (base-64 encoded)
|
|
||||||
/// is decoded and stored in the database, identified by the supplied UUID.
|
|
||||||
/// </summary>
|
|
||||||
private void DoPut(FileRequestData rdata)
|
|
||||||
{
|
|
||||||
bool modified = false;
|
|
||||||
bool created = false;
|
|
||||||
string path = String.Empty;
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST File handler, Method = <{1}> ENTRY", MsgId, rdata.method);
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length > 1)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
path = rdata.path.Substring(rdata.Parameters[0].Length+qPrefix.Length+2);
|
|
||||||
bool maymod = File.Exists(path);
|
|
||||||
|
|
||||||
rdata.initXmlReader();
|
|
||||||
XmlReader xml = rdata.reader;
|
|
||||||
|
|
||||||
if (!xml.ReadToFollowing("File"))
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid request data: <{1}>", MsgId, rdata.path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest,"invalid request data");
|
|
||||||
}
|
|
||||||
|
|
||||||
Byte[] data = Convert.FromBase64String(xml.ReadElementContentAsString("File", ""));
|
|
||||||
|
|
||||||
File.WriteAllBytes(path,data);
|
|
||||||
modified = maymod;
|
|
||||||
created = ! maymod;
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Exception during file processing : {1}", MsgId,
|
|
||||||
e.Message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid parameters: <{1}>", MsgId, rdata.path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotFound, "invalid parameters");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (created)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Created file {0} <p>", path));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeCreated);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (modified)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Modified file {0} <p>", path));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeOK);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeNoContent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.Respond(String.Format("File {0} : Normal completion", rdata.method));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// CREATE new item, replace if it exists. URI identifies the context for the item in question.
|
|
||||||
/// No parameters are required for POST, just thepayload.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
private void DoPost(FileRequestData rdata)
|
|
||||||
{
|
|
||||||
|
|
||||||
bool modified = false;
|
|
||||||
bool created = false;
|
|
||||||
string path = String.Empty;
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST File handler, Method = <{1}> ENTRY", MsgId, rdata.method);
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length > 1)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
path = rdata.path.Substring(rdata.Parameters[0].Length+qPrefix.Length+2);
|
|
||||||
bool maymod = File.Exists(path);
|
|
||||||
|
|
||||||
rdata.initXmlReader();
|
|
||||||
XmlReader xml = rdata.reader;
|
|
||||||
|
|
||||||
if (!xml.ReadToFollowing("File"))
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid request data: <{1}>", MsgId, rdata.path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest,"invalid request data");
|
|
||||||
}
|
|
||||||
|
|
||||||
Byte[] data = Convert.FromBase64String(xml.ReadElementContentAsString("File", ""));
|
|
||||||
|
|
||||||
File.WriteAllBytes(path,data);
|
|
||||||
modified = maymod;
|
|
||||||
created = ! maymod;
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Exception during file processing : {1}", MsgId,
|
|
||||||
e.Message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid parameters: <{1}>", MsgId, rdata.path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotFound, "invalid parameters");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (created)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Created file {0} <p>", path));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeCreated);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (modified)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Modified file {0} <p>", path));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeOK);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeNoContent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.Respond(String.Format("File {0} : Normal completion", rdata.method));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// CREATE new item, replace if it exists. URI identifies the context for the item in question.
|
|
||||||
/// No parameters are required for POST, just thepayload.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
private void DoDelete(FileRequestData rdata)
|
|
||||||
{
|
|
||||||
|
|
||||||
bool modified = false;
|
|
||||||
bool created = false;
|
|
||||||
string path = String.Empty;
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST File handler, Method = <{1}> ENTRY", MsgId, rdata.method);
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length > 1)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
path = rdata.path.Substring(rdata.Parameters[0].Length+qPrefix.Length+2);
|
|
||||||
|
|
||||||
if (File.Exists(path))
|
|
||||||
{
|
|
||||||
File.Delete(path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Exception during file processing : {1}", MsgId,
|
|
||||||
e.Message);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotFound, String.Format("invalid parameters : {0} {1}",
|
|
||||||
path, e.Message));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Invalid parameters: <{1}>", MsgId, rdata.path);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotFound, "invalid parameters");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (created)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Created file {0} <p>", path));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeCreated);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (modified)
|
|
||||||
{
|
|
||||||
rdata.appendStatus(String.Format("<p> Modified file {0} <p>", path));
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeOK);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
rdata.Complete(Rest.HttpStatusCodeNoContent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.Respond(String.Format("File {0} : Normal completion", rdata.method));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// File processing has no special data area requirements.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal class FileRequestData : RequestData
|
|
||||||
{
|
|
||||||
internal FileRequestData(OSHttpRequest request, OSHttpResponse response, string prefix)
|
|
||||||
: base(request, response, prefix)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,662 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Reflection;
|
|
||||||
using OpenSim.Framework.Servers;
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Inventory
|
|
||||||
{
|
|
||||||
/// <remarks>
|
|
||||||
/// The class signature reveals the roles that RestHandler plays.
|
|
||||||
///
|
|
||||||
/// [1] It is a sub-class of RestPlugin. It inherits and extends
|
|
||||||
/// the functionality of this class, constraining it to the
|
|
||||||
/// specific needs of this REST implementation. This relates
|
|
||||||
/// to the plug-in mechanism supported by OpenSim, the specifics
|
|
||||||
/// of which are mostly hidden by RestPlugin.
|
|
||||||
/// [2] IRestHandler describes the interface that this class
|
|
||||||
/// exports to service implementations. This is the services
|
|
||||||
/// management interface.
|
|
||||||
/// [3] IHttpAgentHandler describes the interface that is exported
|
|
||||||
/// to the BaseHttpServer in support of this particular HTTP
|
|
||||||
/// processing model. This is the request interface of the
|
|
||||||
/// handler.
|
|
||||||
/// </remarks>
|
|
||||||
|
|
||||||
public class RestHandler : RestPlugin, IRestHandler, IHttpAgentHandler
|
|
||||||
{
|
|
||||||
// Handler tables: both stream and REST are supported. The path handlers and their
|
|
||||||
// respective allocators are stored in separate tables.
|
|
||||||
|
|
||||||
internal Dictionary<string,RestMethodHandler> pathHandlers = new Dictionary<string,RestMethodHandler>();
|
|
||||||
internal Dictionary<string,RestMethodAllocator> pathAllocators = new Dictionary<string,RestMethodAllocator>();
|
|
||||||
internal Dictionary<string,RestStreamHandler> streamHandlers = new Dictionary<string,RestStreamHandler>();
|
|
||||||
|
|
||||||
#region local static state
|
|
||||||
|
|
||||||
private static bool handlersLoaded = false;
|
|
||||||
private static List<Type> classes = new List<Type>();
|
|
||||||
private static List<IRest> handlers = new List<IRest>();
|
|
||||||
private static Type[] parms = new Type[0];
|
|
||||||
private static Object[] args = new Object[0];
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This static initializer scans the ASSEMBLY for classes that
|
|
||||||
/// export the IRest interface and builds a list of them. These
|
|
||||||
/// are later activated by the handler. To add a new handler it
|
|
||||||
/// is only necessary to create a new services class that implements
|
|
||||||
/// the IRest interface, and recompile the handler. This gives
|
|
||||||
/// all of the build-time flexibility of a modular approach
|
|
||||||
/// while not introducing yet-another module loader. Note that
|
|
||||||
/// multiple assembles can still be built, each with its own set
|
|
||||||
/// of handlers. Examples of services classes are RestInventoryServices
|
|
||||||
/// and RestSkeleton.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
static RestHandler()
|
|
||||||
{
|
|
||||||
Module[] mods = Assembly.GetExecutingAssembly().GetModules();
|
|
||||||
|
|
||||||
foreach (Module m in mods)
|
|
||||||
{
|
|
||||||
Type[] types = m.GetTypes();
|
|
||||||
foreach (Type t in types)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (t.GetInterface("IRest") != null)
|
|
||||||
{
|
|
||||||
classes.Add(t);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("[STATIC-HANDLER]: #0 Error scanning {1}", t);
|
|
||||||
Rest.Log.InfoFormat("[STATIC-HANDLER]: #0 {1} is not included", t);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion local static state
|
|
||||||
|
|
||||||
#region local instance state
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This routine loads all of the handlers discovered during
|
|
||||||
/// instance initialization.
|
|
||||||
/// A table of all loaded and successfully constructed handlers
|
|
||||||
/// is built, and this table is then used by the constructor to
|
|
||||||
/// initialize each of the handlers in turn.
|
|
||||||
/// NOTE: The loading process does not automatically imply that
|
|
||||||
/// the handler has registered any kind of an interface, that
|
|
||||||
/// may be (optionally) done by the handler either during
|
|
||||||
/// construction, or during initialization.
|
|
||||||
///
|
|
||||||
/// I was not able to make this code work within a constructor
|
|
||||||
/// so it is isolated within this method.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
private void LoadHandlers()
|
|
||||||
{
|
|
||||||
lock (handlers)
|
|
||||||
{
|
|
||||||
if (!handlersLoaded)
|
|
||||||
{
|
|
||||||
ConstructorInfo ci;
|
|
||||||
Object ht;
|
|
||||||
|
|
||||||
foreach (Type t in classes)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
ci = t.GetConstructor(parms);
|
|
||||||
ht = ci.Invoke(args);
|
|
||||||
handlers.Add((IRest)ht);
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} Unable to load {1} : {2}", MsgId, t, e.Message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
handlersLoaded = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion local instance state
|
|
||||||
|
|
||||||
#region overriding properties
|
|
||||||
|
|
||||||
// These properties override definitions
|
|
||||||
// in the base class.
|
|
||||||
|
|
||||||
// Name is used to differentiate the message header.
|
|
||||||
|
|
||||||
public override string Name
|
|
||||||
{
|
|
||||||
get { return "HANDLER"; }
|
|
||||||
}
|
|
||||||
|
|
||||||
// Used to partition the .ini configuration space.
|
|
||||||
|
|
||||||
public override string ConfigName
|
|
||||||
{
|
|
||||||
get { return "RestHandler"; }
|
|
||||||
}
|
|
||||||
|
|
||||||
// We have to rename these because we want
|
|
||||||
// to be able to share the values with other
|
|
||||||
// classes in our assembly and the base
|
|
||||||
// names are protected.
|
|
||||||
|
|
||||||
public string MsgId
|
|
||||||
{
|
|
||||||
get { return base.MsgID; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public string RequestId
|
|
||||||
{
|
|
||||||
get { return base.RequestID; }
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion overriding properties
|
|
||||||
|
|
||||||
#region overriding methods
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This method is called by OpenSimMain immediately after loading the
|
|
||||||
/// plugin and after basic server setup, but before running any server commands.
|
|
||||||
/// </summary>
|
|
||||||
/// <remarks>
|
|
||||||
/// Note that entries MUST be added to the active configuration files before
|
|
||||||
/// the plugin can be enabled.
|
|
||||||
/// </remarks>
|
|
||||||
|
|
||||||
public override void Initialise(OpenSimBase openSim)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// This plugin will only be enabled if the broader
|
|
||||||
// REST plugin mechanism is enabled.
|
|
||||||
|
|
||||||
//Rest.Log.InfoFormat("{0} Plugin is initializing", MsgId);
|
|
||||||
|
|
||||||
base.Initialise(openSim);
|
|
||||||
|
|
||||||
// IsEnabled is implemented by the base class and
|
|
||||||
// reflects an overall RestPlugin status
|
|
||||||
|
|
||||||
if (!IsEnabled)
|
|
||||||
{
|
|
||||||
//Rest.Log.WarnFormat("{0} Plugins are disabled", MsgId);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Rest.Log.InfoFormat("{0} Rest <{1}> plugin will be enabled", MsgId, Name);
|
|
||||||
Rest.Log.InfoFormat("{0} Configuration parameters read from <{1}>", MsgId, ConfigName);
|
|
||||||
|
|
||||||
// These are stored in static variables to make
|
|
||||||
// them easy to reach from anywhere in the assembly.
|
|
||||||
|
|
||||||
Rest.main = openSim;
|
|
||||||
if (Rest.main == null)
|
|
||||||
throw new Exception("OpenSim base pointer is null");
|
|
||||||
|
|
||||||
Rest.Plugin = this;
|
|
||||||
Rest.Config = Config;
|
|
||||||
Rest.Prefix = Prefix;
|
|
||||||
Rest.GodKey = GodKey;
|
|
||||||
Rest.Authenticate = Rest.Config.GetBoolean("authenticate", Rest.Authenticate);
|
|
||||||
Rest.Scheme = Rest.Config.GetString("auth-scheme", Rest.Scheme);
|
|
||||||
Rest.Secure = Rest.Config.GetBoolean("secured", Rest.Secure);
|
|
||||||
Rest.ExtendedEscape = Rest.Config.GetBoolean("extended-escape", Rest.ExtendedEscape);
|
|
||||||
Rest.Realm = Rest.Config.GetString("realm", Rest.Realm);
|
|
||||||
Rest.DumpAsset = Rest.Config.GetBoolean("dump-asset", Rest.DumpAsset);
|
|
||||||
Rest.Fill = Rest.Config.GetBoolean("path-fill", Rest.Fill);
|
|
||||||
Rest.DumpLineSize = Rest.Config.GetInt("dump-line-size", Rest.DumpLineSize);
|
|
||||||
Rest.FlushEnabled = Rest.Config.GetBoolean("flush-on-error", Rest.FlushEnabled);
|
|
||||||
|
|
||||||
// Note: Odd spacing is required in the following strings
|
|
||||||
|
|
||||||
Rest.Log.InfoFormat("{0} Authentication is {1}required", MsgId,
|
|
||||||
(Rest.Authenticate ? "" : "not "));
|
|
||||||
|
|
||||||
Rest.Log.InfoFormat("{0} Security is {1}enabled", MsgId,
|
|
||||||
(Rest.Secure ? "" : "not "));
|
|
||||||
|
|
||||||
Rest.Log.InfoFormat("{0} Extended URI escape processing is {1}enabled", MsgId,
|
|
||||||
(Rest.ExtendedEscape ? "" : "not "));
|
|
||||||
|
|
||||||
Rest.Log.InfoFormat("{0} Dumping of asset data is {1}enabled", MsgId,
|
|
||||||
(Rest.DumpAsset ? "" : "not "));
|
|
||||||
|
|
||||||
// The supplied prefix MUST be absolute
|
|
||||||
|
|
||||||
if (Rest.Prefix.Substring(0,1) != Rest.UrlPathSeparator)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} Prefix <{1}> is not absolute and must be", MsgId, Rest.Prefix);
|
|
||||||
Rest.Log.InfoFormat("{0} Prefix changed to </{1}>", MsgId, Rest.Prefix);
|
|
||||||
Rest.Prefix = String.Format("{0}{1}", Rest.UrlPathSeparator, Rest.Prefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If data dumping is requested, report on the chosen line
|
|
||||||
// length.
|
|
||||||
|
|
||||||
if (Rest.DumpAsset)
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} Dump {1} bytes per line", MsgId, Rest.DumpLineSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load all of the handlers present in the
|
|
||||||
// assembly
|
|
||||||
|
|
||||||
// In principle, as we're an application plug-in,
|
|
||||||
// most of what needs to be done could be done using
|
|
||||||
// static resources, however the Open Sim plug-in
|
|
||||||
// model makes this an instance, so that's what we
|
|
||||||
// need to be.
|
|
||||||
// There is only one Communications manager per
|
|
||||||
// server, and by inference, only one each of the
|
|
||||||
// user, asset, and inventory servers. So we can cache
|
|
||||||
// those using a static initializer.
|
|
||||||
// We move all of this processing off to another
|
|
||||||
// services class to minimize overlap between function
|
|
||||||
// and infrastructure.
|
|
||||||
|
|
||||||
LoadHandlers();
|
|
||||||
|
|
||||||
// The intention of a post construction initializer
|
|
||||||
// is to allow for setup that is dependent upon other
|
|
||||||
// activities outside of the agency.
|
|
||||||
|
|
||||||
foreach (IRest handler in handlers)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
handler.Initialize();
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.ErrorFormat("{0} initialization error: {1}", MsgId, e.Message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Now that everything is setup we can proceed to
|
|
||||||
// add THIS agent to the HTTP server's handler list
|
|
||||||
|
|
||||||
// FIXME: If this code is ever to be re-enabled (most of it is disabled already) then this will
|
|
||||||
// have to be handled through the AddHttpHandler interface.
|
|
||||||
// if (!AddAgentHandler(Rest.Name,this))
|
|
||||||
// {
|
|
||||||
// Rest.Log.ErrorFormat("{0} Unable to activate handler interface", MsgId);
|
|
||||||
// foreach (IRest handler in handlers)
|
|
||||||
// {
|
|
||||||
// handler.Close();
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.ErrorFormat("{0} Plugin initialization has failed: {1}", MsgId, e.Message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// In the interests of efficiency, and because we cannot determine whether
|
|
||||||
/// or not this instance will actually be harvested, we clobber the only
|
|
||||||
/// anchoring reference to the working state for this plug-in. What the
|
|
||||||
/// call to close does is irrelevant to this class beyond knowing that it
|
|
||||||
/// can nullify the reference when it returns.
|
|
||||||
/// To make sure everything is copacetic we make sure the primary interface
|
|
||||||
/// is disabled by deleting the handler from the HTTP server tables.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
public override void Close()
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} Plugin is terminating", MsgId);
|
|
||||||
|
|
||||||
// FIXME: If this code is ever to be re-enabled (most of it is disabled already) then this will
|
|
||||||
// have to be handled through the AddHttpHandler interface.
|
|
||||||
// try
|
|
||||||
// {
|
|
||||||
// RemoveAgentHandler(Rest.Name, this);
|
|
||||||
// }
|
|
||||||
// catch (KeyNotFoundException){}
|
|
||||||
|
|
||||||
foreach (IRest handler in handlers)
|
|
||||||
{
|
|
||||||
handler.Close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion overriding methods
|
|
||||||
|
|
||||||
#region interface methods
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This method is called by the HTTP server to match an incoming
|
|
||||||
/// request. It scans all of the strings registered by the
|
|
||||||
/// underlying handlers and looks for the best match. It returns
|
|
||||||
/// true if a match is found.
|
|
||||||
/// The matching process could be made arbitrarily complex.
|
|
||||||
/// Note: The match is case-insensitive.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
public bool Match(OSHttpRequest request, OSHttpResponse response)
|
|
||||||
{
|
|
||||||
|
|
||||||
string path = request.RawUrl.ToLower();
|
|
||||||
|
|
||||||
// Rest.Log.DebugFormat("{0} Match ENTRY", MsgId);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
foreach (string key in pathHandlers.Keys)
|
|
||||||
{
|
|
||||||
// Rest.Log.DebugFormat("{0} Match testing {1} against agent prefix <{2}>", MsgId, path, key);
|
|
||||||
|
|
||||||
// Note that Match will not necessarily find the handler that will
|
|
||||||
// actually be used - it does no test for the "closest" fit. It
|
|
||||||
// simply reflects that at least one possible handler exists.
|
|
||||||
|
|
||||||
if (path.StartsWith(key))
|
|
||||||
{
|
|
||||||
// Rest.Log.DebugFormat("{0} Matched prefix <{1}>", MsgId, key);
|
|
||||||
|
|
||||||
// This apparently odd evaluation is needed to prevent a match
|
|
||||||
// on anything other than a URI token boundary. Otherwise we
|
|
||||||
// may match on URL's that were not intended for this handler.
|
|
||||||
|
|
||||||
return (path.Length == key.Length ||
|
|
||||||
path.Substring(key.Length, 1) == Rest.UrlPathSeparator);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
path = String.Format("{0}{1}{2}", request.HttpMethod, Rest.UrlMethodSeparator, path);
|
|
||||||
|
|
||||||
foreach (string key in streamHandlers.Keys)
|
|
||||||
{
|
|
||||||
// Rest.Log.DebugFormat("{0} Match testing {1} against stream prefix <{2}>", MsgId, path, key);
|
|
||||||
|
|
||||||
// Note that Match will not necessarily find the handler that will
|
|
||||||
// actually be used - it does no test for the "closest" fit. It
|
|
||||||
// simply reflects that at least one possible handler exists.
|
|
||||||
|
|
||||||
if (path.StartsWith(key))
|
|
||||||
{
|
|
||||||
// Rest.Log.DebugFormat("{0} Matched prefix <{1}>", MsgId, key);
|
|
||||||
|
|
||||||
// This apparently odd evaluation is needed to prevent a match
|
|
||||||
// on anything other than a URI token boundary. Otherwise we
|
|
||||||
// may match on URL's that were not intended for this handler.
|
|
||||||
|
|
||||||
return (path.Length == key.Length ||
|
|
||||||
path.Substring(key.Length, 1) == Rest.UrlPathSeparator);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.ErrorFormat("{0} matching exception for path <{1}> : {2}", MsgId, path, e.Message);
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This is called by the HTTP server once the handler has indicated
|
|
||||||
/// that it is able to handle the request.
|
|
||||||
/// Preconditions:
|
|
||||||
/// [1] request != null and is a valid request object
|
|
||||||
/// [2] response != null and is a valid response object
|
|
||||||
/// Behavior is undefined if preconditions are not satisfied.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
public bool Handle(OSHttpRequest request, OSHttpResponse response)
|
|
||||||
{
|
|
||||||
bool handled;
|
|
||||||
base.MsgID = base.RequestID;
|
|
||||||
|
|
||||||
// Debug only
|
|
||||||
|
|
||||||
if (Rest.DEBUG)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} ENTRY", MsgId);
|
|
||||||
Rest.Log.DebugFormat("{0} Agent: {1}", MsgId, request.UserAgent);
|
|
||||||
Rest.Log.DebugFormat("{0} Method: {1}", MsgId, request.HttpMethod);
|
|
||||||
|
|
||||||
for (int i = 0; i < request.Headers.Count; i++)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Header [{1}] : <{2}> = <{3}>",
|
|
||||||
MsgId, i, request.Headers.GetKey(i), request.Headers.Get(i));
|
|
||||||
}
|
|
||||||
Rest.Log.DebugFormat("{0} URI: {1}", MsgId, request.RawUrl);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If a path handler worked we're done, otherwise try any
|
|
||||||
// available stream handlers too.
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
handled = (FindPathHandler(request, response) ||
|
|
||||||
FindStreamHandler(request, response));
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
// A raw exception indicates that something we weren't expecting has
|
|
||||||
// happened. This should always reflect a shortcoming in the plugin,
|
|
||||||
// or a failure to satisfy the preconditions. It should not reflect
|
|
||||||
// an error in the request itself. Under such circumstances the state
|
|
||||||
// of the request cannot be determined and we are obliged to mark it
|
|
||||||
// as 'handled'.
|
|
||||||
|
|
||||||
Rest.Log.ErrorFormat("{0} Plugin error: {1}", MsgId, e.Message);
|
|
||||||
handled = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} EXIT", MsgId);
|
|
||||||
|
|
||||||
return handled;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion interface methods
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// If there is a stream handler registered that can handle the
|
|
||||||
/// request, then fine. If the request is not matched, do
|
|
||||||
/// nothing.
|
|
||||||
/// Note: The selection is case-insensitive
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
private bool FindStreamHandler(OSHttpRequest request, OSHttpResponse response)
|
|
||||||
{
|
|
||||||
RequestData rdata = new RequestData(request, response, String.Empty);
|
|
||||||
|
|
||||||
string bestMatch = String.Empty;
|
|
||||||
string path = String.Format("{0}:{1}", rdata.method, rdata.path).ToLower();
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} Checking for stream handler for <{1}>", MsgId, path);
|
|
||||||
|
|
||||||
if (!IsEnabled)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (string pattern in streamHandlers.Keys)
|
|
||||||
{
|
|
||||||
if (path.StartsWith(pattern))
|
|
||||||
{
|
|
||||||
if (pattern.Length > bestMatch.Length)
|
|
||||||
{
|
|
||||||
bestMatch = pattern;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle using the best match available
|
|
||||||
|
|
||||||
if (bestMatch.Length > 0)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Stream-based handler matched with <{1}>", MsgId, bestMatch);
|
|
||||||
RestStreamHandler handler = streamHandlers[bestMatch];
|
|
||||||
rdata.buffer = handler.Handle(rdata.path, rdata.request.InputStream, rdata.request, rdata.response);
|
|
||||||
rdata.AddHeader(rdata.response.ContentType,handler.ContentType);
|
|
||||||
rdata.Respond("FindStreamHandler Completion");
|
|
||||||
}
|
|
||||||
|
|
||||||
return rdata.handled;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Add a stream handler for the designated HTTP method and path prefix.
|
|
||||||
/// If the handler is not enabled, the request is ignored. If the path
|
|
||||||
/// does not start with the REST prefix, it is added. If method-qualified
|
|
||||||
/// path has not already been registered, the method is added to the active
|
|
||||||
/// handler table.
|
|
||||||
/// </summary>
|
|
||||||
public void AddStreamHandler(string httpMethod, string path, RestMethod method)
|
|
||||||
{
|
|
||||||
if (!IsEnabled)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!path.StartsWith(Rest.Prefix))
|
|
||||||
{
|
|
||||||
path = String.Format("{0}{1}", Rest.Prefix, path);
|
|
||||||
}
|
|
||||||
|
|
||||||
path = String.Format("{0}{1}{2}", httpMethod, Rest.UrlMethodSeparator, path);
|
|
||||||
|
|
||||||
// Conditionally add to the list
|
|
||||||
|
|
||||||
if (!streamHandlers.ContainsKey(path))
|
|
||||||
{
|
|
||||||
streamHandlers.Add(path, new RestStreamHandler(httpMethod, path, method));
|
|
||||||
Rest.Log.DebugFormat("{0} Added handler for {1}", MsgId, path);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} Ignoring duplicate handler for {1}", MsgId, path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Given the supplied request/response, if the handler is enabled, the inbound
|
|
||||||
/// information is used to match an entry in the active path handler tables, using
|
|
||||||
/// the method-qualified path information. If a match is found, then the handler is
|
|
||||||
/// invoked. The result is the boolean result of the handler, or false if no
|
|
||||||
/// handler was located. The boolean indicates whether or not the request has been
|
|
||||||
/// handled, not whether or not the request was successful - that information is in
|
|
||||||
/// the response.
|
|
||||||
/// Note: The selection process is case-insensitive
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal bool FindPathHandler(OSHttpRequest request, OSHttpResponse response)
|
|
||||||
{
|
|
||||||
RequestData rdata = null;
|
|
||||||
string bestMatch = null;
|
|
||||||
|
|
||||||
if (!IsEnabled)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Conditionally add to the list
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} Checking for path handler for <{1}>", MsgId, request.RawUrl);
|
|
||||||
|
|
||||||
foreach (string pattern in pathHandlers.Keys)
|
|
||||||
{
|
|
||||||
if (request.RawUrl.ToLower().StartsWith(pattern))
|
|
||||||
{
|
|
||||||
if (String.IsNullOrEmpty(bestMatch) || pattern.Length > bestMatch.Length)
|
|
||||||
{
|
|
||||||
bestMatch = pattern;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!String.IsNullOrEmpty(bestMatch))
|
|
||||||
{
|
|
||||||
rdata = pathAllocators[bestMatch](request, response, bestMatch);
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} Path based REST handler matched with <{1}>", MsgId, bestMatch);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
pathHandlers[bestMatch](rdata);
|
|
||||||
}
|
|
||||||
|
|
||||||
// A plugin generated error indicates a request-related error
|
|
||||||
// that has been handled by the plugin.
|
|
||||||
|
|
||||||
catch (RestException r)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} Request failed: {1}", MsgId, r.Message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return (rdata == null) ? false : rdata.handled;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// A method handler and a request allocator are stored using the designated
|
|
||||||
/// path as a key. If an entry already exists, it is replaced by the new one.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
public void AddPathHandler(RestMethodHandler mh, string path, RestMethodAllocator ra)
|
|
||||||
{
|
|
||||||
if (!IsEnabled)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (pathHandlers.ContainsKey(path))
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Replacing handler for <${1}>", MsgId, path);
|
|
||||||
pathHandlers.Remove(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (pathAllocators.ContainsKey(path))
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Replacing allocator for <${1}>", MsgId, path);
|
|
||||||
pathAllocators.Remove(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} Adding path handler for {1}", MsgId, path);
|
|
||||||
|
|
||||||
pathHandlers.Add(path, mh);
|
|
||||||
pathAllocators.Add(path, ra);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,246 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Reflection;
|
|
||||||
using OpenSim.Framework.Servers;
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Inventory
|
|
||||||
{
|
|
||||||
public class RestTestServices : IRest
|
|
||||||
{
|
|
||||||
private bool enabled = false;
|
|
||||||
private string qPrefix = "test";
|
|
||||||
|
|
||||||
// A simple constructor is used to handle any once-only
|
|
||||||
// initialization of working classes.
|
|
||||||
|
|
||||||
public RestTestServices()
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} Test services initializing", MsgId);
|
|
||||||
Rest.Log.InfoFormat("{0} Using REST Implementation Version {1}", MsgId, Rest.Version);
|
|
||||||
|
|
||||||
// If a relative path was specified, make it absolute by adding
|
|
||||||
// the standard prefix, e.g. /admin
|
|
||||||
|
|
||||||
if (!qPrefix.StartsWith(Rest.UrlPathSeparator))
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} Domain is relative, adding absolute prefix", MsgId);
|
|
||||||
qPrefix = String.Format("{0}{1}{2}", Rest.Prefix, Rest.UrlPathSeparator, qPrefix);
|
|
||||||
Rest.Log.InfoFormat("{0} Domain is now <{1}>", MsgId, qPrefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load test cases
|
|
||||||
|
|
||||||
loadTests();
|
|
||||||
foreach (ITest test in tests)
|
|
||||||
{
|
|
||||||
test.Initialize();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Register interface
|
|
||||||
|
|
||||||
Rest.Plugin.AddPathHandler(DoTests,qPrefix,Allocate);
|
|
||||||
|
|
||||||
// Activate
|
|
||||||
|
|
||||||
enabled = true;
|
|
||||||
|
|
||||||
Rest.Log.InfoFormat("{0} Test services initialization complete", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Post-construction, pre-enabled initialization opportunity
|
|
||||||
// Not currently exploited.
|
|
||||||
|
|
||||||
public void Initialize()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// Called by the plug-in to halt REST processing. Local processing is
|
|
||||||
// disabled, and control blocks until all current processing has
|
|
||||||
// completed. No new processing will be started
|
|
||||||
|
|
||||||
public void Close()
|
|
||||||
{
|
|
||||||
enabled = false;
|
|
||||||
foreach (ITest test in tests)
|
|
||||||
{
|
|
||||||
test.Close();
|
|
||||||
}
|
|
||||||
Rest.Log.InfoFormat("{0} Test services closing down", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Properties
|
|
||||||
|
|
||||||
internal string MsgId
|
|
||||||
{
|
|
||||||
get { return Rest.MsgId; }
|
|
||||||
}
|
|
||||||
|
|
||||||
#region Interface
|
|
||||||
|
|
||||||
private RequestData Allocate(OSHttpRequest request, OSHttpResponse response, string prefix)
|
|
||||||
{
|
|
||||||
return new RequestData(request, response, prefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Inventory Handler
|
|
||||||
|
|
||||||
private void DoTests(RequestData rdata)
|
|
||||||
{
|
|
||||||
if (!enabled)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Now that we know this is a serious attempt to
|
|
||||||
// access inventory data, we should find out who
|
|
||||||
// is asking, and make sure they are authorized
|
|
||||||
// to do so. We need to validate the caller's
|
|
||||||
// identity before revealing anything about the
|
|
||||||
// status quo. Authenticate throws an exception
|
|
||||||
// via Fail if no identity information is present.
|
|
||||||
//
|
|
||||||
// With the present HTTP server we can't use the
|
|
||||||
// builtin authentication mechanisms because they
|
|
||||||
// would be enforced for all in-bound requests.
|
|
||||||
// Instead we look at the headers ourselves and
|
|
||||||
// handle authentication directly.
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (!rdata.IsAuthenticated)
|
|
||||||
{
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeNotAuthorized,
|
|
||||||
String.Format("user \"{0}\" could not be authenticated", rdata.userName));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (RestException e)
|
|
||||||
{
|
|
||||||
if (e.statusCode == Rest.HttpStatusCodeNotAuthorized)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} User not authenticated", MsgId);
|
|
||||||
Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId, rdata.request.Headers.Get("Authorization"));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Rest.Log.ErrorFormat("{0} User authentication failed", MsgId);
|
|
||||||
Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId, rdata.request.Headers.Get("Authorization"));
|
|
||||||
}
|
|
||||||
throw (e);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check that a test was specified
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length < 1)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Insufficient parameters", MsgId);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest, "not enough parameters");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Select the test
|
|
||||||
|
|
||||||
foreach (ITest test in tests)
|
|
||||||
{
|
|
||||||
if (!rdata.handled)
|
|
||||||
test.Execute(rdata);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion Interface
|
|
||||||
|
|
||||||
private static bool testsLoaded = false;
|
|
||||||
private static List<Type> classes = new List<Type>();
|
|
||||||
private static List<ITest> tests = new List<ITest>();
|
|
||||||
private static Type[] parms = new Type[0];
|
|
||||||
private static Object[] args = new Object[0];
|
|
||||||
|
|
||||||
static RestTestServices()
|
|
||||||
{
|
|
||||||
Module[] mods = Assembly.GetExecutingAssembly().GetModules();
|
|
||||||
foreach (Module m in mods)
|
|
||||||
{
|
|
||||||
Type[] types = m.GetTypes();
|
|
||||||
foreach (Type t in types)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (t.GetInterface("ITest") != null)
|
|
||||||
{
|
|
||||||
classes.Add(t);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("[STATIC-TEST] Unable to include test {0} : {1}", t, e.Message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This routine loads all of the handlers discovered during
|
|
||||||
/// instance initialization. Each handler is responsible for
|
|
||||||
/// registering itself with this handler.
|
|
||||||
/// I was not able to make this code work in a constructor.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
private void loadTests()
|
|
||||||
{
|
|
||||||
lock (tests)
|
|
||||||
{
|
|
||||||
if (!testsLoaded)
|
|
||||||
{
|
|
||||||
|
|
||||||
ConstructorInfo ci;
|
|
||||||
Object ht;
|
|
||||||
|
|
||||||
foreach (Type t in classes)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (t.GetInterface("ITest") != null)
|
|
||||||
{
|
|
||||||
ci = t.GetConstructor(parms);
|
|
||||||
ht = ci.Invoke(args);
|
|
||||||
tests.Add((ITest)ht);
|
|
||||||
Rest.Log.InfoFormat("{0} Test {1} added", MsgId, t);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} Unable to load test {1} : {2}", MsgId, t, e.Message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
testsLoaded = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,46 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Inventory
|
|
||||||
{
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This interface represents the boundary between the general purpose
|
|
||||||
/// REST plugin handling, and the functionally specific handlers. The
|
|
||||||
/// handler knows only to initialzie and terminate all such handlers
|
|
||||||
/// that it finds.
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal interface ITest
|
|
||||||
{
|
|
||||||
void Initialize();
|
|
||||||
void Execute(RequestData rdata);
|
|
||||||
void Close();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,204 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using OpenMetaverse;
|
|
||||||
using OpenSim.Region.Framework.Scenes;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Inventory.Tests
|
|
||||||
{
|
|
||||||
public class Remote : ITest
|
|
||||||
{
|
|
||||||
private static readonly int PARM_TESTID = 0;
|
|
||||||
private static readonly int PARM_COMMAND = 1;
|
|
||||||
|
|
||||||
private static readonly int PARM_MOVE_AVATAR = 2;
|
|
||||||
private static readonly int PARM_MOVE_X = 3;
|
|
||||||
private static readonly int PARM_MOVE_Y = 4;
|
|
||||||
private static readonly int PARM_MOVE_Z = 5;
|
|
||||||
|
|
||||||
private bool enabled = false;
|
|
||||||
|
|
||||||
// No constructor code is required.
|
|
||||||
|
|
||||||
public Remote()
|
|
||||||
{
|
|
||||||
Rest.Log.InfoFormat("{0} Remote services constructor", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Post-construction, pre-enabled initialization opportunity
|
|
||||||
// Not currently exploited.
|
|
||||||
|
|
||||||
public void Initialize()
|
|
||||||
{
|
|
||||||
enabled = true;
|
|
||||||
Rest.Log.InfoFormat("{0} Remote services initialized", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Called by the plug-in to halt REST processing. Local processing is
|
|
||||||
// disabled, and control blocks until all current processing has
|
|
||||||
// completed. No new processing will be started
|
|
||||||
|
|
||||||
public void Close()
|
|
||||||
{
|
|
||||||
enabled = false;
|
|
||||||
Rest.Log.InfoFormat("{0} Remote services closing down", MsgId);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Properties
|
|
||||||
|
|
||||||
internal string MsgId
|
|
||||||
{
|
|
||||||
get { return Rest.MsgId; }
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remote Handler
|
|
||||||
// Key information of interest here is the Parameters array, each
|
|
||||||
// entry represents an element of the URI, with element zero being
|
|
||||||
// the
|
|
||||||
|
|
||||||
public void Execute(RequestData rdata)
|
|
||||||
{
|
|
||||||
if (!enabled) return;
|
|
||||||
|
|
||||||
// If we can't relate to what's there, leave it for others.
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length == 0 || rdata.Parameters[PARM_TESTID] != "remote")
|
|
||||||
return;
|
|
||||||
|
|
||||||
Rest.Log.DebugFormat("{0} REST Remote handler ENTRY", MsgId);
|
|
||||||
|
|
||||||
// Remove the prefix and what's left are the parameters. If we don't have
|
|
||||||
// the parameters we need, fail the request. Parameters do NOT include
|
|
||||||
// any supplied query values.
|
|
||||||
|
|
||||||
if (rdata.Parameters.Length > 1)
|
|
||||||
{
|
|
||||||
switch (rdata.Parameters[PARM_COMMAND].ToLower())
|
|
||||||
{
|
|
||||||
case "move" :
|
|
||||||
DoMove(rdata);
|
|
||||||
break;
|
|
||||||
default :
|
|
||||||
DoHelp(rdata);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
DoHelp(rdata);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void DoHelp(RequestData rdata)
|
|
||||||
{
|
|
||||||
rdata.body = Help;
|
|
||||||
rdata.Complete();
|
|
||||||
rdata.Respond("Help");
|
|
||||||
}
|
|
||||||
|
|
||||||
private void DoMove(RequestData rdata)
|
|
||||||
{
|
|
||||||
if (rdata.Parameters.Length < 6)
|
|
||||||
{
|
|
||||||
Rest.Log.WarnFormat("{0} Move: No movement information provided", MsgId);
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest, "no movement information provided");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
string[] names = rdata.Parameters[PARM_MOVE_AVATAR].Split(Rest.CA_SPACE);
|
|
||||||
ScenePresence presence = null;
|
|
||||||
Scene scene = null;
|
|
||||||
|
|
||||||
if (names.Length != 2)
|
|
||||||
{
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest,
|
|
||||||
String.Format("invalid avatar name: <{0}>",rdata.Parameters[PARM_MOVE_AVATAR]));
|
|
||||||
}
|
|
||||||
|
|
||||||
Rest.Log.WarnFormat("{0} '{1}' command received for {2} {3}",
|
|
||||||
MsgId, rdata.Parameters[0], names[0], names[1]);
|
|
||||||
|
|
||||||
// The first parameter should be an avatar name, look for the
|
|
||||||
// avatar in the known regions first.
|
|
||||||
|
|
||||||
Rest.main.SceneManager.ForEachScene(delegate(Scene s)
|
|
||||||
{
|
|
||||||
s.ForEachRootScenePresence(delegate(ScenePresence sp)
|
|
||||||
{
|
|
||||||
if (sp.Firstname == names[0] && sp.Lastname == names[1])
|
|
||||||
{
|
|
||||||
scene = s;
|
|
||||||
presence = sp;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
if (presence != null)
|
|
||||||
{
|
|
||||||
Rest.Log.DebugFormat("{0} Move : Avatar {1} located in region {2}",
|
|
||||||
MsgId, rdata.Parameters[PARM_MOVE_AVATAR], scene.RegionInfo.RegionName);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
float x = Convert.ToSingle(rdata.Parameters[PARM_MOVE_X]);
|
|
||||||
float y = Convert.ToSingle(rdata.Parameters[PARM_MOVE_Y]);
|
|
||||||
float z = Convert.ToSingle(rdata.Parameters[PARM_MOVE_Z]);
|
|
||||||
Vector3 vector = new Vector3(x, y, z);
|
|
||||||
presence.MoveToTarget(vector, false, false);
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest,
|
|
||||||
String.Format("invalid parameters: {0}", e.Message));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
rdata.Fail(Rest.HttpStatusCodeBadRequest,
|
|
||||||
String.Format("avatar {0} not present", rdata.Parameters[PARM_MOVE_AVATAR]));
|
|
||||||
}
|
|
||||||
|
|
||||||
rdata.Complete();
|
|
||||||
rdata.Respond("OK");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static readonly string Help =
|
|
||||||
"<html>"
|
|
||||||
+ "<head><title>Remote Command Usage</title></head>"
|
|
||||||
+ "<body>"
|
|
||||||
+ "<p>Supported commands are:</p>"
|
|
||||||
+ "<dl>"
|
|
||||||
+ "<dt>move/avatar-name/x/y/z</dt>"
|
|
||||||
+ "<dd>moves the specified avatar to another location</dd>"
|
|
||||||
+ "</dl>"
|
|
||||||
+ "</body>"
|
|
||||||
+ "</html>"
|
|
||||||
;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,228 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.IO;
|
|
||||||
using System.Xml.Serialization;
|
|
||||||
using OpenMetaverse;
|
|
||||||
using OpenSim.Framework;
|
|
||||||
using OpenSim.Framework.Servers;
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
|
||||||
using OpenSim.Region.Framework.Interfaces;
|
|
||||||
using OpenSim.Region.Framework.Scenes;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Regions
|
|
||||||
{
|
|
||||||
public partial class RestRegionPlugin : RestPlugin
|
|
||||||
{
|
|
||||||
#region GET methods
|
|
||||||
public string GetHandler(string request, string path, string param,
|
|
||||||
IOSHttpRequest httpRequest, IOSHttpResponse httpResponse)
|
|
||||||
{
|
|
||||||
// foreach (string h in httpRequest.Headers.AllKeys)
|
|
||||||
// foreach (string v in httpRequest.Headers.GetValues(h))
|
|
||||||
// m_log.DebugFormat("{0} IsGod: {1} -> {2}", MsgID, h, v);
|
|
||||||
|
|
||||||
MsgID = RequestID;
|
|
||||||
m_log.DebugFormat("{0} GET path {1} param {2}", MsgID, path, param);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// param empty: regions list
|
|
||||||
if (String.IsNullOrEmpty(param)) return GetHandlerRegions(httpResponse);
|
|
||||||
|
|
||||||
// param not empty: specific region
|
|
||||||
return GetHandlerRegion(httpResponse, param);
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
return Failure(httpResponse, OSHttpStatusCode.ServerErrorInternalError, "GET", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public string GetHandlerRegions(IOSHttpResponse httpResponse)
|
|
||||||
{
|
|
||||||
RestXmlWriter rxw = new RestXmlWriter(new StringWriter());
|
|
||||||
|
|
||||||
rxw.WriteStartElement(String.Empty, "regions", String.Empty);
|
|
||||||
foreach (Scene s in App.SceneManager.Scenes)
|
|
||||||
{
|
|
||||||
rxw.WriteStartElement(String.Empty, "uuid", String.Empty);
|
|
||||||
rxw.WriteString(s.RegionInfo.RegionID.ToString());
|
|
||||||
rxw.WriteEndElement();
|
|
||||||
}
|
|
||||||
rxw.WriteEndElement();
|
|
||||||
|
|
||||||
return rxw.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected string ShortRegionInfo(string key, string value)
|
|
||||||
{
|
|
||||||
RestXmlWriter rxw = new RestXmlWriter(new StringWriter());
|
|
||||||
|
|
||||||
if (String.IsNullOrEmpty(value) ||
|
|
||||||
String.IsNullOrEmpty(key)) return null;
|
|
||||||
|
|
||||||
rxw.WriteStartElement(String.Empty, "region", String.Empty);
|
|
||||||
rxw.WriteStartElement(String.Empty, key, String.Empty);
|
|
||||||
rxw.WriteString(value);
|
|
||||||
rxw.WriteEndDocument();
|
|
||||||
|
|
||||||
return rxw.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
public string GetHandlerRegion(IOSHttpResponse httpResponse, string param)
|
|
||||||
{
|
|
||||||
// be resilient and don't get confused by a terminating '/'
|
|
||||||
param = param.TrimEnd(new char[]{'/'});
|
|
||||||
string[] comps = param.Split('/');
|
|
||||||
UUID regionID = (UUID)comps[0];
|
|
||||||
|
|
||||||
m_log.DebugFormat("{0} GET region UUID {1}", MsgID, regionID.ToString());
|
|
||||||
|
|
||||||
if (UUID.Zero == regionID) throw new Exception("missing region ID");
|
|
||||||
|
|
||||||
Scene scene = null;
|
|
||||||
App.SceneManager.TryGetScene(regionID, out scene);
|
|
||||||
if (null == scene) return Failure(httpResponse, OSHttpStatusCode.ClientErrorNotFound,
|
|
||||||
"GET", "cannot find region {0}", regionID.ToString());
|
|
||||||
|
|
||||||
RegionDetails details = new RegionDetails(scene.RegionInfo);
|
|
||||||
|
|
||||||
// m_log.DebugFormat("{0} GET comps {1}", MsgID, comps.Length);
|
|
||||||
// for (int i = 0; i < comps.Length; i++) m_log.DebugFormat("{0} GET comps[{1}] >{2}<", MsgID, i, comps[i]);
|
|
||||||
|
|
||||||
if (1 == comps.Length)
|
|
||||||
{
|
|
||||||
// complete region details requested
|
|
||||||
RestXmlWriter rxw = new RestXmlWriter(new StringWriter());
|
|
||||||
XmlSerializer xs = new XmlSerializer(typeof(RegionDetails));
|
|
||||||
xs.Serialize(rxw, details, _xmlNs);
|
|
||||||
return rxw.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (2 == comps.Length)
|
|
||||||
{
|
|
||||||
string resp = ShortRegionInfo(comps[1], details[comps[1]]);
|
|
||||||
if (null != resp) return resp;
|
|
||||||
|
|
||||||
// m_log.DebugFormat("{0} GET comps advanced: >{1}<", MsgID, comps[1]);
|
|
||||||
|
|
||||||
// check for {terrain,stats,prims}
|
|
||||||
switch (comps[1].ToLower())
|
|
||||||
{
|
|
||||||
case "terrain":
|
|
||||||
return RegionTerrain(httpResponse, scene);
|
|
||||||
|
|
||||||
case "stats":
|
|
||||||
return RegionStats(httpResponse, scene);
|
|
||||||
|
|
||||||
case "prims":
|
|
||||||
return RegionPrims(httpResponse, scene, Vector3.Zero, Vector3.Zero);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (3 == comps.Length)
|
|
||||||
{
|
|
||||||
switch (comps[1].ToLower())
|
|
||||||
{
|
|
||||||
case "prims":
|
|
||||||
string[] subregion = comps[2].Split(',');
|
|
||||||
if (subregion.Length == 6)
|
|
||||||
{
|
|
||||||
Vector3 min, max;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
min = new Vector3((float)Double.Parse(subregion[0], Culture.NumberFormatInfo), (float)Double.Parse(subregion[1], Culture.NumberFormatInfo), (float)Double.Parse(subregion[2], Culture.NumberFormatInfo));
|
|
||||||
max = new Vector3((float)Double.Parse(subregion[3], Culture.NumberFormatInfo), (float)Double.Parse(subregion[4], Culture.NumberFormatInfo), (float)Double.Parse(subregion[5], Culture.NumberFormatInfo));
|
|
||||||
}
|
|
||||||
catch (Exception)
|
|
||||||
{
|
|
||||||
return Failure(httpResponse, OSHttpStatusCode.ClientErrorBadRequest,
|
|
||||||
"GET", "invalid subregion parameter");
|
|
||||||
}
|
|
||||||
return RegionPrims(httpResponse, scene, min, max);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return Failure(httpResponse, OSHttpStatusCode.ClientErrorBadRequest,
|
|
||||||
"GET", "invalid subregion parameter");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return Failure(httpResponse, OSHttpStatusCode.ClientErrorBadRequest,
|
|
||||||
"GET", "too many parameters {0}", param);
|
|
||||||
}
|
|
||||||
#endregion GET methods
|
|
||||||
|
|
||||||
protected string RegionTerrain(IOSHttpResponse httpResponse, Scene scene)
|
|
||||||
{
|
|
||||||
httpResponse.SendChunked = true;
|
|
||||||
httpResponse.ContentType = "text/xml";
|
|
||||||
|
|
||||||
return scene.Heightmap.SaveToXmlString();
|
|
||||||
//return Failure(httpResponse, OSHttpStatusCode.ServerErrorNotImplemented,
|
|
||||||
// "GET", "terrain not implemented");
|
|
||||||
}
|
|
||||||
|
|
||||||
protected string RegionStats(IOSHttpResponse httpResponse, Scene scene)
|
|
||||||
{
|
|
||||||
int users = scene.GetRootAgentCount();
|
|
||||||
int objects = scene.Entities.Count - users;
|
|
||||||
|
|
||||||
RestXmlWriter rxw = new RestXmlWriter(new StringWriter());
|
|
||||||
|
|
||||||
rxw.WriteStartElement(String.Empty, "region", String.Empty);
|
|
||||||
rxw.WriteStartElement(String.Empty, "stats", String.Empty);
|
|
||||||
|
|
||||||
rxw.WriteStartElement(String.Empty, "users", String.Empty);
|
|
||||||
rxw.WriteString(users.ToString());
|
|
||||||
rxw.WriteEndElement();
|
|
||||||
|
|
||||||
rxw.WriteStartElement(String.Empty, "objects", String.Empty);
|
|
||||||
rxw.WriteString(objects.ToString());
|
|
||||||
rxw.WriteEndElement();
|
|
||||||
|
|
||||||
rxw.WriteEndDocument();
|
|
||||||
|
|
||||||
return rxw.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected string RegionPrims(IOSHttpResponse httpResponse, Scene scene, Vector3 min, Vector3 max)
|
|
||||||
{
|
|
||||||
httpResponse.SendChunked = true;
|
|
||||||
httpResponse.ContentType = "text/xml";
|
|
||||||
|
|
||||||
IRegionSerialiserModule serialiser = scene.RequestModuleInterface<IRegionSerialiserModule>();
|
|
||||||
if (serialiser != null)
|
|
||||||
serialiser.SavePrimsToXml2(scene, new StreamWriter(httpResponse.OutputStream), min, max);
|
|
||||||
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,136 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.IO;
|
|
||||||
using System.Xml.Serialization;
|
|
||||||
using OpenMetaverse;
|
|
||||||
using OpenSim.Framework.Servers;
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
|
||||||
using OpenSim.Region.Framework.Interfaces;
|
|
||||||
using OpenSim.Region.Framework.Scenes;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Regions
|
|
||||||
{
|
|
||||||
public partial class RestRegionPlugin : RestPlugin
|
|
||||||
{
|
|
||||||
#region GET methods
|
|
||||||
public string GetRegionInfoHandler(string request, string path, string param,
|
|
||||||
IOSHttpRequest httpRequest, IOSHttpResponse httpResponse)
|
|
||||||
{
|
|
||||||
// foreach (string h in httpRequest.Headers.AllKeys)
|
|
||||||
// foreach (string v in httpRequest.Headers.GetValues(h))
|
|
||||||
// m_log.DebugFormat("{0} IsGod: {1} -> {2}", MsgID, h, v);
|
|
||||||
|
|
||||||
MsgID = RequestID;
|
|
||||||
m_log.DebugFormat("{0} GET path {1} param {2}", MsgID, path, param);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// param empty: regions list
|
|
||||||
// if (String.IsNullOrEmpty(param))
|
|
||||||
return GetRegionInfoHandlerRegions(httpResponse);
|
|
||||||
|
|
||||||
// // param not empty: specific region
|
|
||||||
// return GetRegionInfoHandlerRegion(httpResponse, param);
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
return Failure(httpResponse, OSHttpStatusCode.ServerErrorInternalError, "GET", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public string GetRegionInfoHandlerRegions(IOSHttpResponse httpResponse)
|
|
||||||
{
|
|
||||||
RestXmlWriter rxw = new RestXmlWriter(new StringWriter());
|
|
||||||
|
|
||||||
// regions info
|
|
||||||
rxw.WriteStartElement(String.Empty, "regions", String.Empty);
|
|
||||||
{
|
|
||||||
// regions info: number of regions
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "number", String.Empty);
|
|
||||||
rxw.WriteValue(App.SceneManager.Scenes.Count);
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
// regions info: max number of regions
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "max", String.Empty);
|
|
||||||
if (App.ConfigSource.Source.Configs["RemoteAdmin"] != null)
|
|
||||||
{
|
|
||||||
rxw.WriteValue(App.ConfigSource.Source.Configs["RemoteAdmin"].GetInt("region_limit", -1));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
rxw.WriteValue(-1);
|
|
||||||
}
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
// regions info: region
|
|
||||||
foreach (Scene s in App.SceneManager.Scenes)
|
|
||||||
{
|
|
||||||
rxw.WriteStartElement(String.Empty, "region", String.Empty);
|
|
||||||
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "uuid", String.Empty);
|
|
||||||
rxw.WriteString(s.RegionInfo.RegionID.ToString());
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "name", String.Empty);
|
|
||||||
rxw.WriteString(s.RegionInfo.RegionName);
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "x", String.Empty);
|
|
||||||
rxw.WriteValue(s.RegionInfo.RegionLocX);
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "y", String.Empty);
|
|
||||||
rxw.WriteValue(s.RegionInfo.RegionLocY);
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "external_hostname", String.Empty);
|
|
||||||
rxw.WriteString(s.RegionInfo.ExternalHostName);
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "ip", String.Empty);
|
|
||||||
rxw.WriteString(s.RegionInfo.InternalEndPoint.ToString());
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
int users = s.GetRootAgentCount();
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "avatars", String.Empty);
|
|
||||||
rxw.WriteValue(users);
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
rxw.WriteStartAttribute(String.Empty, "objects", String.Empty);
|
|
||||||
rxw.WriteValue(s.Entities.Count - users);
|
|
||||||
rxw.WriteEndAttribute();
|
|
||||||
|
|
||||||
rxw.WriteEndElement();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return rxw.ToString();
|
|
||||||
}
|
|
||||||
#endregion GET methods
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,122 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.IO;
|
|
||||||
using OpenMetaverse;
|
|
||||||
using OpenSim.Framework.Servers;
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
|
||||||
using OpenSim.Region.Framework.Interfaces;
|
|
||||||
using OpenSim.Region.Framework.Scenes;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Regions
|
|
||||||
{
|
|
||||||
public partial class RestRegionPlugin : RestPlugin
|
|
||||||
{
|
|
||||||
#region POST methods
|
|
||||||
|
|
||||||
public string PostHandler(string request, string path, string param,
|
|
||||||
IOSHttpRequest httpRequest, IOSHttpResponse httpResponse)
|
|
||||||
{
|
|
||||||
// foreach (string h in httpRequest.Headers.AllKeys)
|
|
||||||
// foreach (string v in httpRequest.Headers.GetValues(h))
|
|
||||||
// m_log.DebugFormat("{0} IsGod: {1} -> {2}", MsgID, h, v);
|
|
||||||
|
|
||||||
MsgID = RequestID;
|
|
||||||
m_log.DebugFormat("{0} POST path {1} param {2}", MsgID, path, param);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// param empty: new region post
|
|
||||||
if (!IsGod(httpRequest))
|
|
||||||
// XXX: this needs to be turned into a FailureUnauthorized(...)
|
|
||||||
return Failure(httpResponse, OSHttpStatusCode.ClientErrorUnauthorized,
|
|
||||||
"GET", "you are not god");
|
|
||||||
|
|
||||||
if (String.IsNullOrEmpty(param)) return CreateRegion(httpRequest, httpResponse);
|
|
||||||
|
|
||||||
// Parse region ID and other parameters
|
|
||||||
param = param.TrimEnd(new char[] {'/'});
|
|
||||||
string[] comps = param.Split('/');
|
|
||||||
UUID regionID = (UUID) comps[0];
|
|
||||||
|
|
||||||
m_log.DebugFormat("{0} POST region UUID {1}", MsgID, regionID.ToString());
|
|
||||||
if (UUID.Zero == regionID) throw new Exception("missing region ID");
|
|
||||||
|
|
||||||
Scene scene = null;
|
|
||||||
App.SceneManager.TryGetScene(regionID, out scene);
|
|
||||||
if (null == scene)
|
|
||||||
return Failure(httpResponse, OSHttpStatusCode.ClientErrorNotFound,
|
|
||||||
"POST", "cannot find region {0}", regionID.ToString());
|
|
||||||
|
|
||||||
if (2 == comps.Length)
|
|
||||||
{
|
|
||||||
// check for {prims}
|
|
||||||
switch (comps[1].ToLower())
|
|
||||||
{
|
|
||||||
case "prims":
|
|
||||||
return LoadPrims(request, httpRequest, httpResponse, scene);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return Failure(httpResponse, OSHttpStatusCode.ClientErrorNotFound,
|
|
||||||
"POST", "url {0} not supported", param);
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
return Failure(httpResponse, OSHttpStatusCode.ServerErrorInternalError, "POST", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public string CreateRegion(IOSHttpRequest request, IOSHttpResponse response)
|
|
||||||
{
|
|
||||||
RestXmlWriter rxw = new RestXmlWriter(new StringWriter());
|
|
||||||
|
|
||||||
rxw.WriteStartElement(String.Empty, "regions", String.Empty);
|
|
||||||
foreach (Scene s in App.SceneManager.Scenes)
|
|
||||||
{
|
|
||||||
rxw.WriteStartElement(String.Empty, "uuid", String.Empty);
|
|
||||||
rxw.WriteString(s.RegionInfo.RegionID.ToString());
|
|
||||||
rxw.WriteEndElement();
|
|
||||||
}
|
|
||||||
rxw.WriteEndElement();
|
|
||||||
|
|
||||||
return rxw.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
public string LoadPrims(string requestBody, IOSHttpRequest request, IOSHttpResponse response, Scene scene)
|
|
||||||
{
|
|
||||||
IRegionSerialiserModule serialiser = scene.RequestModuleInterface<IRegionSerialiserModule>();
|
|
||||||
if (serialiser != null)
|
|
||||||
serialiser.LoadPrimsFromXml2(scene, new StringReader(requestBody), true);
|
|
||||||
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion POST methods
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,98 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Xml.Serialization;
|
|
||||||
using OpenMetaverse;
|
|
||||||
using OpenSim.Framework;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Regions
|
|
||||||
{
|
|
||||||
[XmlRoot(ElementName="region", IsNullable = false)]
|
|
||||||
public class RegionDetails
|
|
||||||
{
|
|
||||||
public string region_name;
|
|
||||||
public string region_id;
|
|
||||||
public uint region_x;
|
|
||||||
public uint region_y;
|
|
||||||
public string region_owner;
|
|
||||||
public string region_owner_id;
|
|
||||||
public uint region_http_port;
|
|
||||||
public uint region_port;
|
|
||||||
public string region_server_uri;
|
|
||||||
public string region_external_hostname;
|
|
||||||
|
|
||||||
public RegionDetails()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public RegionDetails(RegionInfo regInfo)
|
|
||||||
{
|
|
||||||
region_name = regInfo.RegionName;
|
|
||||||
region_id = regInfo.RegionID.ToString();
|
|
||||||
region_x = regInfo.RegionLocX;
|
|
||||||
region_y = regInfo.RegionLocY;
|
|
||||||
region_owner_id = regInfo.EstateSettings.EstateOwner.ToString();
|
|
||||||
region_http_port = regInfo.HttpPort;
|
|
||||||
region_server_uri = regInfo.ServerURI;
|
|
||||||
region_external_hostname = regInfo.ExternalHostName;
|
|
||||||
|
|
||||||
Uri uri = new Uri(region_server_uri);
|
|
||||||
region_port = (uint)uri.Port;
|
|
||||||
}
|
|
||||||
|
|
||||||
public string this[string idx]
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
switch (idx.ToLower())
|
|
||||||
{
|
|
||||||
case "name":
|
|
||||||
return region_name;
|
|
||||||
case "id":
|
|
||||||
return region_id;
|
|
||||||
case "location":
|
|
||||||
return String.Format("<x>{0}</x><y>{1}</y>", region_x, region_y);
|
|
||||||
case "owner":
|
|
||||||
return region_owner;
|
|
||||||
case "owner_id":
|
|
||||||
return region_owner_id;
|
|
||||||
case "http_port":
|
|
||||||
return region_http_port.ToString();
|
|
||||||
case "server_uri":
|
|
||||||
return region_server_uri;
|
|
||||||
case "external_hostname":
|
|
||||||
case "hostname":
|
|
||||||
return region_external_hostname;
|
|
||||||
|
|
||||||
default:
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,11 +0,0 @@
|
||||||
<Addin id="OpenSim.ApplicationPlugins.Rest.Regions" version="0.1">
|
|
||||||
<Runtime>
|
|
||||||
<Import assembly="OpenSim.ApplicationPlugins.Rest.Regions.dll"/>
|
|
||||||
</Runtime>
|
|
||||||
<Dependencies>
|
|
||||||
<Addin id="OpenSim" version="0.5" />
|
|
||||||
</Dependencies>
|
|
||||||
<Extension path = "/OpenSim/Startup">
|
|
||||||
<Plugin id="RestRegions" type="OpenSim.ApplicationPlugins.Rest.Regions.RestRegionPlugin" />
|
|
||||||
</Extension>
|
|
||||||
</Addin>
|
|
|
@ -1,417 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
* * Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of the OpenSimulator Project nor the
|
|
||||||
* names of its contributors may be used to endorse or promote products
|
|
||||||
* derived from this software without specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
|
||||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
|
||||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.IO;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Xml;
|
|
||||||
using log4net;
|
|
||||||
using Nini.Config;
|
|
||||||
using OpenMetaverse;
|
|
||||||
using OpenSim.Framework;
|
|
||||||
using OpenSim.Framework.Servers;
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest
|
|
||||||
{
|
|
||||||
public abstract class RestPlugin : IApplicationPlugin
|
|
||||||
{
|
|
||||||
#region properties
|
|
||||||
|
|
||||||
protected static readonly ILog m_log =
|
|
||||||
LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
|
||||||
|
|
||||||
private IConfig _config; // Configuration source: Rest Plugins
|
|
||||||
private IConfig _pluginConfig; // Configuration source: Plugin specific
|
|
||||||
private OpenSimBase _app; // The 'server'
|
|
||||||
private BaseHttpServer _httpd; // The server's RPC interface
|
|
||||||
private string _prefix; // URL prefix below
|
|
||||||
// which all REST URLs
|
|
||||||
// are living
|
|
||||||
// private StringWriter _sw = null;
|
|
||||||
// private RestXmlWriter _xw = null;
|
|
||||||
|
|
||||||
private string _godkey;
|
|
||||||
private int _reqk;
|
|
||||||
|
|
||||||
[ThreadStatic]
|
|
||||||
private static string _threadRequestID = String.Empty;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Return an ever increasing request ID for logging
|
|
||||||
/// </summary>
|
|
||||||
protected string RequestID
|
|
||||||
{
|
|
||||||
get { return _reqk++.ToString(); }
|
|
||||||
set { _reqk = Convert.ToInt32(value); }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Thread-constant message IDs for logging.
|
|
||||||
/// </summary>
|
|
||||||
protected string MsgID
|
|
||||||
{
|
|
||||||
get { return String.Format("[REST-{0}] #{1}", Name, _threadRequestID); }
|
|
||||||
set { _threadRequestID = value; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Returns true if Rest Plugins are enabled.
|
|
||||||
/// </summary>
|
|
||||||
public bool PluginsAreEnabled
|
|
||||||
{
|
|
||||||
get { return null != _config; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Returns true if specific Rest Plugin is enabled.
|
|
||||||
/// </summary>
|
|
||||||
public bool IsEnabled
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
return (null != _pluginConfig) && _pluginConfig.GetBoolean("enabled", false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// OpenSimMain application
|
|
||||||
/// </summary>
|
|
||||||
public OpenSimBase App
|
|
||||||
{
|
|
||||||
get { return _app; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// RPC server
|
|
||||||
/// </summary>
|
|
||||||
public BaseHttpServer HttpServer
|
|
||||||
{
|
|
||||||
get { return _httpd; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// URL prefix to use for all REST handlers
|
|
||||||
/// </summary>
|
|
||||||
public string Prefix
|
|
||||||
{
|
|
||||||
get { return _prefix; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Access to GOD password string
|
|
||||||
/// </summary>
|
|
||||||
protected string GodKey
|
|
||||||
{
|
|
||||||
get { return _godkey; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Configuration of the plugin
|
|
||||||
/// </summary>
|
|
||||||
public IConfig Config
|
|
||||||
{
|
|
||||||
get { return _pluginConfig; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Name of the plugin
|
|
||||||
/// </summary>
|
|
||||||
public abstract string Name { get; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Return the config section name
|
|
||||||
/// </summary>
|
|
||||||
public abstract string ConfigName { get; }
|
|
||||||
|
|
||||||
// public XmlTextWriter XmlWriter
|
|
||||||
// {
|
|
||||||
// get
|
|
||||||
// {
|
|
||||||
// if (null == _xw)
|
|
||||||
// {
|
|
||||||
// _sw = new StringWriter();
|
|
||||||
// _xw = new RestXmlWriter(_sw);
|
|
||||||
// _xw.Formatting = Formatting.Indented;
|
|
||||||
// }
|
|
||||||
// return _xw;
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
// public string XmlWriterResult
|
|
||||||
// {
|
|
||||||
// get
|
|
||||||
// {
|
|
||||||
// _xw.Flush();
|
|
||||||
// _xw.Close();
|
|
||||||
// _xw = null;
|
|
||||||
|
|
||||||
// return _sw.ToString();
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
#endregion properties
|
|
||||||
|
|
||||||
#region methods
|
|
||||||
|
|
||||||
// TODO: required by IPlugin, but likely not at all right
|
|
||||||
private string m_version = "0.0";
|
|
||||||
|
|
||||||
public string Version
|
|
||||||
{
|
|
||||||
get { return m_version; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Initialise()
|
|
||||||
{
|
|
||||||
m_log.Info("[RESTPLUGIN]: " + Name + " cannot be default-initialized!");
|
|
||||||
throw new PluginNotInitialisedException(Name);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This method is called by OpenSimMain immediately after loading the
|
|
||||||
/// plugin and after basic server setup, but before running any server commands.
|
|
||||||
/// </summary>
|
|
||||||
/// <remarks>
|
|
||||||
/// Note that entries MUST be added to the active configuration files before
|
|
||||||
/// the plugin can be enabled.
|
|
||||||
/// </remarks>
|
|
||||||
public virtual void Initialise(OpenSimBase openSim)
|
|
||||||
{
|
|
||||||
RequestID = "0";
|
|
||||||
MsgID = RequestID;
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if ((_config = openSim.ConfigSource.Source.Configs["RestPlugins"]) == null)
|
|
||||||
{
|
|
||||||
m_log.WarnFormat("{0} Rest Plugins not configured", MsgID);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!_config.GetBoolean("enabled", false))
|
|
||||||
{
|
|
||||||
//m_log.WarnFormat("{0} Rest Plugins are disabled", MsgID);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
_app = openSim;
|
|
||||||
_httpd = openSim.HttpServer;
|
|
||||||
|
|
||||||
// Retrieve GOD key value, if any.
|
|
||||||
_godkey = _config.GetString("god_key", String.Empty);
|
|
||||||
|
|
||||||
// Retrive prefix if any.
|
|
||||||
_prefix = _config.GetString("prefix", "/admin");
|
|
||||||
|
|
||||||
// Get plugin specific config
|
|
||||||
_pluginConfig = openSim.ConfigSource.Source.Configs[ConfigName];
|
|
||||||
|
|
||||||
m_log.InfoFormat("{0} Rest Plugins Enabled", MsgID);
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
// we can safely ignore this, as it just means that
|
|
||||||
// the key lookup in Configs failed, which signals to
|
|
||||||
// us that noone is interested in our services...they
|
|
||||||
// don't know what they are missing out on...
|
|
||||||
// NOTE: Under the present OpenSimulator implementation it is
|
|
||||||
// not possible for the openSimulator pointer to be null. However
|
|
||||||
// were the implementation to be changed, this could
|
|
||||||
// result in a silent initialization failure. Harmless
|
|
||||||
// except for lack of function and lack of any
|
|
||||||
// diagnostic indication as to why. The same is true if
|
|
||||||
// the HTTP server reference is bad.
|
|
||||||
// We should at least issue a message...
|
|
||||||
m_log.WarnFormat("{0} Initialization failed: {1}", MsgID, e.Message);
|
|
||||||
m_log.DebugFormat("{0} Initialization failed: {1}", MsgID, e.ToString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual void PostInitialise()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
private List<RestStreamHandler> _handlers = new List<RestStreamHandler>();
|
|
||||||
private Dictionary<string, IHttpAgentHandler> _agents = new Dictionary<string, IHttpAgentHandler>();
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Add a REST stream handler to the underlying HTTP server.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="httpMethod">GET/PUT/POST/DELETE or
|
|
||||||
/// similar</param>
|
|
||||||
/// <param name="path">URL prefix</param>
|
|
||||||
/// <param name="method">RestMethod handler doing the actual work</param>
|
|
||||||
public virtual void AddRestStreamHandler(string httpMethod, string path, RestMethod method)
|
|
||||||
{
|
|
||||||
if (!IsEnabled) return;
|
|
||||||
|
|
||||||
if (!path.StartsWith(_prefix))
|
|
||||||
{
|
|
||||||
path = String.Format("{0}{1}", _prefix, path);
|
|
||||||
}
|
|
||||||
|
|
||||||
RestStreamHandler h = new RestStreamHandler(httpMethod, path, method);
|
|
||||||
_httpd.AddStreamHandler(h);
|
|
||||||
_handlers.Add(h);
|
|
||||||
|
|
||||||
m_log.DebugFormat("{0} Added REST handler {1} {2}", MsgID, httpMethod, path);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Add a powerful Agent handler to the underlying HTTP
|
|
||||||
/// server.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="agentName">name of agent handler</param>
|
|
||||||
/// <param name="handler">agent handler method</param>
|
|
||||||
/// <returns>false when the plugin is disabled or the agent
|
|
||||||
/// handler could not be added. Any generated exceptions are
|
|
||||||
/// allowed to drop through to the caller, i.e. ArgumentException.
|
|
||||||
/// </returns>
|
|
||||||
public bool AddAgentHandler(string agentName, IHttpAgentHandler handler)
|
|
||||||
{
|
|
||||||
if (!IsEnabled) return false;
|
|
||||||
_agents.Add(agentName, handler);
|
|
||||||
// return _httpd.AddAgentHandler(agentName, handler);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Remove a powerful Agent handler from the underlying HTTP
|
|
||||||
/// server.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="agentName">name of agent handler</param>
|
|
||||||
/// <param name="handler">agent handler method</param>
|
|
||||||
/// <returns>false when the plugin is disabled or the agent
|
|
||||||
/// handler could not be removed. Any generated exceptions are
|
|
||||||
/// allowed to drop through to the caller, i.e. KeyNotFound.
|
|
||||||
/// </returns>
|
|
||||||
public bool RemoveAgentHandler(string agentName, IHttpAgentHandler handler)
|
|
||||||
{
|
|
||||||
if (!IsEnabled) return false;
|
|
||||||
if (_agents[agentName] == handler)
|
|
||||||
{
|
|
||||||
_agents.Remove(agentName);
|
|
||||||
// return _httpd.RemoveAgentHandler(agentName, handler);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Check whether the HTTP request came from god; that is, is
|
|
||||||
/// the god_key as configured in the config section supplied
|
|
||||||
/// via X-OpenSim-Godkey?
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="request">HTTP request header</param>
|
|
||||||
/// <returns>true when the HTTP request came from god.</returns>
|
|
||||||
protected bool IsGod(IOSHttpRequest request)
|
|
||||||
{
|
|
||||||
string[] keys = request.Headers.GetValues("X-OpenSim-Godkey");
|
|
||||||
if (null == keys) return false;
|
|
||||||
|
|
||||||
// we take the last key supplied
|
|
||||||
return keys[keys.Length - 1] == _godkey;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Checks wether the X-OpenSim-Password value provided in the
|
|
||||||
/// HTTP header is indeed the password on file for the avatar
|
|
||||||
/// specified by the UUID
|
|
||||||
/// </summary>
|
|
||||||
protected bool IsVerifiedUser(IOSHttpRequest request, UUID uuid)
|
|
||||||
{
|
|
||||||
// XXX under construction
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Clean up and remove all handlers that were added earlier.
|
|
||||||
/// </summary>
|
|
||||||
public virtual void Close()
|
|
||||||
{
|
|
||||||
foreach (RestStreamHandler h in _handlers)
|
|
||||||
{
|
|
||||||
_httpd.RemoveStreamHandler(h.HttpMethod, h.Path);
|
|
||||||
}
|
|
||||||
_handlers = null;
|
|
||||||
// foreach (KeyValuePair<string, IHttpAgentHandler> h in _agents)
|
|
||||||
// {
|
|
||||||
// _httpd.RemoveAgentHandler(h.Key, h.Value);
|
|
||||||
// }
|
|
||||||
_agents = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual void Dispose()
|
|
||||||
{
|
|
||||||
Close();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Return a failure message.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="method">origin of the failure message</param>
|
|
||||||
/// <param name="message">failure message</param>
|
|
||||||
/// <remarks>This should probably set a return code as
|
|
||||||
/// well. (?)</remarks>
|
|
||||||
protected string Failure(IOSHttpResponse response, OSHttpStatusCode status,
|
|
||||||
string method, string format, params string[] msg)
|
|
||||||
{
|
|
||||||
string m = String.Format(format, msg);
|
|
||||||
|
|
||||||
response.StatusCode = (int) status;
|
|
||||||
response.StatusDescription = m;
|
|
||||||
|
|
||||||
m_log.ErrorFormat("{0} {1} failed: {2}", MsgID, method, m);
|
|
||||||
return String.Format("<error>{0}</error>", m);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Return a failure message.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="method">origin of the failure message</param>
|
|
||||||
/// <param name="e">exception causing the failure message</param>
|
|
||||||
/// <remarks>This should probably set a return code as
|
|
||||||
/// well. (?)</remarks>
|
|
||||||
public string Failure(IOSHttpResponse response, OSHttpStatusCode status,
|
|
||||||
string method, Exception e)
|
|
||||||
{
|
|
||||||
string m = String.Format("exception occurred: {0}", e.Message);
|
|
||||||
|
|
||||||
response.StatusCode = (int) status;
|
|
||||||
response.StatusDescription = m;
|
|
||||||
|
|
||||||
m_log.DebugFormat("{0} {1} failed: {2}", MsgID, method, e.ToString());
|
|
||||||
m_log.ErrorFormat("{0} {1} failed: {2}", MsgID, method, e.Message);
|
|
||||||
|
|
||||||
return String.Format("<error>{0}</error>", e.Message);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion methods
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,276 +0,0 @@
|
||||||
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema">
|
|
||||||
|
|
||||||
<xsd:annotation>
|
|
||||||
<xsd:documentation xml:lang="en">
|
|
||||||
Open Simulator Export/Import XML schema
|
|
||||||
August 2008
|
|
||||||
</xsd:documentation>
|
|
||||||
</xsd:annotation>
|
|
||||||
|
|
||||||
<!-- WARNING!!!
|
|
||||||
This is currently a draft, it does not reflect
|
|
||||||
what is exported, nor what will be understood
|
|
||||||
on import. It is included as a working document
|
|
||||||
and this comment will be removed at such time as
|
|
||||||
the schema corresponds to reality.
|
|
||||||
-->
|
|
||||||
|
|
||||||
<!--
|
|
||||||
REST-related information
|
|
||||||
Inventory data is always framed by an
|
|
||||||
inventory element. Consists of zero or
|
|
||||||
more elements representing either folders
|
|
||||||
or items within those folders. The inventory
|
|
||||||
element represents the "real" root folder.
|
|
||||||
-->
|
|
||||||
|
|
||||||
<xsd:element name="inventory" type="inventory_ct" />
|
|
||||||
|
|
||||||
<!--
|
|
||||||
The inventory complex type is just an arbitrary
|
|
||||||
sequence of folders and items. In reality it is
|
|
||||||
typically just folders. Both item and folder
|
|
||||||
have corresponding complex types. It is distinct
|
|
||||||
from folders insofar as it has no other defining
|
|
||||||
attributes.
|
|
||||||
-->
|
|
||||||
|
|
||||||
<xsd:complexType name="inventory_ct">
|
|
||||||
<xsd:element name="folder" type="folder_ct" maxOccurs="unbounded"/>
|
|
||||||
<xsd:element name="item" type="item_ct" maxOccurs="unbounded" />
|
|
||||||
</xsd:complexType>
|
|
||||||
|
|
||||||
<xsd:complexType name="folder_ct">
|
|
||||||
<xsd:attribute name="UUID" type="uuid_st" />
|
|
||||||
<xsd:attribute name="name" type="name_st" />
|
|
||||||
<xsd:attribute name="type" type="folder_type_st" />
|
|
||||||
<xsd:attribute name="description" type="xsd:string" /> <!-- added -->
|
|
||||||
<xsd:attribute name="version" type="unsignedShort" />
|
|
||||||
<xsd:attribute name="owner" type="uuid_st" />
|
|
||||||
|
|
||||||
<xsd:attribute name="creator" type="uuid_st" /> <!-- added -->
|
|
||||||
<xsd:attribute name="creationdate" type="date_st" /> <!-- added -->
|
|
||||||
|
|
||||||
<xsd:attribute name="parent" type="uuid_st" />
|
|
||||||
|
|
||||||
<xsd:element name="permissions" type="permissions_ct" maxOccurs="unbounded" /> <!-- added -->
|
|
||||||
<xsd:element name="folder" type="folder_ct" maxOccurs="unbounded" />
|
|
||||||
<xsd:element name="item" type="item_ct" maxOccurs="unbounded" />
|
|
||||||
</xsd:complexType>
|
|
||||||
|
|
||||||
<xsd:complexType name="item_ct">
|
|
||||||
<xsd:attribute name="UUID" type="uuid_st" />
|
|
||||||
<xsd:attribute name="name" type="name_st" />
|
|
||||||
<xsd:attribute name="type" type="inventory_type_st" />
|
|
||||||
<xsd:attribute name="description" type="xsd:string" />
|
|
||||||
<xsd:attribute name="version" type="unsignedShort" /> <!-- added -->
|
|
||||||
<xsd:attribute name="owner" type="uuid_st" />
|
|
||||||
|
|
||||||
<xsd:attribute name="creator" type="uuid_st" />
|
|
||||||
<xsd:attribute name="creationdate" type="date_st" />
|
|
||||||
|
|
||||||
<xsd:attribute name="folder" type="uuid_st" />
|
|
||||||
<xsd:attribute name="groupid" type="uuid_st" />
|
|
||||||
<xsd:attribute name="groupowned" type="xsd:boolean" />
|
|
||||||
<xsd:attribute name="saletype" type="sale_st" />
|
|
||||||
<xsd:attribute name="saleprice" type="xsd:decimal" />
|
|
||||||
|
|
||||||
<xsd:element name="permissions" type="permissions_ct" maxOccurs="unbounded" />
|
|
||||||
</xsd:complexType>
|
|
||||||
|
|
||||||
<xsd:complexType name="asset_ct">
|
|
||||||
<xsd:attribute name="UUID" type="uuid_st" />
|
|
||||||
<xsd:attribute name="name" type="name_st" />
|
|
||||||
<xsd:attribute name="type" type="asset_type_st" />
|
|
||||||
<xsd:attribute name="description" type="xsd:string" />
|
|
||||||
<xsd:attribute name="version" type="unsignedShort" /> <!-- added -->
|
|
||||||
<xsd:attribute name="owner" type="uuid_st" />
|
|
||||||
|
|
||||||
<xsd:attribute name="creator" type="uuid_st" />
|
|
||||||
<xsd:attribute name="creationdate" type="date_st" />
|
|
||||||
|
|
||||||
<xsd:attribute name="temporary" type="xsd:boolean" />
|
|
||||||
<xsd:attribute name="local" type="xsd:boolean" />
|
|
||||||
<xsd:attribute name="inline" type="xsd:boolean" />
|
|
||||||
</xsd:complexType>
|
|
||||||
|
|
||||||
<!-- Constrained Simple Data types -->
|
|
||||||
|
|
||||||
<!--
|
|
||||||
We need to specify name as a simple type because on
|
|
||||||
some platforms it is constrained by a certain length
|
|
||||||
limitation. For completeness we indicate that whitespace
|
|
||||||
should be preserved exactly as specified.
|
|
||||||
-->
|
|
||||||
|
|
||||||
<xsd:simpleType name="name_st">
|
|
||||||
<xsd:restriction base="xsd:string">
|
|
||||||
<whiteSpace value="preserve" />
|
|
||||||
<minLength value="0" />
|
|
||||||
<maxLength value="64" />
|
|
||||||
</xsd:restriction>
|
|
||||||
</xsd:simpleType>
|
|
||||||
|
|
||||||
<!--
|
|
||||||
Type information in the folder is meant to indicate
|
|
||||||
the preferred asset type for this folder. As such, that
|
|
||||||
currently corresponds to the type values allowed for
|
|
||||||
assets, however that is not mandated, so for
|
|
||||||
now at least I'll represent this as a distinct
|
|
||||||
enumeration.
|
|
||||||
This seems inappropriate; it seems like the folder's
|
|
||||||
content should reflect the InventoryType classifications
|
|
||||||
rather than the asset types.
|
|
||||||
-->
|
|
||||||
|
|
||||||
<xsd:simpleType name="folder_type_st">
|
|
||||||
<xsd:restriction base="xsd:string">
|
|
||||||
<xsd:enumeration value="Texture" />
|
|
||||||
<xsd:enumeration value="Sound" />
|
|
||||||
<xsd:enumeration value="CallingCard" />
|
|
||||||
<xsd:enumeration value="Landmark" />
|
|
||||||
<xsd:enumeration value="Script" />
|
|
||||||
<xsd:enumeration value="Clothing" />
|
|
||||||
<xsd:enumeration value="Object" />
|
|
||||||
<xsd:enumeration value="Notecard" />
|
|
||||||
<xsd:enumeration value="LSLText" />
|
|
||||||
<xsd:enumeration value="LSLByteCode" />
|
|
||||||
<xsd:enumeration value="TextureTGA" />
|
|
||||||
<xsd:enumeration value="BodyPart" />
|
|
||||||
<xsd:enumeration value="SoundWAV" />
|
|
||||||
<xsd:enumeration value="ImageTGA" />
|
|
||||||
<xsd:enumeration value="ImageJPEG" />
|
|
||||||
<xsd:enumeration value="Animation" />
|
|
||||||
<xsd:enumeration value="Gesture" />
|
|
||||||
<xsd:enumeration value="Simstate" />
|
|
||||||
<xsd:enumeration value="Unknown" />
|
|
||||||
<xsd:enumeration value="LostAndFoundFolder" />
|
|
||||||
<xsd:enumeration value="SnapshotFolder" />
|
|
||||||
<xsd:enumeration value="TrashFolder" />
|
|
||||||
<xsd:enumeration value="Folder" />
|
|
||||||
<xsd:enumeration value="RootFolder" />
|
|
||||||
</xsd:restriction>
|
|
||||||
</xsd:simpleType>
|
|
||||||
|
|
||||||
<!--
|
|
||||||
Inventory item type designates an asset class, rather
|
|
||||||
than a specific asset type. For example, "SnapShot"
|
|
||||||
might include a number of asset types such as JPEG,
|
|
||||||
TGA, etc.. This is not a consistent interpretation,
|
|
||||||
classifications such as LostAndFound are meta-types
|
|
||||||
relative to asset classes.
|
|
||||||
|
|
||||||
These types should be abstract and not be tied to a
|
|
||||||
specific platform. A world's import facility should be
|
|
||||||
responsible for mapping these to meaningful internal
|
|
||||||
representations.
|
|
||||||
|
|
||||||
These types were based on information in:
|
|
||||||
libsecondlife/InventoryManager.cs
|
|
||||||
-->
|
|
||||||
|
|
||||||
<xsd:simpleType name="inventory_type_st">
|
|
||||||
<xsd:restriction base="xsd:string">
|
|
||||||
<xsd:enumeration value="Texture" />
|
|
||||||
<xsd:enumeration value="Sound" />
|
|
||||||
<xsd:enumeration value="CallingCard" />
|
|
||||||
<xsd:enumeration value="Landmark" />
|
|
||||||
<xsd:enumeration value="Script" />
|
|
||||||
<xsd:enumeration value="Clothing" />
|
|
||||||
<xsd:enumeration value="Object" />
|
|
||||||
<xsd:enumeration value="Notecard" />
|
|
||||||
<xsd:enumeration value="LSL" />
|
|
||||||
<xsd:enumeration value="LSLBytecode" />
|
|
||||||
<xsd:enumeration value="TextureTGA" />
|
|
||||||
<xsd:enumeration value="BodyPart" />
|
|
||||||
<xsd:enumeration value="Snapshot" />
|
|
||||||
<xsd:enumeration value="Attachment" />
|
|
||||||
<xsd:enumeration value="Wearable" />
|
|
||||||
<xsd:enumeration value="Animation" />
|
|
||||||
<xsd:enumeration value="Gesture" />
|
|
||||||
<xsd:enumeration value="Folder" />
|
|
||||||
<xsd:enumeration value="Unknown" />
|
|
||||||
<xsd:enumeration value="LostAndFound" />
|
|
||||||
<xsd:enumeration value="Trash" />
|
|
||||||
<xsd:enumeration value="Root" />
|
|
||||||
</xsd:restriction>
|
|
||||||
</xsd:simpleType>
|
|
||||||
|
|
||||||
<!--
|
|
||||||
The asset types seem to be even more disarrayed than
|
|
||||||
the inventory types. It seems to be little more than
|
|
||||||
a reiteration of the inventory type information,
|
|
||||||
which adds little or nothing to the overall data
|
|
||||||
model.
|
|
||||||
|
|
||||||
Of course, given that these are drawn from the
|
|
||||||
libsecondlife definitions, we aren't at liberty to
|
|
||||||
simply redefine them in place. But the XML definitions
|
|
||||||
here could be made more useful.
|
|
||||||
|
|
||||||
These types were based on information in:
|
|
||||||
libsecondlife/AssetManager.cs
|
|
||||||
-->
|
|
||||||
|
|
||||||
<xsd:simpleType name="asset_type_st">
|
|
||||||
<xsd:restriction base="xsd:string">
|
|
||||||
<xsd:enumeration value="Texture" />
|
|
||||||
<xsd:enumeration value="Sound" />
|
|
||||||
<xsd:enumeration value="CallingCard" />
|
|
||||||
<xsd:enumeration value="Landmark" />
|
|
||||||
<xsd:enumeration value="Script" />
|
|
||||||
<xsd:enumeration value="Clothing" />
|
|
||||||
<xsd:enumeration value="Object" />
|
|
||||||
<xsd:enumeration value="Notecard" />
|
|
||||||
<xsd:enumeration value="LSLText" />
|
|
||||||
<xsd:enumeration value="LSLByteCode" />
|
|
||||||
<xsd:enumeration value="TextureTGA" />
|
|
||||||
<xsd:enumeration value="BodyPart" />
|
|
||||||
<xsd:enumeration value="SoundWAV" />
|
|
||||||
<xsd:enumeration value="ImageTGA" />
|
|
||||||
<xsd:enumeration value="ImageJPEG" />
|
|
||||||
<xsd:enumeration value="Animation" />
|
|
||||||
<xsd:enumeration value="Gesture" />
|
|
||||||
<xsd:enumeration value="Simstate" />
|
|
||||||
<xsd:enumeration value="Unknown" />
|
|
||||||
<xsd:enumeration value="LostAndFoundFolder" />
|
|
||||||
<xsd:enumeration value="SnapshotFolder" />
|
|
||||||
<xsd:enumeration value="TrashFolder" />
|
|
||||||
<xsd:enumeration value="Folder" />
|
|
||||||
<xsd:enumeration value="RootFolder" />
|
|
||||||
</xsd:restriction>
|
|
||||||
</xsd:simpleType>
|
|
||||||
|
|
||||||
<!-- This is describing the apparent form of a UUID. If
|
|
||||||
we ever want a more metaphysical definition we'll
|
|
||||||
need to add to it.
|
|
||||||
-->
|
|
||||||
|
|
||||||
<xsd:simpleType name="uuid_st">
|
|
||||||
<xsd:restriction base="xsd:string">
|
|
||||||
<xsd:pattern value="[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}"/>
|
|
||||||
</xsd:restriction>
|
|
||||||
</xsd:simpleType>
|
|
||||||
|
|
||||||
<!-- This constrains the date representation. Currently
|
|
||||||
it is simply an integer representing the elapsed
|
|
||||||
?? since ??.
|
|
||||||
-->
|
|
||||||
|
|
||||||
<xsd:simpleType name="date_st">
|
|
||||||
<xsd:restriction base="xsd:positiveInteger">
|
|
||||||
</xsd:restriction>
|
|
||||||
</xsd:simpleType>
|
|
||||||
|
|
||||||
<!-- This constrains the representation of sale price.
|
|
||||||
Currently it is a simple decimal with no unit
|
|
||||||
specified.
|
|
||||||
Issues: interoperability.
|
|
||||||
-->
|
|
||||||
|
|
||||||
<xsd:simpleType name="sale_st">
|
|
||||||
<xsd:restriction base="xsd:decimal">
|
|
||||||
</xsd:restriction>
|
|
||||||
</xsd:simpleType>
|
|
||||||
|
|
||||||
</xsd:schema>
|
|
|
@ -30,6 +30,7 @@ using System.Collections;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
|
using System.Threading;
|
||||||
using log4net;
|
using log4net;
|
||||||
using Nini.Config;
|
using Nini.Config;
|
||||||
using OpenMetaverse;
|
using OpenMetaverse;
|
||||||
|
@ -68,6 +69,7 @@ namespace OpenSim.Framework.Capabilities
|
||||||
private IHttpServer m_httpListener;
|
private IHttpServer m_httpListener;
|
||||||
private UUID m_agentID;
|
private UUID m_agentID;
|
||||||
private string m_regionName;
|
private string m_regionName;
|
||||||
|
private ManualResetEvent m_capsActive = new ManualResetEvent(false);
|
||||||
|
|
||||||
public UUID AgentID
|
public UUID AgentID
|
||||||
{
|
{
|
||||||
|
@ -171,5 +173,16 @@ namespace OpenSim.Framework.Capabilities
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void Activate()
|
||||||
|
{
|
||||||
|
m_capsActive.Set();
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool WaitForActivation()
|
||||||
|
{
|
||||||
|
// Wait for 30s. If that elapses, return false and run without caps
|
||||||
|
return m_capsActive.WaitOne(30000);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -240,6 +240,11 @@ namespace OpenSim.Capabilities.Handlers
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
// Handle the case where no second range value was given. This is equivalent to requesting
|
||||||
|
// the rest of the entity.
|
||||||
|
if (end == -1)
|
||||||
|
end = int.MaxValue;
|
||||||
|
|
||||||
end = Utils.Clamp(end, 0, texture.Data.Length - 1);
|
end = Utils.Clamp(end, 0, texture.Data.Length - 1);
|
||||||
start = Utils.Clamp(start, 0, end);
|
start = Utils.Clamp(start, 0, end);
|
||||||
int len = end - start + 1;
|
int len = end - start + 1;
|
||||||
|
@ -298,15 +303,43 @@ namespace OpenSim.Capabilities.Handlers
|
||||||
// texture.FullID, range, response.StatusCode, response.ContentLength, texture.Data.Length);
|
// texture.FullID, range, response.StatusCode, response.ContentLength, texture.Data.Length);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Parse a range header.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// As per http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html,
|
||||||
|
/// this obeys range headers with two values (e.g. 533-4165) and no second value (e.g. 533-).
|
||||||
|
/// Where there is no value, -1 is returned.
|
||||||
|
/// FIXME: Need to cover the case where only a second value is specified (e.g. -4165), probably by returning -1
|
||||||
|
/// for start.</remarks>
|
||||||
|
/// <returns></returns>
|
||||||
|
/// <param name='header'></param>
|
||||||
|
/// <param name='start'>Start of the range. Undefined if this was not a number.</param>
|
||||||
|
/// <param name='end'>End of the range. Will be -1 if no end specified. Undefined if there was a raw string but this was not a number.</param>
|
||||||
private bool TryParseRange(string header, out int start, out int end)
|
private bool TryParseRange(string header, out int start, out int end)
|
||||||
{
|
{
|
||||||
|
start = end = 0;
|
||||||
|
|
||||||
if (header.StartsWith("bytes="))
|
if (header.StartsWith("bytes="))
|
||||||
{
|
{
|
||||||
string[] rangeValues = header.Substring(6).Split('-');
|
string[] rangeValues = header.Substring(6).Split('-');
|
||||||
|
|
||||||
if (rangeValues.Length == 2)
|
if (rangeValues.Length == 2)
|
||||||
{
|
{
|
||||||
if (Int32.TryParse(rangeValues[0], out start) && Int32.TryParse(rangeValues[1], out end))
|
if (!Int32.TryParse(rangeValues[0], out start))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
string rawEnd = rangeValues[1];
|
||||||
|
|
||||||
|
if (rawEnd == "")
|
||||||
|
{
|
||||||
|
end = -1;
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
else if (Int32.TryParse(rawEnd, out end))
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -43,7 +43,7 @@ using OpenSim.Tests.Common.Mock;
|
||||||
namespace OpenSim.Capabilities.Handlers.GetTexture.Tests
|
namespace OpenSim.Capabilities.Handlers.GetTexture.Tests
|
||||||
{
|
{
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class GetTextureHandlerTests
|
public class GetTextureHandlerTests : OpenSimTestCase
|
||||||
{
|
{
|
||||||
[Test]
|
[Test]
|
||||||
public void TestTextureNotFound()
|
public void TestTextureNotFound()
|
||||||
|
|
|
@ -29,5 +29,5 @@ using System.Runtime.InteropServices;
|
||||||
// Build Number
|
// Build Number
|
||||||
// Revision
|
// Revision
|
||||||
//
|
//
|
||||||
[assembly: AssemblyVersion("0.7.5.*")]
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
|
||||||
|
|
|
@ -435,4 +435,4 @@ namespace OpenSim.Capabilities.Handlers
|
||||||
return llsdItem;
|
return llsdItem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,5 +29,5 @@ using System.Runtime.InteropServices;
|
||||||
// Build Number
|
// Build Number
|
||||||
// Revision
|
// Revision
|
||||||
//
|
//
|
||||||
[assembly: AssemblyVersion("0.7.5.*")]
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
[assembly: AssemblyFileVersion("1.0.0.0")]
|
||||||
|
|
|
@ -29,5 +29,5 @@ using System.Runtime.InteropServices;
|
||||||
// Build Number
|
// Build Number
|
||||||
// Revision
|
// Revision
|
||||||
//
|
//
|
||||||
[assembly: AssemblyVersion("0.7.5.*")]
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
[assembly: AssemblyFileVersion("1.0.0.0")]
|
||||||
|
|
|
@ -44,7 +44,6 @@ namespace OpenSim.ConsoleClient
|
||||||
ReplyDelegate action)
|
ReplyDelegate action)
|
||||||
{
|
{
|
||||||
WebRequest request = WebRequest.Create(requestUrl);
|
WebRequest request = WebRequest.Create(requestUrl);
|
||||||
WebResponse response = null;
|
|
||||||
|
|
||||||
request.Method = "POST";
|
request.Method = "POST";
|
||||||
|
|
||||||
|
@ -64,16 +63,18 @@ namespace OpenSim.ConsoleClient
|
||||||
{
|
{
|
||||||
string reply = String.Empty;
|
string reply = String.Empty;
|
||||||
|
|
||||||
response = request.EndGetResponse(ar);
|
using (WebResponse response = request.EndGetResponse(ar))
|
||||||
|
|
||||||
try
|
|
||||||
{
|
{
|
||||||
StreamReader r = new StreamReader(response.GetResponseStream());
|
try
|
||||||
reply = r.ReadToEnd();
|
{
|
||||||
|
using (Stream s = response.GetResponseStream())
|
||||||
|
using (StreamReader r = new StreamReader(s))
|
||||||
|
reply = r.ReadToEnd();
|
||||||
|
|
||||||
}
|
}
|
||||||
catch (System.InvalidOperationException)
|
catch (System.InvalidOperationException)
|
||||||
{
|
{
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
action(requestUrl, data, reply);
|
action(requestUrl, data, reply);
|
||||||
|
|
|
@ -0,0 +1,144 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using OpenSim.Data;
|
||||||
|
using OpenMetaverse;
|
||||||
|
|
||||||
|
namespace OpenSim.Data
|
||||||
|
{
|
||||||
|
public class GroupData
|
||||||
|
{
|
||||||
|
public UUID GroupID;
|
||||||
|
public Dictionary<string, string> Data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class MembershipData
|
||||||
|
{
|
||||||
|
public UUID GroupID;
|
||||||
|
public string PrincipalID;
|
||||||
|
public Dictionary<string, string> Data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class RoleData
|
||||||
|
{
|
||||||
|
public UUID GroupID;
|
||||||
|
public UUID RoleID;
|
||||||
|
public Dictionary<string, string> Data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class RoleMembershipData
|
||||||
|
{
|
||||||
|
public UUID GroupID;
|
||||||
|
public UUID RoleID;
|
||||||
|
public string PrincipalID;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class PrincipalData
|
||||||
|
{
|
||||||
|
public string PrincipalID;
|
||||||
|
public UUID ActiveGroupID;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class InvitationData
|
||||||
|
{
|
||||||
|
public UUID InviteID;
|
||||||
|
public UUID GroupID;
|
||||||
|
public UUID RoleID;
|
||||||
|
public string PrincipalID;
|
||||||
|
public Dictionary<string, string> Data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class NoticeData
|
||||||
|
{
|
||||||
|
public UUID GroupID;
|
||||||
|
public UUID NoticeID;
|
||||||
|
public Dictionary<string, string> Data;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public interface IGroupsData
|
||||||
|
{
|
||||||
|
// groups table
|
||||||
|
bool StoreGroup(GroupData data);
|
||||||
|
GroupData RetrieveGroup(UUID groupID);
|
||||||
|
GroupData RetrieveGroup(string name);
|
||||||
|
GroupData[] RetrieveGroups(string pattern);
|
||||||
|
bool DeleteGroup(UUID groupID);
|
||||||
|
int GroupsCount();
|
||||||
|
|
||||||
|
// membership table
|
||||||
|
MembershipData RetrieveMember(UUID groupID, string pricipalID);
|
||||||
|
MembershipData[] RetrieveMembers(UUID groupID);
|
||||||
|
MembershipData[] RetrieveMemberships(string pricipalID);
|
||||||
|
bool StoreMember(MembershipData data);
|
||||||
|
bool DeleteMember(UUID groupID, string pricipalID);
|
||||||
|
int MemberCount(UUID groupID);
|
||||||
|
|
||||||
|
// roles table
|
||||||
|
bool StoreRole(RoleData data);
|
||||||
|
RoleData RetrieveRole(UUID groupID, UUID roleID);
|
||||||
|
RoleData[] RetrieveRoles(UUID groupID);
|
||||||
|
bool DeleteRole(UUID groupID, UUID roleID);
|
||||||
|
int RoleCount(UUID groupID);
|
||||||
|
|
||||||
|
// rolememberhip table
|
||||||
|
RoleMembershipData[] RetrieveRolesMembers(UUID groupID);
|
||||||
|
RoleMembershipData[] RetrieveRoleMembers(UUID groupID, UUID roleID);
|
||||||
|
RoleMembershipData[] RetrieveMemberRoles(UUID groupID, string principalID);
|
||||||
|
RoleMembershipData RetrieveRoleMember(UUID groupID, UUID roleID, string principalID);
|
||||||
|
int RoleMemberCount(UUID groupID, UUID roleID);
|
||||||
|
bool StoreRoleMember(RoleMembershipData data);
|
||||||
|
bool DeleteRoleMember(RoleMembershipData data);
|
||||||
|
bool DeleteMemberAllRoles(UUID groupID, string principalID);
|
||||||
|
|
||||||
|
// principals table
|
||||||
|
bool StorePrincipal(PrincipalData data);
|
||||||
|
PrincipalData RetrievePrincipal(string principalID);
|
||||||
|
bool DeletePrincipal(string principalID);
|
||||||
|
|
||||||
|
// invites table
|
||||||
|
bool StoreInvitation(InvitationData data);
|
||||||
|
InvitationData RetrieveInvitation(UUID inviteID);
|
||||||
|
InvitationData RetrieveInvitation(UUID groupID, string principalID);
|
||||||
|
bool DeleteInvite(UUID inviteID);
|
||||||
|
void DeleteOldInvites();
|
||||||
|
|
||||||
|
// notices table
|
||||||
|
bool StoreNotice(NoticeData data);
|
||||||
|
NoticeData RetrieveNotice(UUID noticeID);
|
||||||
|
NoticeData[] RetrieveNotices(UUID groupID);
|
||||||
|
bool DeleteNotice(UUID noticeID);
|
||||||
|
void DeleteOldNotices();
|
||||||
|
|
||||||
|
// combinations
|
||||||
|
MembershipData RetrievePrincipalGroupMembership(string principalID, UUID groupID);
|
||||||
|
MembershipData[] RetrievePrincipalGroupMemberships(string principalID);
|
||||||
|
|
||||||
|
// Misc
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) Contributors, http://opensimulator.org/
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
*
|
*
|
||||||
|
@ -25,22 +25,25 @@
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using OpenSim.Data;
|
||||||
using OpenMetaverse;
|
using OpenMetaverse;
|
||||||
|
|
||||||
namespace OpenSim.Region.Framework.Scenes.Scripting
|
namespace OpenSim.Data
|
||||||
{
|
{
|
||||||
public interface IScriptHost
|
public class OfflineIMData
|
||||||
{
|
{
|
||||||
string Name { get; set; }
|
public UUID PrincipalID;
|
||||||
string Description { get; set; }
|
public Dictionary<string, string> Data;
|
||||||
|
}
|
||||||
|
|
||||||
UUID UUID { get; }
|
|
||||||
UUID OwnerID { get; }
|
|
||||||
UUID CreatorID { get; }
|
|
||||||
Vector3 AbsolutePosition { get; }
|
|
||||||
|
|
||||||
string SitName { get; set; }
|
public interface IOfflineIMData
|
||||||
string TouchName { get; set; }
|
{
|
||||||
void SetText(string text, Vector3 color, double alpha);
|
OfflineIMData[] Get(string field, string val);
|
||||||
|
long GetCount(string field, string key);
|
||||||
|
bool Store(OfflineIMData data);
|
||||||
|
bool Delete(string field, string val);
|
||||||
|
void DeleteOld();
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -53,5 +53,6 @@ namespace OpenSim.Data
|
||||||
bool ReportAgent(UUID sessionID, UUID regionID);
|
bool ReportAgent(UUID sessionID, UUID regionID);
|
||||||
PresenceData[] Get(string field, string data);
|
PresenceData[] Get(string field, string data);
|
||||||
bool Delete(string field, string val);
|
bool Delete(string field, string val);
|
||||||
|
bool VerifyAgent(UUID agentId, UUID secureSessionID);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,35 +25,47 @@
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using OpenMetaverse;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest.Inventory
|
namespace OpenSim.Data
|
||||||
{
|
{
|
||||||
|
public class XGroup
|
||||||
/// <remarks>
|
|
||||||
/// The handler delegates are not noteworthy. The allocator allows
|
|
||||||
/// a given handler to optionally subclass the base RequestData
|
|
||||||
/// structure to carry any locally required per-request state
|
|
||||||
/// needed.
|
|
||||||
/// </remarks>
|
|
||||||
|
|
||||||
public delegate void RestMethodHandler(RequestData rdata);
|
|
||||||
public delegate RequestData RestMethodAllocator(OSHttpRequest request, OSHttpResponse response, string path);
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This interface exports the generic plugin-handling services
|
|
||||||
/// available to each loaded REST services module (IRest implementation)
|
|
||||||
/// </summary>
|
|
||||||
|
|
||||||
internal interface IRestHandler
|
|
||||||
{
|
{
|
||||||
|
public UUID groupID;
|
||||||
|
public UUID ownerRoleID;
|
||||||
|
public string name;
|
||||||
|
public string charter;
|
||||||
|
public bool showInList;
|
||||||
|
public UUID insigniaID;
|
||||||
|
public int membershipFee;
|
||||||
|
public bool openEnrollment;
|
||||||
|
public bool allowPublish;
|
||||||
|
public bool maturePublish;
|
||||||
|
public UUID founderID;
|
||||||
|
public ulong everyonePowers;
|
||||||
|
public ulong ownersPowers;
|
||||||
|
|
||||||
string MsgId { get; }
|
public XGroup Clone()
|
||||||
string RequestId { get; }
|
{
|
||||||
|
return (XGroup)MemberwiseClone();
|
||||||
void AddPathHandler(RestMethodHandler mh, string path, RestMethodAllocator ma);
|
}
|
||||||
void AddStreamHandler(string httpMethod, string path, RestMethod method);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
/// <summary>
|
||||||
|
/// Early stub interface for groups data, not final.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Currently in-use only for regression test purposes. Needs to be filled out over time.
|
||||||
|
/// </remarks>
|
||||||
|
public interface IXGroupData
|
||||||
|
{
|
||||||
|
bool StoreGroup(XGroup group);
|
||||||
|
XGroup[] GetGroups(string field, string val);
|
||||||
|
XGroup[] GetGroups(string[] fields, string[] vals);
|
||||||
|
bool DeleteGroups(string field, string val);
|
||||||
|
bool DeleteGroups(string[] fields, string[] vals);
|
||||||
|
}
|
||||||
|
}
|
|
@ -163,14 +163,18 @@ namespace OpenSim.Data.MSSQL
|
||||||
if (asset.Name.Length > 64)
|
if (asset.Name.Length > 64)
|
||||||
{
|
{
|
||||||
assetName = asset.Name.Substring(0, 64);
|
assetName = asset.Name.Substring(0, 64);
|
||||||
m_log.Warn("[ASSET DB]: Name field truncated from " + asset.Name.Length + " to " + assetName.Length + " characters on add");
|
m_log.WarnFormat(
|
||||||
|
"[ASSET DB]: Name '{0}' for asset {1} truncated from {2} to {3} characters on add",
|
||||||
|
asset.Name, asset.ID, asset.Name.Length, assetName.Length);
|
||||||
}
|
}
|
||||||
|
|
||||||
string assetDescription = asset.Description;
|
string assetDescription = asset.Description;
|
||||||
if (asset.Description.Length > 64)
|
if (asset.Description.Length > 64)
|
||||||
{
|
{
|
||||||
assetDescription = asset.Description.Substring(0, 64);
|
assetDescription = asset.Description.Substring(0, 64);
|
||||||
m_log.Warn("[ASSET DB]: Description field truncated from " + asset.Description.Length + " to " + assetDescription.Length + " characters on add");
|
m_log.WarnFormat(
|
||||||
|
"[ASSET DB]: Description '{0}' for asset {1} truncated from {2} to {3} characters on add",
|
||||||
|
asset.Description, asset.ID, asset.Description.Length, assetDescription.Length);
|
||||||
}
|
}
|
||||||
|
|
||||||
using (SqlConnection conn = new SqlConnection(m_connectionString))
|
using (SqlConnection conn = new SqlConnection(m_connectionString))
|
||||||
|
|
|
@ -100,5 +100,18 @@ namespace OpenSim.Data.MSSQL
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public bool VerifyAgent(UUID agentId, UUID secureSessionID)
|
||||||
|
{
|
||||||
|
PresenceData[] ret = Get("SecureSessionID",
|
||||||
|
secureSessionID.ToString());
|
||||||
|
|
||||||
|
if (ret.Length == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if(ret[0].UserID != agentId.ToString())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -351,7 +351,8 @@ IF EXISTS (SELECT UUID FROM prims WHERE UUID = @UUID)
|
||||||
ScriptAccessPin = @ScriptAccessPin, AllowedDrop = @AllowedDrop, DieAtEdge = @DieAtEdge, SalePrice = @SalePrice,
|
ScriptAccessPin = @ScriptAccessPin, AllowedDrop = @AllowedDrop, DieAtEdge = @DieAtEdge, SalePrice = @SalePrice,
|
||||||
SaleType = @SaleType, ColorR = @ColorR, ColorG = @ColorG, ColorB = @ColorB, ColorA = @ColorA, ParticleSystem = @ParticleSystem,
|
SaleType = @SaleType, ColorR = @ColorR, ColorG = @ColorG, ColorB = @ColorB, ColorA = @ColorA, ParticleSystem = @ParticleSystem,
|
||||||
ClickAction = @ClickAction, Material = @Material, CollisionSound = @CollisionSound, CollisionSoundVolume = @CollisionSoundVolume, PassTouches = @PassTouches,
|
ClickAction = @ClickAction, Material = @Material, CollisionSound = @CollisionSound, CollisionSoundVolume = @CollisionSoundVolume, PassTouches = @PassTouches,
|
||||||
LinkNumber = @LinkNumber, MediaURL = @MediaURL
|
LinkNumber = @LinkNumber, MediaURL = @MediaURL, DynAttrs = @DynAttrs,
|
||||||
|
PhysicsShapeType = @PhysicsShapeType, Density = @Density, GravityModifier = @GravityModifier, Friction = @Friction, Restitution = @Restitution
|
||||||
WHERE UUID = @UUID
|
WHERE UUID = @UUID
|
||||||
END
|
END
|
||||||
ELSE
|
ELSE
|
||||||
|
@ -366,7 +367,8 @@ ELSE
|
||||||
PayPrice, PayButton1, PayButton2, PayButton3, PayButton4, LoopedSound, LoopedSoundGain, TextureAnimation, OmegaX,
|
PayPrice, PayButton1, PayButton2, PayButton3, PayButton4, LoopedSound, LoopedSoundGain, TextureAnimation, OmegaX,
|
||||||
OmegaY, OmegaZ, CameraEyeOffsetX, CameraEyeOffsetY, CameraEyeOffsetZ, CameraAtOffsetX, CameraAtOffsetY, CameraAtOffsetZ,
|
OmegaY, OmegaZ, CameraEyeOffsetX, CameraEyeOffsetY, CameraEyeOffsetZ, CameraAtOffsetX, CameraAtOffsetY, CameraAtOffsetZ,
|
||||||
ForceMouselook, ScriptAccessPin, AllowedDrop, DieAtEdge, SalePrice, SaleType, ColorR, ColorG, ColorB, ColorA,
|
ForceMouselook, ScriptAccessPin, AllowedDrop, DieAtEdge, SalePrice, SaleType, ColorR, ColorG, ColorB, ColorA,
|
||||||
ParticleSystem, ClickAction, Material, CollisionSound, CollisionSoundVolume, PassTouches, LinkNumber, MediaURL
|
ParticleSystem, ClickAction, Material, CollisionSound, CollisionSoundVolume, PassTouches, LinkNumber, MediaURL, DynAttrs,
|
||||||
|
PhysicsShapeType, Density, GravityModifier, Friction, Restitution
|
||||||
) VALUES (
|
) VALUES (
|
||||||
@UUID, @CreationDate, @Name, @Text, @Description, @SitName, @TouchName, @ObjectFlags, @OwnerMask, @NextOwnerMask, @GroupMask,
|
@UUID, @CreationDate, @Name, @Text, @Description, @SitName, @TouchName, @ObjectFlags, @OwnerMask, @NextOwnerMask, @GroupMask,
|
||||||
@EveryoneMask, @BaseMask, @PositionX, @PositionY, @PositionZ, @GroupPositionX, @GroupPositionY, @GroupPositionZ, @VelocityX,
|
@EveryoneMask, @BaseMask, @PositionX, @PositionY, @PositionZ, @GroupPositionX, @GroupPositionY, @GroupPositionZ, @VelocityX,
|
||||||
|
@ -376,7 +378,8 @@ ELSE
|
||||||
@PayPrice, @PayButton1, @PayButton2, @PayButton3, @PayButton4, @LoopedSound, @LoopedSoundGain, @TextureAnimation, @OmegaX,
|
@PayPrice, @PayButton1, @PayButton2, @PayButton3, @PayButton4, @LoopedSound, @LoopedSoundGain, @TextureAnimation, @OmegaX,
|
||||||
@OmegaY, @OmegaZ, @CameraEyeOffsetX, @CameraEyeOffsetY, @CameraEyeOffsetZ, @CameraAtOffsetX, @CameraAtOffsetY, @CameraAtOffsetZ,
|
@OmegaY, @OmegaZ, @CameraEyeOffsetX, @CameraEyeOffsetY, @CameraEyeOffsetZ, @CameraAtOffsetX, @CameraAtOffsetY, @CameraAtOffsetZ,
|
||||||
@ForceMouselook, @ScriptAccessPin, @AllowedDrop, @DieAtEdge, @SalePrice, @SaleType, @ColorR, @ColorG, @ColorB, @ColorA,
|
@ForceMouselook, @ScriptAccessPin, @AllowedDrop, @DieAtEdge, @SalePrice, @SaleType, @ColorR, @ColorG, @ColorB, @ColorA,
|
||||||
@ParticleSystem, @ClickAction, @Material, @CollisionSound, @CollisionSoundVolume, @PassTouches, @LinkNumber, @MediaURL
|
@ParticleSystem, @ClickAction, @Material, @CollisionSound, @CollisionSoundVolume, @PassTouches, @LinkNumber, @MediaURL, @DynAttrs,
|
||||||
|
@PhysicsShapeType, @Density, @GravityModifier, @Friction, @Restitution
|
||||||
)
|
)
|
||||||
END";
|
END";
|
||||||
|
|
||||||
|
@ -1691,6 +1694,17 @@ VALUES
|
||||||
|
|
||||||
if (!(primRow["MediaURL"] is System.DBNull))
|
if (!(primRow["MediaURL"] is System.DBNull))
|
||||||
prim.MediaUrl = (string)primRow["MediaURL"];
|
prim.MediaUrl = (string)primRow["MediaURL"];
|
||||||
|
|
||||||
|
if (!(primRow["DynAttrs"] is System.DBNull))
|
||||||
|
prim.DynAttrs = DAMap.FromXml((string)primRow["DynAttrs"]);
|
||||||
|
else
|
||||||
|
prim.DynAttrs = new DAMap();
|
||||||
|
|
||||||
|
prim.PhysicsShapeType = Convert.ToByte(primRow["PhysicsShapeType"]);
|
||||||
|
prim.Density = Convert.ToSingle(primRow["Density"]);
|
||||||
|
prim.GravityModifier = Convert.ToSingle(primRow["GravityModifier"]);
|
||||||
|
prim.Friction = Convert.ToSingle(primRow["Friction"]);
|
||||||
|
prim.Restitution = Convert.ToSingle(primRow["Restitution"]);
|
||||||
|
|
||||||
return prim;
|
return prim;
|
||||||
}
|
}
|
||||||
|
@ -1749,7 +1763,6 @@ VALUES
|
||||||
baseShape.Media = PrimitiveBaseShape.MediaList.FromXml((string)shapeRow["Media"]);
|
baseShape.Media = PrimitiveBaseShape.MediaList.FromXml((string)shapeRow["Media"]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
return baseShape;
|
return baseShape;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2086,6 +2099,17 @@ VALUES
|
||||||
parameters.Add(_Database.CreateParameter("PassTouches", 0));
|
parameters.Add(_Database.CreateParameter("PassTouches", 0));
|
||||||
parameters.Add(_Database.CreateParameter("LinkNumber", prim.LinkNum));
|
parameters.Add(_Database.CreateParameter("LinkNumber", prim.LinkNum));
|
||||||
parameters.Add(_Database.CreateParameter("MediaURL", prim.MediaUrl));
|
parameters.Add(_Database.CreateParameter("MediaURL", prim.MediaUrl));
|
||||||
|
|
||||||
|
if (prim.DynAttrs.Count > 0)
|
||||||
|
parameters.Add(_Database.CreateParameter("DynAttrs", prim.DynAttrs.ToXml()));
|
||||||
|
else
|
||||||
|
parameters.Add(_Database.CreateParameter("DynAttrs", null));
|
||||||
|
|
||||||
|
parameters.Add(_Database.CreateParameter("PhysicsShapeType", prim.PhysicsShapeType));
|
||||||
|
parameters.Add(_Database.CreateParameter("Density", (double)prim.Density));
|
||||||
|
parameters.Add(_Database.CreateParameter("GravityModifier", (double)prim.GravityModifier));
|
||||||
|
parameters.Add(_Database.CreateParameter("Friction", (double)prim.Friction));
|
||||||
|
parameters.Add(_Database.CreateParameter("Restitution", (double)prim.Restitution));
|
||||||
|
|
||||||
return parameters.ToArray();
|
return parameters.ToArray();
|
||||||
}
|
}
|
||||||
|
@ -2143,7 +2167,6 @@ VALUES
|
||||||
parameters.Add(_Database.CreateParameter("Media", s.Media.ToXml()));
|
parameters.Add(_Database.CreateParameter("Media", s.Media.ToXml()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
return parameters.ToArray();
|
return parameters.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -61,5 +61,5 @@ using System.Runtime.InteropServices;
|
||||||
// You can specify all the values or you can default the Revision and Build Numbers
|
// You can specify all the values or you can default the Revision and Build Numbers
|
||||||
// by using the '*' as shown below:
|
// by using the '*' as shown below:
|
||||||
|
|
||||||
[assembly : AssemblyVersion("0.7.5.*")]
|
[assembly : AssemblyVersion("0.7.6.*")]
|
||||||
[assembly : AssemblyFileVersion("0.6.5.0")]
|
|
||||||
|
|
|
@ -1148,3 +1148,23 @@ CREATE TABLE [dbo].[regionenvironment](
|
||||||
) ON [PRIMARY]
|
) ON [PRIMARY]
|
||||||
|
|
||||||
COMMIT
|
COMMIT
|
||||||
|
|
||||||
|
:VERSION 38 #---------------- Dynamic attributes
|
||||||
|
|
||||||
|
BEGIN TRANSACTION
|
||||||
|
|
||||||
|
ALTER TABLE prims ADD COLUMN DynAttrs TEXT;
|
||||||
|
|
||||||
|
COMMIT
|
||||||
|
|
||||||
|
:VERSION 39 #---------------- Extra physics params
|
||||||
|
|
||||||
|
BEGIN TRANSACTION
|
||||||
|
|
||||||
|
ALTER TABLE prims ADD COLUMN `PhysicsShapeType` tinyint(4) NOT NULL default '0';
|
||||||
|
ALTER TABLE prims ADD COLUMN `Density` double NOT NULL default '1000';
|
||||||
|
ALTER TABLE prims ADD COLUMN `GravityModifier` double NOT NULL default '1';
|
||||||
|
ALTER TABLE prims ADD COLUMN `Friction` double NOT NULL default '0.6';
|
||||||
|
ALTER TABLE prims ADD COLUMN `Restitution` double NOT NULL default '0.5';
|
||||||
|
|
||||||
|
COMMIT
|
||||||
|
|
|
@ -173,14 +173,18 @@ namespace OpenSim.Data.MySQL
|
||||||
if (asset.Name.Length > 64)
|
if (asset.Name.Length > 64)
|
||||||
{
|
{
|
||||||
assetName = asset.Name.Substring(0, 64);
|
assetName = asset.Name.Substring(0, 64);
|
||||||
m_log.Warn("[ASSET DB]: Name field truncated from " + asset.Name.Length + " to " + assetName.Length + " characters on add");
|
m_log.WarnFormat(
|
||||||
|
"[ASSET DB]: Name '{0}' for asset {1} truncated from {2} to {3} characters on add",
|
||||||
|
asset.Name, asset.ID, asset.Name.Length, assetName.Length);
|
||||||
}
|
}
|
||||||
|
|
||||||
string assetDescription = asset.Description;
|
string assetDescription = asset.Description;
|
||||||
if (asset.Description.Length > 64)
|
if (asset.Description.Length > 64)
|
||||||
{
|
{
|
||||||
assetDescription = asset.Description.Substring(0, 64);
|
assetDescription = asset.Description.Substring(0, 64);
|
||||||
m_log.Warn("[ASSET DB]: Description field truncated from " + asset.Description.Length + " to " + assetDescription.Length + " characters on add");
|
m_log.WarnFormat(
|
||||||
|
"[ASSET DB]: Description '{0}' for asset {1} truncated from {2} to {3} characters on add",
|
||||||
|
asset.Description, asset.ID, asset.Description.Length, assetDescription.Length);
|
||||||
}
|
}
|
||||||
|
|
||||||
try
|
try
|
||||||
|
|
|
@ -306,5 +306,65 @@ namespace OpenSim.Data.MySQL
|
||||||
return ExecuteNonQuery(cmd) > 0;
|
return ExecuteNonQuery(cmd) > 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public long GetCount(string field, string key)
|
||||||
|
{
|
||||||
|
return GetCount(new string[] { field }, new string[] { key });
|
||||||
|
}
|
||||||
|
|
||||||
|
public long GetCount(string[] fields, string[] keys)
|
||||||
|
{
|
||||||
|
if (fields.Length != keys.Length)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
List<string> terms = new List<string>();
|
||||||
|
|
||||||
|
using (MySqlCommand cmd = new MySqlCommand())
|
||||||
|
{
|
||||||
|
for (int i = 0; i < fields.Length; i++)
|
||||||
|
{
|
||||||
|
cmd.Parameters.AddWithValue(fields[i], keys[i]);
|
||||||
|
terms.Add("`" + fields[i] + "` = ?" + fields[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
string where = String.Join(" and ", terms.ToArray());
|
||||||
|
|
||||||
|
string query = String.Format("select count(*) from {0} where {1}",
|
||||||
|
m_Realm, where);
|
||||||
|
|
||||||
|
cmd.CommandText = query;
|
||||||
|
|
||||||
|
Object result = DoQueryScalar(cmd);
|
||||||
|
|
||||||
|
return Convert.ToInt64(result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public long GetCount(string where)
|
||||||
|
{
|
||||||
|
using (MySqlCommand cmd = new MySqlCommand())
|
||||||
|
{
|
||||||
|
string query = String.Format("select count(*) from {0} where {1}",
|
||||||
|
m_Realm, where);
|
||||||
|
|
||||||
|
cmd.CommandText = query;
|
||||||
|
|
||||||
|
object result = DoQueryScalar(cmd);
|
||||||
|
|
||||||
|
return Convert.ToInt64(result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public object DoQueryScalar(MySqlCommand cmd)
|
||||||
|
{
|
||||||
|
using (MySqlConnection dbcon = new MySqlConnection(m_connectionString))
|
||||||
|
{
|
||||||
|
dbcon.Open();
|
||||||
|
cmd.Connection = dbcon;
|
||||||
|
|
||||||
|
return cmd.ExecuteScalar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,484 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Reflection;
|
||||||
|
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Data.MySQL;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using MySql.Data.MySqlClient;
|
||||||
|
|
||||||
|
namespace OpenSim.Data.MySQL
|
||||||
|
{
|
||||||
|
public class MySQLGroupsData : IGroupsData
|
||||||
|
{
|
||||||
|
private MySqlGroupsGroupsHandler m_Groups;
|
||||||
|
private MySqlGroupsMembershipHandler m_Membership;
|
||||||
|
private MySqlGroupsRolesHandler m_Roles;
|
||||||
|
private MySqlGroupsRoleMembershipHandler m_RoleMembership;
|
||||||
|
private MySqlGroupsInvitesHandler m_Invites;
|
||||||
|
private MySqlGroupsNoticesHandler m_Notices;
|
||||||
|
private MySqlGroupsPrincipalsHandler m_Principals;
|
||||||
|
|
||||||
|
public MySQLGroupsData(string connectionString, string realm)
|
||||||
|
{
|
||||||
|
m_Groups = new MySqlGroupsGroupsHandler(connectionString, realm + "_groups", realm + "_Store");
|
||||||
|
m_Membership = new MySqlGroupsMembershipHandler(connectionString, realm + "_membership");
|
||||||
|
m_Roles = new MySqlGroupsRolesHandler(connectionString, realm + "_roles");
|
||||||
|
m_RoleMembership = new MySqlGroupsRoleMembershipHandler(connectionString, realm + "_rolemembership");
|
||||||
|
m_Invites = new MySqlGroupsInvitesHandler(connectionString, realm + "_invites");
|
||||||
|
m_Notices = new MySqlGroupsNoticesHandler(connectionString, realm + "_notices");
|
||||||
|
m_Principals = new MySqlGroupsPrincipalsHandler(connectionString, realm + "_principals");
|
||||||
|
}
|
||||||
|
|
||||||
|
#region groups table
|
||||||
|
public bool StoreGroup(GroupData data)
|
||||||
|
{
|
||||||
|
return m_Groups.Store(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupData RetrieveGroup(UUID groupID)
|
||||||
|
{
|
||||||
|
GroupData[] groups = m_Groups.Get("GroupID", groupID.ToString());
|
||||||
|
if (groups.Length > 0)
|
||||||
|
return groups[0];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupData RetrieveGroup(string name)
|
||||||
|
{
|
||||||
|
GroupData[] groups = m_Groups.Get("Name", name);
|
||||||
|
if (groups.Length > 0)
|
||||||
|
return groups[0];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GroupData[] RetrieveGroups(string pattern)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(pattern))
|
||||||
|
pattern = "1 ORDER BY Name LIMIT 100";
|
||||||
|
else
|
||||||
|
pattern = string.Format("Name LIKE %{0}% ORDER BY Name LIMIT 100", pattern);
|
||||||
|
|
||||||
|
return m_Groups.Get(pattern);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool DeleteGroup(UUID groupID)
|
||||||
|
{
|
||||||
|
return m_Groups.Delete("GroupID", groupID.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public int GroupsCount()
|
||||||
|
{
|
||||||
|
return (int)m_Groups.GetCount("Location=\"\"");
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region membership table
|
||||||
|
public MembershipData[] RetrieveMembers(UUID groupID)
|
||||||
|
{
|
||||||
|
return m_Membership.Get("GroupID", groupID.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public MembershipData RetrieveMember(UUID groupID, string pricipalID)
|
||||||
|
{
|
||||||
|
MembershipData[] m = m_Membership.Get(new string[] { "GroupID", "PrincipalID" },
|
||||||
|
new string[] { groupID.ToString(), pricipalID });
|
||||||
|
if (m != null && m.Length > 0)
|
||||||
|
return m[0];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public MembershipData[] RetrieveMemberships(string pricipalID)
|
||||||
|
{
|
||||||
|
return m_Membership.Get("PrincipalID", pricipalID.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool StoreMember(MembershipData data)
|
||||||
|
{
|
||||||
|
return m_Membership.Store(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool DeleteMember(UUID groupID, string pricipalID)
|
||||||
|
{
|
||||||
|
return m_Membership.Delete(new string[] { "GroupID", "PrincipalID" },
|
||||||
|
new string[] { groupID.ToString(), pricipalID });
|
||||||
|
}
|
||||||
|
|
||||||
|
public int MemberCount(UUID groupID)
|
||||||
|
{
|
||||||
|
return (int)m_Membership.GetCount("GroupID", groupID.ToString());
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region roles table
|
||||||
|
public bool StoreRole(RoleData data)
|
||||||
|
{
|
||||||
|
return m_Roles.Store(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public RoleData RetrieveRole(UUID groupID, UUID roleID)
|
||||||
|
{
|
||||||
|
RoleData[] data = m_Roles.Get(new string[] { "GroupID", "RoleID" },
|
||||||
|
new string[] { groupID.ToString(), roleID.ToString() });
|
||||||
|
|
||||||
|
if (data != null && data.Length > 0)
|
||||||
|
return data[0];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RoleData[] RetrieveRoles(UUID groupID)
|
||||||
|
{
|
||||||
|
//return m_Roles.RetrieveRoles(groupID);
|
||||||
|
return m_Roles.Get("GroupID", groupID.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool DeleteRole(UUID groupID, UUID roleID)
|
||||||
|
{
|
||||||
|
return m_Roles.Delete(new string[] { "GroupID", "RoleID" },
|
||||||
|
new string[] { groupID.ToString(), roleID.ToString() });
|
||||||
|
}
|
||||||
|
|
||||||
|
public int RoleCount(UUID groupID)
|
||||||
|
{
|
||||||
|
return (int)m_Roles.GetCount("GroupID", groupID.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region rolememberhip table
|
||||||
|
public RoleMembershipData[] RetrieveRolesMembers(UUID groupID)
|
||||||
|
{
|
||||||
|
RoleMembershipData[] data = m_RoleMembership.Get("GroupID", groupID.ToString());
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RoleMembershipData[] RetrieveRoleMembers(UUID groupID, UUID roleID)
|
||||||
|
{
|
||||||
|
RoleMembershipData[] data = m_RoleMembership.Get(new string[] { "GroupID", "RoleID" },
|
||||||
|
new string[] { groupID.ToString(), roleID.ToString() });
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RoleMembershipData[] RetrieveMemberRoles(UUID groupID, string principalID)
|
||||||
|
{
|
||||||
|
RoleMembershipData[] data = m_RoleMembership.Get(new string[] { "GroupID", "PrincipalID" },
|
||||||
|
new string[] { groupID.ToString(), principalID.ToString() });
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RoleMembershipData RetrieveRoleMember(UUID groupID, UUID roleID, string principalID)
|
||||||
|
{
|
||||||
|
RoleMembershipData[] data = m_RoleMembership.Get(new string[] { "GroupID", "RoleID", "PrincipalID" },
|
||||||
|
new string[] { groupID.ToString(), roleID.ToString(), principalID.ToString() });
|
||||||
|
|
||||||
|
if (data != null && data.Length > 0)
|
||||||
|
return data[0];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int RoleMemberCount(UUID groupID, UUID roleID)
|
||||||
|
{
|
||||||
|
return (int)m_RoleMembership.GetCount(new string[] { "GroupID", "RoleID" },
|
||||||
|
new string[] { groupID.ToString(), roleID.ToString() });
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool StoreRoleMember(RoleMembershipData data)
|
||||||
|
{
|
||||||
|
return m_RoleMembership.Store(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool DeleteRoleMember(RoleMembershipData data)
|
||||||
|
{
|
||||||
|
return m_RoleMembership.Delete(new string[] { "GroupID", "RoleID", "PrincipalID"},
|
||||||
|
new string[] { data.GroupID.ToString(), data.RoleID.ToString(), data.PrincipalID });
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool DeleteMemberAllRoles(UUID groupID, string principalID)
|
||||||
|
{
|
||||||
|
return m_RoleMembership.Delete(new string[] { "GroupID", "PrincipalID" },
|
||||||
|
new string[] { groupID.ToString(), principalID });
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region principals table
|
||||||
|
public bool StorePrincipal(PrincipalData data)
|
||||||
|
{
|
||||||
|
return m_Principals.Store(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public PrincipalData RetrievePrincipal(string principalID)
|
||||||
|
{
|
||||||
|
PrincipalData[] p = m_Principals.Get("PrincipalID", principalID);
|
||||||
|
if (p != null && p.Length > 0)
|
||||||
|
return p[0];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool DeletePrincipal(string principalID)
|
||||||
|
{
|
||||||
|
return m_Principals.Delete("PrincipalID", principalID);
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region invites table
|
||||||
|
|
||||||
|
public bool StoreInvitation(InvitationData data)
|
||||||
|
{
|
||||||
|
return m_Invites.Store(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public InvitationData RetrieveInvitation(UUID inviteID)
|
||||||
|
{
|
||||||
|
InvitationData[] invites = m_Invites.Get("InviteID", inviteID.ToString());
|
||||||
|
|
||||||
|
if (invites != null && invites.Length > 0)
|
||||||
|
return invites[0];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public InvitationData RetrieveInvitation(UUID groupID, string principalID)
|
||||||
|
{
|
||||||
|
InvitationData[] invites = m_Invites.Get(new string[] { "GroupID", "PrincipalID" },
|
||||||
|
new string[] { groupID.ToString(), principalID });
|
||||||
|
|
||||||
|
if (invites != null && invites.Length > 0)
|
||||||
|
return invites[0];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool DeleteInvite(UUID inviteID)
|
||||||
|
{
|
||||||
|
return m_Invites.Delete("InviteID", inviteID.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void DeleteOldInvites()
|
||||||
|
{
|
||||||
|
m_Invites.DeleteOld();
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region notices table
|
||||||
|
|
||||||
|
public bool StoreNotice(NoticeData data)
|
||||||
|
{
|
||||||
|
return m_Notices.Store(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public NoticeData RetrieveNotice(UUID noticeID)
|
||||||
|
{
|
||||||
|
NoticeData[] notices = m_Notices.Get("NoticeID", noticeID.ToString());
|
||||||
|
|
||||||
|
if (notices != null && notices.Length > 0)
|
||||||
|
return notices[0];
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public NoticeData[] RetrieveNotices(UUID groupID)
|
||||||
|
{
|
||||||
|
NoticeData[] notices = m_Notices.Get("GroupID", groupID.ToString());
|
||||||
|
|
||||||
|
return notices;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool DeleteNotice(UUID noticeID)
|
||||||
|
{
|
||||||
|
return m_Notices.Delete("NoticeID", noticeID.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void DeleteOldNotices()
|
||||||
|
{
|
||||||
|
m_Notices.DeleteOld();
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region combinations
|
||||||
|
public MembershipData RetrievePrincipalGroupMembership(string principalID, UUID groupID)
|
||||||
|
{
|
||||||
|
// TODO
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
public MembershipData[] RetrievePrincipalGroupMemberships(string principalID)
|
||||||
|
{
|
||||||
|
// TODO
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
|
||||||
|
public class MySqlGroupsGroupsHandler : MySQLGenericTableHandler<GroupData>
|
||||||
|
{
|
||||||
|
protected override Assembly Assembly
|
||||||
|
{
|
||||||
|
// WARNING! Moving migrations to this assembly!!!
|
||||||
|
get { return GetType().Assembly; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public MySqlGroupsGroupsHandler(string connectionString, string realm, string store)
|
||||||
|
: base(connectionString, realm, store)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class MySqlGroupsMembershipHandler : MySQLGenericTableHandler<MembershipData>
|
||||||
|
{
|
||||||
|
protected override Assembly Assembly
|
||||||
|
{
|
||||||
|
// WARNING! Moving migrations to this assembly!!!
|
||||||
|
get { return GetType().Assembly; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public MySqlGroupsMembershipHandler(string connectionString, string realm)
|
||||||
|
: base(connectionString, realm, string.Empty)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class MySqlGroupsRolesHandler : MySQLGenericTableHandler<RoleData>
|
||||||
|
{
|
||||||
|
protected override Assembly Assembly
|
||||||
|
{
|
||||||
|
// WARNING! Moving migrations to this assembly!!!
|
||||||
|
get { return GetType().Assembly; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public MySqlGroupsRolesHandler(string connectionString, string realm)
|
||||||
|
: base(connectionString, realm, string.Empty)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class MySqlGroupsRoleMembershipHandler : MySQLGenericTableHandler<RoleMembershipData>
|
||||||
|
{
|
||||||
|
protected override Assembly Assembly
|
||||||
|
{
|
||||||
|
// WARNING! Moving migrations to this assembly!!!
|
||||||
|
get { return GetType().Assembly; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public MySqlGroupsRoleMembershipHandler(string connectionString, string realm)
|
||||||
|
: base(connectionString, realm, string.Empty)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class MySqlGroupsInvitesHandler : MySQLGenericTableHandler<InvitationData>
|
||||||
|
{
|
||||||
|
protected override Assembly Assembly
|
||||||
|
{
|
||||||
|
// WARNING! Moving migrations to this assembly!!!
|
||||||
|
get { return GetType().Assembly; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public MySqlGroupsInvitesHandler(string connectionString, string realm)
|
||||||
|
: base(connectionString, realm, string.Empty)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void DeleteOld()
|
||||||
|
{
|
||||||
|
uint now = (uint)Util.UnixTimeSinceEpoch();
|
||||||
|
|
||||||
|
using (MySqlCommand cmd = new MySqlCommand())
|
||||||
|
{
|
||||||
|
cmd.CommandText = String.Format("delete from {0} where TMStamp < ?tstamp", m_Realm);
|
||||||
|
cmd.Parameters.AddWithValue("?tstamp", now - 14 * 24 * 60 * 60); // > 2 weeks old
|
||||||
|
|
||||||
|
ExecuteNonQuery(cmd);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class MySqlGroupsNoticesHandler : MySQLGenericTableHandler<NoticeData>
|
||||||
|
{
|
||||||
|
protected override Assembly Assembly
|
||||||
|
{
|
||||||
|
// WARNING! Moving migrations to this assembly!!!
|
||||||
|
get { return GetType().Assembly; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public MySqlGroupsNoticesHandler(string connectionString, string realm)
|
||||||
|
: base(connectionString, realm, string.Empty)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void DeleteOld()
|
||||||
|
{
|
||||||
|
uint now = (uint)Util.UnixTimeSinceEpoch();
|
||||||
|
|
||||||
|
using (MySqlCommand cmd = new MySqlCommand())
|
||||||
|
{
|
||||||
|
cmd.CommandText = String.Format("delete from {0} where TMStamp < ?tstamp", m_Realm);
|
||||||
|
cmd.Parameters.AddWithValue("?tstamp", now - 14 * 24 * 60 * 60); // > 2 weeks old
|
||||||
|
|
||||||
|
ExecuteNonQuery(cmd);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class MySqlGroupsPrincipalsHandler : MySQLGenericTableHandler<PrincipalData>
|
||||||
|
{
|
||||||
|
protected override Assembly Assembly
|
||||||
|
{
|
||||||
|
// WARNING! Moving migrations to this assembly!!!
|
||||||
|
get { return GetType().Assembly; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public MySqlGroupsPrincipalsHandler(string connectionString, string realm)
|
||||||
|
: base(connectionString, realm, string.Empty)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -25,48 +25,38 @@
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
using System.IO;
|
using System;
|
||||||
using System.Text;
|
using System.Collections;
|
||||||
using System.Xml;
|
using System.Collections.Generic;
|
||||||
|
using System.Reflection;
|
||||||
|
|
||||||
namespace OpenSim.ApplicationPlugins.Rest
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Data.MySQL;
|
||||||
|
|
||||||
|
using OpenMetaverse;
|
||||||
|
using MySql.Data.MySqlClient;
|
||||||
|
|
||||||
|
namespace OpenSim.Data.MySQL
|
||||||
{
|
{
|
||||||
public class RestXmlWriter: XmlTextWriter
|
public class MySQLOfflineIMData : MySQLGenericTableHandler<OfflineIMData>, IOfflineIMData
|
||||||
{
|
{
|
||||||
private StringWriter m_sw = null;
|
public MySQLOfflineIMData(string connectionString, string realm)
|
||||||
|
: base(connectionString, realm, "IM_Store")
|
||||||
public RestXmlWriter(StringWriter sw) : base(sw)
|
|
||||||
{
|
|
||||||
m_sw = sw;
|
|
||||||
Formatting = Formatting.Indented;
|
|
||||||
}
|
|
||||||
|
|
||||||
public RestXmlWriter(TextWriter textWriter) : base(textWriter)
|
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
public RestXmlWriter(Stream stream)
|
public void DeleteOld()
|
||||||
: this(stream, Encoding.UTF8)
|
|
||||||
{
|
{
|
||||||
}
|
uint now = (uint)Util.UnixTimeSinceEpoch();
|
||||||
|
|
||||||
public RestXmlWriter(Stream stream, Encoding enc) : base(stream, enc)
|
using (MySqlCommand cmd = new MySqlCommand())
|
||||||
{
|
{
|
||||||
}
|
cmd.CommandText = String.Format("delete from {0} where TMStamp < ?tstamp", m_Realm);
|
||||||
|
cmd.Parameters.AddWithValue("?tstamp", now - 14 * 24 * 60 * 60); // > 2 weeks old
|
||||||
|
|
||||||
public override void WriteStartDocument()
|
ExecuteNonQuery(cmd);
|
||||||
{
|
}
|
||||||
}
|
|
||||||
|
|
||||||
public override void WriteStartDocument(bool standalone)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string ToString()
|
|
||||||
{
|
|
||||||
Flush();
|
|
||||||
Close();
|
|
||||||
return m_sw.ToString();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -95,5 +95,19 @@ namespace OpenSim.Data.MySQL
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public bool VerifyAgent(UUID agentId, UUID secureSessionID)
|
||||||
|
{
|
||||||
|
PresenceData[] ret = Get("SecureSessionID",
|
||||||
|
secureSessionID.ToString());
|
||||||
|
|
||||||
|
if (ret.Length == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if(ret[0].UserID != agentId.ToString())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -176,7 +176,7 @@ namespace OpenSim.Data.MySQL
|
||||||
"PassCollisions, " +
|
"PassCollisions, " +
|
||||||
"LinkNumber, MediaURL, KeyframeMotion, " +
|
"LinkNumber, MediaURL, KeyframeMotion, " +
|
||||||
"PhysicsShapeType, Density, GravityModifier, " +
|
"PhysicsShapeType, Density, GravityModifier, " +
|
||||||
"Friction, Restitution, Vehicle " +
|
"Friction, Restitution, Vehicle, DynAttrs " +
|
||||||
") values (" + "?UUID, " +
|
") values (" + "?UUID, " +
|
||||||
"?CreationDate, ?Name, ?Text, " +
|
"?CreationDate, ?Name, ?Text, " +
|
||||||
"?Description, ?SitName, ?TouchName, " +
|
"?Description, ?SitName, ?TouchName, " +
|
||||||
|
@ -211,7 +211,7 @@ namespace OpenSim.Data.MySQL
|
||||||
"?CollisionSoundVolume, ?PassTouches, ?PassCollisions, " +
|
"?CollisionSoundVolume, ?PassTouches, ?PassCollisions, " +
|
||||||
"?LinkNumber, ?MediaURL, ?KeyframeMotion, " +
|
"?LinkNumber, ?MediaURL, ?KeyframeMotion, " +
|
||||||
"?PhysicsShapeType, ?Density, ?GravityModifier, " +
|
"?PhysicsShapeType, ?Density, ?GravityModifier, " +
|
||||||
"?Friction, ?Restitution, ?Vehicle)";
|
"?Friction, ?Restitution, ?Vehicle, ?DynAttrs)";
|
||||||
|
|
||||||
FillPrimCommand(cmd, prim, obj.UUID, regionUUID);
|
FillPrimCommand(cmd, prim, obj.UUID, regionUUID);
|
||||||
|
|
||||||
|
@ -228,7 +228,8 @@ namespace OpenSim.Data.MySQL
|
||||||
"PathTaperX, PathTaperY, PathTwist, " +
|
"PathTaperX, PathTaperY, PathTwist, " +
|
||||||
"PathTwistBegin, ProfileBegin, ProfileEnd, " +
|
"PathTwistBegin, ProfileBegin, ProfileEnd, " +
|
||||||
"ProfileCurve, ProfileHollow, Texture, " +
|
"ProfileCurve, ProfileHollow, Texture, " +
|
||||||
"ExtraParams, State, Media) values (?UUID, " +
|
"ExtraParams, State, Media) " +
|
||||||
|
"values (?UUID, " +
|
||||||
"?Shape, ?ScaleX, ?ScaleY, ?ScaleZ, " +
|
"?Shape, ?ScaleX, ?ScaleY, ?ScaleZ, " +
|
||||||
"?PCode, ?PathBegin, ?PathEnd, " +
|
"?PCode, ?PathBegin, ?PathEnd, " +
|
||||||
"?PathScaleX, ?PathScaleY, " +
|
"?PathScaleX, ?PathScaleY, " +
|
||||||
|
@ -1321,6 +1322,11 @@ namespace OpenSim.Data.MySQL
|
||||||
|
|
||||||
if (!(row["MediaURL"] is System.DBNull))
|
if (!(row["MediaURL"] is System.DBNull))
|
||||||
prim.MediaUrl = (string)row["MediaURL"];
|
prim.MediaUrl = (string)row["MediaURL"];
|
||||||
|
|
||||||
|
if (!(row["DynAttrs"] is System.DBNull))
|
||||||
|
prim.DynAttrs = DAMap.FromXml((string)row["DynAttrs"]);
|
||||||
|
else
|
||||||
|
prim.DynAttrs = new DAMap();
|
||||||
|
|
||||||
if (!(row["KeyframeMotion"] is DBNull))
|
if (!(row["KeyframeMotion"] is DBNull))
|
||||||
{
|
{
|
||||||
|
@ -1339,7 +1345,7 @@ namespace OpenSim.Data.MySQL
|
||||||
prim.Density = (float)(double)row["Density"];
|
prim.Density = (float)(double)row["Density"];
|
||||||
prim.GravityModifier = (float)(double)row["GravityModifier"];
|
prim.GravityModifier = (float)(double)row["GravityModifier"];
|
||||||
prim.Friction = (float)(double)row["Friction"];
|
prim.Friction = (float)(double)row["Friction"];
|
||||||
prim.Bounciness = (float)(double)row["Restitution"];
|
prim.Restitution = (float)(double)row["Restitution"];
|
||||||
|
|
||||||
SOPVehicle vehicle = null;
|
SOPVehicle vehicle = null;
|
||||||
|
|
||||||
|
@ -1721,16 +1727,21 @@ namespace OpenSim.Data.MySQL
|
||||||
else
|
else
|
||||||
cmd.Parameters.AddWithValue("KeyframeMotion", new Byte[0]);
|
cmd.Parameters.AddWithValue("KeyframeMotion", new Byte[0]);
|
||||||
|
|
||||||
cmd.Parameters.AddWithValue("PhysicsShapeType", prim.PhysicsShapeType);
|
|
||||||
cmd.Parameters.AddWithValue("Density", (double)prim.Density);
|
|
||||||
cmd.Parameters.AddWithValue("GravityModifier", (double)prim.GravityModifier);
|
|
||||||
cmd.Parameters.AddWithValue("Friction", (double)prim.Friction);
|
|
||||||
cmd.Parameters.AddWithValue("Restitution", (double)prim.Bounciness);
|
|
||||||
|
|
||||||
if (prim.VehicleParams != null)
|
if (prim.VehicleParams != null)
|
||||||
cmd.Parameters.AddWithValue("Vehicle", prim.VehicleParams.ToXml2());
|
cmd.Parameters.AddWithValue("Vehicle", prim.VehicleParams.ToXml2());
|
||||||
else
|
else
|
||||||
cmd.Parameters.AddWithValue("Vehicle", String.Empty);
|
cmd.Parameters.AddWithValue("Vehicle", String.Empty);
|
||||||
|
|
||||||
|
if (prim.DynAttrs.Count > 0)
|
||||||
|
cmd.Parameters.AddWithValue("DynAttrs", prim.DynAttrs.ToXml());
|
||||||
|
else
|
||||||
|
cmd.Parameters.AddWithValue("DynAttrs", null);
|
||||||
|
|
||||||
|
cmd.Parameters.AddWithValue("PhysicsShapeType", prim.PhysicsShapeType);
|
||||||
|
cmd.Parameters.AddWithValue("Density", (double)prim.Density);
|
||||||
|
cmd.Parameters.AddWithValue("GravityModifier", (double)prim.GravityModifier);
|
||||||
|
cmd.Parameters.AddWithValue("Friction", (double)prim.Friction);
|
||||||
|
cmd.Parameters.AddWithValue("Restitution", (double)prim.Restitution);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|
|
@ -50,6 +50,11 @@ namespace OpenSim.Data.MySQL
|
||||||
get { return GetType().Assembly; }
|
get { return GetType().Assembly; }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Number of days that must pass before we update the access time on an asset when it has been fetched.
|
||||||
|
/// </summary>
|
||||||
|
private const int DaysBetweenAccessTimeUpdates = 30;
|
||||||
|
|
||||||
private bool m_enableCompression = false;
|
private bool m_enableCompression = false;
|
||||||
private string m_connectionString;
|
private string m_connectionString;
|
||||||
private object m_dbLock = new object();
|
private object m_dbLock = new object();
|
||||||
|
@ -133,10 +138,10 @@ namespace OpenSim.Data.MySQL
|
||||||
dbcon.Open();
|
dbcon.Open();
|
||||||
|
|
||||||
using (MySqlCommand cmd = new MySqlCommand(
|
using (MySqlCommand cmd = new MySqlCommand(
|
||||||
"SELECT name, description, asset_type, local, temporary, asset_flags, creator_id, data FROM xassetsmeta JOIN xassetsdata ON xassetsmeta.hash = xassetsdata.hash WHERE id=?id",
|
"SELECT Name, Description, AccessTime, AssetType, Local, Temporary, AssetFlags, CreatorID, Data FROM XAssetsMeta JOIN XAssetsData ON XAssetsMeta.Hash = XAssetsData.Hash WHERE ID=?ID",
|
||||||
dbcon))
|
dbcon))
|
||||||
{
|
{
|
||||||
cmd.Parameters.AddWithValue("?id", assetID.ToString());
|
cmd.Parameters.AddWithValue("?ID", assetID.ToString());
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
@ -144,18 +149,18 @@ namespace OpenSim.Data.MySQL
|
||||||
{
|
{
|
||||||
if (dbReader.Read())
|
if (dbReader.Read())
|
||||||
{
|
{
|
||||||
asset = new AssetBase(assetID, (string)dbReader["name"], (sbyte)dbReader["asset_type"], dbReader["creator_id"].ToString());
|
asset = new AssetBase(assetID, (string)dbReader["Name"], (sbyte)dbReader["AssetType"], dbReader["CreatorID"].ToString());
|
||||||
asset.Data = (byte[])dbReader["data"];
|
asset.Data = (byte[])dbReader["Data"];
|
||||||
asset.Description = (string)dbReader["description"];
|
asset.Description = (string)dbReader["Description"];
|
||||||
|
|
||||||
string local = dbReader["local"].ToString();
|
string local = dbReader["Local"].ToString();
|
||||||
if (local.Equals("1") || local.Equals("true", StringComparison.InvariantCultureIgnoreCase))
|
if (local.Equals("1") || local.Equals("true", StringComparison.InvariantCultureIgnoreCase))
|
||||||
asset.Local = true;
|
asset.Local = true;
|
||||||
else
|
else
|
||||||
asset.Local = false;
|
asset.Local = false;
|
||||||
|
|
||||||
asset.Temporary = Convert.ToBoolean(dbReader["temporary"]);
|
asset.Temporary = Convert.ToBoolean(dbReader["Temporary"]);
|
||||||
asset.Flags = (AssetFlags)Convert.ToInt32(dbReader["asset_flags"]);
|
asset.Flags = (AssetFlags)Convert.ToInt32(dbReader["AssetFlags"]);
|
||||||
|
|
||||||
if (m_enableCompression)
|
if (m_enableCompression)
|
||||||
{
|
{
|
||||||
|
@ -171,12 +176,14 @@ namespace OpenSim.Data.MySQL
|
||||||
// asset.ID, asset.Name, asset.Data.Length, compressedLength);
|
// asset.ID, asset.Name, asset.Data.Length, compressedLength);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
UpdateAccessTime(asset.Metadata, (int)dbReader["AccessTime"]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
m_log.Error("[MYSQL XASSET DATA]: MySql failure fetching asset " + assetID + ": " + e.Message);
|
m_log.Error(string.Format("[MYSQL XASSET DATA]: Failure fetching asset {0}", assetID), e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -204,14 +211,18 @@ namespace OpenSim.Data.MySQL
|
||||||
if (asset.Name.Length > 64)
|
if (asset.Name.Length > 64)
|
||||||
{
|
{
|
||||||
assetName = asset.Name.Substring(0, 64);
|
assetName = asset.Name.Substring(0, 64);
|
||||||
m_log.Warn("[XASSET DB]: Name field truncated from " + asset.Name.Length + " to " + assetName.Length + " characters on add");
|
m_log.WarnFormat(
|
||||||
|
"[XASSET DB]: Name '{0}' for asset {1} truncated from {2} to {3} characters on add",
|
||||||
|
asset.Name, asset.ID, asset.Name.Length, assetName.Length);
|
||||||
}
|
}
|
||||||
|
|
||||||
string assetDescription = asset.Description;
|
string assetDescription = asset.Description;
|
||||||
if (asset.Description.Length > 64)
|
if (asset.Description.Length > 64)
|
||||||
{
|
{
|
||||||
assetDescription = asset.Description.Substring(0, 64);
|
assetDescription = asset.Description.Substring(0, 64);
|
||||||
m_log.Warn("[XASSET DB]: Description field truncated from " + asset.Description.Length + " to " + assetDescription.Length + " characters on add");
|
m_log.WarnFormat(
|
||||||
|
"[XASSET DB]: Description '{0}' for asset {1} truncated from {2} to {3} characters on add",
|
||||||
|
asset.Description, asset.ID, asset.Description.Length, assetDescription.Length);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (m_enableCompression)
|
if (m_enableCompression)
|
||||||
|
@ -238,23 +249,23 @@ namespace OpenSim.Data.MySQL
|
||||||
{
|
{
|
||||||
using (MySqlCommand cmd =
|
using (MySqlCommand cmd =
|
||||||
new MySqlCommand(
|
new MySqlCommand(
|
||||||
"replace INTO xassetsmeta(id, hash, name, description, asset_type, local, temporary, create_time, access_time, asset_flags, creator_id)" +
|
"replace INTO XAssetsMeta(ID, Hash, Name, Description, AssetType, Local, Temporary, CreateTime, AccessTime, AssetFlags, CreatorID)" +
|
||||||
"VALUES(?id, ?hash, ?name, ?description, ?asset_type, ?local, ?temporary, ?create_time, ?access_time, ?asset_flags, ?creator_id)",
|
"VALUES(?ID, ?Hash, ?Name, ?Description, ?AssetType, ?Local, ?Temporary, ?CreateTime, ?AccessTime, ?AssetFlags, ?CreatorID)",
|
||||||
dbcon))
|
dbcon))
|
||||||
{
|
{
|
||||||
// create unix epoch time
|
// create unix epoch time
|
||||||
int now = (int)Utils.DateTimeToUnixTime(DateTime.UtcNow);
|
int now = (int)Utils.DateTimeToUnixTime(DateTime.UtcNow);
|
||||||
cmd.Parameters.AddWithValue("?id", asset.ID);
|
cmd.Parameters.AddWithValue("?ID", asset.ID);
|
||||||
cmd.Parameters.AddWithValue("?hash", hash);
|
cmd.Parameters.AddWithValue("?Hash", hash);
|
||||||
cmd.Parameters.AddWithValue("?name", assetName);
|
cmd.Parameters.AddWithValue("?Name", assetName);
|
||||||
cmd.Parameters.AddWithValue("?description", assetDescription);
|
cmd.Parameters.AddWithValue("?Description", assetDescription);
|
||||||
cmd.Parameters.AddWithValue("?asset_type", asset.Type);
|
cmd.Parameters.AddWithValue("?AssetType", asset.Type);
|
||||||
cmd.Parameters.AddWithValue("?local", asset.Local);
|
cmd.Parameters.AddWithValue("?Local", asset.Local);
|
||||||
cmd.Parameters.AddWithValue("?temporary", asset.Temporary);
|
cmd.Parameters.AddWithValue("?Temporary", asset.Temporary);
|
||||||
cmd.Parameters.AddWithValue("?create_time", now);
|
cmd.Parameters.AddWithValue("?CreateTime", now);
|
||||||
cmd.Parameters.AddWithValue("?access_time", now);
|
cmd.Parameters.AddWithValue("?AccessTime", now);
|
||||||
cmd.Parameters.AddWithValue("?creator_id", asset.Metadata.CreatorID);
|
cmd.Parameters.AddWithValue("?CreatorID", asset.Metadata.CreatorID);
|
||||||
cmd.Parameters.AddWithValue("?asset_flags", (int)asset.Flags);
|
cmd.Parameters.AddWithValue("?AssetFlags", (int)asset.Flags);
|
||||||
cmd.ExecuteNonQuery();
|
cmd.ExecuteNonQuery();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -274,11 +285,11 @@ namespace OpenSim.Data.MySQL
|
||||||
{
|
{
|
||||||
using (MySqlCommand cmd =
|
using (MySqlCommand cmd =
|
||||||
new MySqlCommand(
|
new MySqlCommand(
|
||||||
"INSERT INTO xassetsdata(hash, data) VALUES(?hash, ?data)",
|
"INSERT INTO XAssetsData(Hash, Data) VALUES(?Hash, ?Data)",
|
||||||
dbcon))
|
dbcon))
|
||||||
{
|
{
|
||||||
cmd.Parameters.AddWithValue("?hash", hash);
|
cmd.Parameters.AddWithValue("?Hash", hash);
|
||||||
cmd.Parameters.AddWithValue("?data", asset.Data);
|
cmd.Parameters.AddWithValue("?Data", asset.Data);
|
||||||
cmd.ExecuteNonQuery();
|
cmd.ExecuteNonQuery();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -299,41 +310,49 @@ namespace OpenSim.Data.MySQL
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// private void UpdateAccessTime(AssetBase asset)
|
/// <summary>
|
||||||
// {
|
/// Updates the access time of the asset if it was accessed above a given threshhold amount of time.
|
||||||
// lock (m_dbLock)
|
/// </summary>
|
||||||
// {
|
/// <remarks>
|
||||||
// using (MySqlConnection dbcon = new MySqlConnection(m_connectionString))
|
/// This gives us some insight into assets which haven't ben accessed for a long period. This is only done
|
||||||
// {
|
/// over the threshold time to avoid excessive database writes as assets are fetched.
|
||||||
// dbcon.Open();
|
/// </remarks>
|
||||||
// MySqlCommand cmd =
|
/// <param name='asset'></param>
|
||||||
// new MySqlCommand("update assets set access_time=?access_time where id=?id",
|
/// <param name='accessTime'></param>
|
||||||
// dbcon);
|
private void UpdateAccessTime(AssetMetadata assetMetadata, int accessTime)
|
||||||
//
|
{
|
||||||
// // need to ensure we dispose
|
DateTime now = DateTime.UtcNow;
|
||||||
// try
|
|
||||||
// {
|
if ((now - Utils.UnixTimeToDateTime(accessTime)).TotalDays < DaysBetweenAccessTimeUpdates)
|
||||||
// using (cmd)
|
return;
|
||||||
// {
|
|
||||||
// // create unix epoch time
|
lock (m_dbLock)
|
||||||
// int now = (int)Utils.DateTimeToUnixTime(DateTime.UtcNow);
|
{
|
||||||
// cmd.Parameters.AddWithValue("?id", asset.ID);
|
using (MySqlConnection dbcon = new MySqlConnection(m_connectionString))
|
||||||
// cmd.Parameters.AddWithValue("?access_time", now);
|
{
|
||||||
// cmd.ExecuteNonQuery();
|
dbcon.Open();
|
||||||
// cmd.Dispose();
|
MySqlCommand cmd =
|
||||||
// }
|
new MySqlCommand("update XAssetsMeta set AccessTime=?AccessTime where ID=?ID", dbcon);
|
||||||
// }
|
|
||||||
// catch (Exception e)
|
try
|
||||||
// {
|
{
|
||||||
// m_log.ErrorFormat(
|
using (cmd)
|
||||||
// "[ASSETS DB]: " +
|
{
|
||||||
// "MySql failure updating access_time for asset {0} with name {1}" + Environment.NewLine + e.ToString()
|
// create unix epoch time
|
||||||
// + Environment.NewLine + "Attempting reconnection", asset.FullID, asset.Name);
|
cmd.Parameters.AddWithValue("?ID", assetMetadata.ID);
|
||||||
// }
|
cmd.Parameters.AddWithValue("?AccessTime", (int)Utils.DateTimeToUnixTime(now));
|
||||||
// }
|
cmd.ExecuteNonQuery();
|
||||||
// }
|
}
|
||||||
//
|
}
|
||||||
// }
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
m_log.ErrorFormat(
|
||||||
|
"[XASSET MYSQL DB]: Failure updating access_time for asset {0} with name {1}",
|
||||||
|
assetMetadata.ID, assetMetadata.Name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// We assume we already have the m_dbLock.
|
/// We assume we already have the m_dbLock.
|
||||||
|
@ -349,9 +368,9 @@ namespace OpenSim.Data.MySQL
|
||||||
|
|
||||||
bool exists = false;
|
bool exists = false;
|
||||||
|
|
||||||
using (MySqlCommand cmd = new MySqlCommand("SELECT hash FROM xassetsdata WHERE hash=?hash", dbcon))
|
using (MySqlCommand cmd = new MySqlCommand("SELECT Hash FROM XAssetsData WHERE Hash=?Hash", dbcon))
|
||||||
{
|
{
|
||||||
cmd.Parameters.AddWithValue("?hash", hash);
|
cmd.Parameters.AddWithValue("?Hash", hash);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
@ -391,9 +410,9 @@ namespace OpenSim.Data.MySQL
|
||||||
using (MySqlConnection dbcon = new MySqlConnection(m_connectionString))
|
using (MySqlConnection dbcon = new MySqlConnection(m_connectionString))
|
||||||
{
|
{
|
||||||
dbcon.Open();
|
dbcon.Open();
|
||||||
using (MySqlCommand cmd = new MySqlCommand("SELECT id FROM xassetsmeta WHERE id=?id", dbcon))
|
using (MySqlCommand cmd = new MySqlCommand("SELECT ID FROM XAssetsMeta WHERE ID=?ID", dbcon))
|
||||||
{
|
{
|
||||||
cmd.Parameters.AddWithValue("?id", uuid.ToString());
|
cmd.Parameters.AddWithValue("?ID", uuid.ToString());
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
@ -408,8 +427,7 @@ namespace OpenSim.Data.MySQL
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
m_log.ErrorFormat(
|
m_log.Error(string.Format("[XASSETS DB]: MySql failure fetching asset {0}", uuid), e);
|
||||||
"[XASSETS DB]: MySql failure fetching asset {0}" + Environment.NewLine + e.ToString(), uuid);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -418,6 +436,7 @@ namespace OpenSim.Data.MySQL
|
||||||
return assetExists;
|
return assetExists;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Returns a list of AssetMetadata objects. The list is a subset of
|
/// Returns a list of AssetMetadata objects. The list is a subset of
|
||||||
/// the entire data set offset by <paramref name="start" /> containing
|
/// the entire data set offset by <paramref name="start" /> containing
|
||||||
|
@ -435,7 +454,7 @@ namespace OpenSim.Data.MySQL
|
||||||
using (MySqlConnection dbcon = new MySqlConnection(m_connectionString))
|
using (MySqlConnection dbcon = new MySqlConnection(m_connectionString))
|
||||||
{
|
{
|
||||||
dbcon.Open();
|
dbcon.Open();
|
||||||
MySqlCommand cmd = new MySqlCommand("SELECT name,description,asset_type,temporary,id,asset_flags,creator_id FROM xassetsmeta LIMIT ?start, ?count", dbcon);
|
MySqlCommand cmd = new MySqlCommand("SELECT Name, Description, AccessTime, AssetType, Temporary, ID, AssetFlags, CreatorID FROM XAssetsMeta LIMIT ?start, ?count", dbcon);
|
||||||
cmd.Parameters.AddWithValue("?start", start);
|
cmd.Parameters.AddWithValue("?start", start);
|
||||||
cmd.Parameters.AddWithValue("?count", count);
|
cmd.Parameters.AddWithValue("?count", count);
|
||||||
|
|
||||||
|
@ -446,17 +465,19 @@ namespace OpenSim.Data.MySQL
|
||||||
while (dbReader.Read())
|
while (dbReader.Read())
|
||||||
{
|
{
|
||||||
AssetMetadata metadata = new AssetMetadata();
|
AssetMetadata metadata = new AssetMetadata();
|
||||||
metadata.Name = (string)dbReader["name"];
|
metadata.Name = (string)dbReader["Name"];
|
||||||
metadata.Description = (string)dbReader["description"];
|
metadata.Description = (string)dbReader["Description"];
|
||||||
metadata.Type = (sbyte)dbReader["asset_type"];
|
metadata.Type = (sbyte)dbReader["AssetType"];
|
||||||
metadata.Temporary = Convert.ToBoolean(dbReader["temporary"]); // Not sure if this is correct.
|
metadata.Temporary = Convert.ToBoolean(dbReader["Temporary"]); // Not sure if this is correct.
|
||||||
metadata.Flags = (AssetFlags)Convert.ToInt32(dbReader["asset_flags"]);
|
metadata.Flags = (AssetFlags)Convert.ToInt32(dbReader["AssetFlags"]);
|
||||||
metadata.FullID = DBGuid.FromDB(dbReader["id"]);
|
metadata.FullID = DBGuid.FromDB(dbReader["ID"]);
|
||||||
metadata.CreatorID = dbReader["creator_id"].ToString();
|
metadata.CreatorID = dbReader["CreatorID"].ToString();
|
||||||
|
|
||||||
// We'll ignore this for now - it appears unused!
|
// We'll ignore this for now - it appears unused!
|
||||||
// metadata.SHA1 = dbReader["hash"]);
|
// metadata.SHA1 = dbReader["hash"]);
|
||||||
|
|
||||||
|
UpdateAccessTime(metadata, (int)dbReader["AccessTime"]);
|
||||||
|
|
||||||
retList.Add(metadata);
|
retList.Add(metadata);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -481,9 +502,9 @@ namespace OpenSim.Data.MySQL
|
||||||
{
|
{
|
||||||
dbcon.Open();
|
dbcon.Open();
|
||||||
|
|
||||||
using (MySqlCommand cmd = new MySqlCommand("delete from xassetsmeta where id=?id", dbcon))
|
using (MySqlCommand cmd = new MySqlCommand("delete from XAssetsMeta where ID=?ID", dbcon))
|
||||||
{
|
{
|
||||||
cmd.Parameters.AddWithValue("?id", id);
|
cmd.Parameters.AddWithValue("?ID", id);
|
||||||
cmd.ExecuteNonQuery();
|
cmd.ExecuteNonQuery();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -61,5 +61,5 @@ using System.Runtime.InteropServices;
|
||||||
// You can specify all the values or you can default the Revision and Build Numbers
|
// You can specify all the values or you can default the Revision and Build Numbers
|
||||||
// by using the '*' as shown below:
|
// by using the '*' as shown below:
|
||||||
|
|
||||||
[assembly : AssemblyVersion("0.7.5.*")]
|
[assembly : AssemblyVersion("0.7.6.*")]
|
||||||
[assembly : AssemblyFileVersion("0.6.5.0")]
|
|
||||||
|
|
|
@ -0,0 +1,24 @@
|
||||||
|
:VERSION 1 # --------------------------
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
CREATE TABLE `im_offline` (
|
||||||
|
`ID` MEDIUMINT NOT NULL AUTO_INCREMENT,
|
||||||
|
`PrincipalID` char(36) NOT NULL default '',
|
||||||
|
`Message` text NOT NULL,
|
||||||
|
`TMStamp` timestamp NOT NULL,
|
||||||
|
PRIMARY KEY (`ID`),
|
||||||
|
KEY `PrincipalID` (`PrincipalID`)
|
||||||
|
) ENGINE=MyISAM;
|
||||||
|
|
||||||
|
COMMIT;
|
||||||
|
|
||||||
|
:VERSION 2 # --------------------------
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
INSERT INTO `im_offline` SELECT * from `diva_im_offline`;
|
||||||
|
DROP TABLE `diva_im_offline`;
|
||||||
|
DELETE FROM `migrations` WHERE name='diva_im_Store';
|
||||||
|
|
||||||
|
COMMIT;
|
|
@ -902,3 +902,23 @@ BEGIN;
|
||||||
CREATE TABLE `regionextra` (`RegionID` char(36) not null, `Name` varchar(32) not null, `value` text, primary key(`RegionID`, `Name`));
|
CREATE TABLE `regionextra` (`RegionID` char(36) not null, `Name` varchar(32) not null, `value` text, primary key(`RegionID`, `Name`));
|
||||||
|
|
||||||
COMMIT;
|
COMMIT;
|
||||||
|
|
||||||
|
:VERSION 46 #---------------- Dynamic attributes
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
ALTER TABLE prims ADD COLUMN DynAttrs TEXT;
|
||||||
|
|
||||||
|
COMMIT;
|
||||||
|
|
||||||
|
:VERSION 47 #---------------- Extra physics params
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
ALTER TABLE prims ADD COLUMN `PhysicsShapeType` tinyint(4) NOT NULL default '0';
|
||||||
|
ALTER TABLE prims ADD COLUMN `Density` double NOT NULL default '1000';
|
||||||
|
ALTER TABLE prims ADD COLUMN `GravityModifier` double NOT NULL default '1';
|
||||||
|
ALTER TABLE prims ADD COLUMN `Friction` double NOT NULL default '0.6';
|
||||||
|
ALTER TABLE prims ADD COLUMN `Restitution` double NOT NULL default '0.5';
|
||||||
|
|
||||||
|
COMMIT;
|
||||||
|
|
|
@ -3,24 +3,24 @@
|
||||||
|
|
||||||
BEGIN;
|
BEGIN;
|
||||||
|
|
||||||
CREATE TABLE `xassetsmeta` (
|
CREATE TABLE `XAssetsMeta` (
|
||||||
`id` char(36) NOT NULL,
|
`ID` char(36) NOT NULL,
|
||||||
`hash` binary(32) NOT NULL,
|
`Hash` binary(32) NOT NULL,
|
||||||
`name` varchar(64) NOT NULL,
|
`Name` varchar(64) NOT NULL,
|
||||||
`description` varchar(64) NOT NULL,
|
`Description` varchar(64) NOT NULL,
|
||||||
`asset_type` tinyint(4) NOT NULL,
|
`AssetType` tinyint(4) NOT NULL,
|
||||||
`local` tinyint(1) NOT NULL,
|
`Local` tinyint(1) NOT NULL,
|
||||||
`temporary` tinyint(1) NOT NULL,
|
`Temporary` tinyint(1) NOT NULL,
|
||||||
`create_time` int(11) NOT NULL,
|
`CreateTime` int(11) NOT NULL,
|
||||||
`access_time` int(11) NOT NULL,
|
`AccessTime` int(11) NOT NULL,
|
||||||
`asset_flags` int(11) NOT NULL,
|
`AssetFlags` int(11) NOT NULL,
|
||||||
`creator_id` varchar(128) NOT NULL,
|
`CreatorID` varchar(128) NOT NULL,
|
||||||
PRIMARY KEY (`id`)
|
PRIMARY KEY (`id`)
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='Version 1';
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='Version 1';
|
||||||
|
|
||||||
CREATE TABLE `xassetsdata` (
|
CREATE TABLE `XAssetsData` (
|
||||||
`hash` binary(32) NOT NULL,
|
`Hash` binary(32) NOT NULL,
|
||||||
`data` longblob NOT NULL,
|
`Data` longblob NOT NULL,
|
||||||
PRIMARY KEY (`hash`)
|
PRIMARY KEY (`hash`)
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='Version 1';
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='Version 1';
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,115 @@
|
||||||
|
:VERSION 1 # --------------------------
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
CREATE TABLE `os_groups_groups` (
|
||||||
|
`GroupID` char(36) NOT NULL default '',
|
||||||
|
`Location` varchar(255) NOT NULL default '',
|
||||||
|
`Name` varchar(255) NOT NULL default '',
|
||||||
|
`Charter` text NOT NULL,
|
||||||
|
`InsigniaID` char(36) NOT NULL default '',
|
||||||
|
`FounderID` char(36) NOT NULL default '',
|
||||||
|
`MembershipFee` int(11) NOT NULL default '0',
|
||||||
|
`OpenEnrollment` varchar(255) NOT NULL default '',
|
||||||
|
`ShowInList` int(4) NOT NULL default '0',
|
||||||
|
`AllowPublish` int(4) NOT NULL default '0',
|
||||||
|
`MaturePublish` int(4) NOT NULL default '0',
|
||||||
|
`OwnerRoleID` char(36) NOT NULL default '',
|
||||||
|
PRIMARY KEY (`GroupID`),
|
||||||
|
UNIQUE KEY `Name` (`Name`),
|
||||||
|
FULLTEXT KEY `Name_2` (`Name`)
|
||||||
|
) ENGINE=MyISAM;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE TABLE `os_groups_membership` (
|
||||||
|
`GroupID`char(36) NOT NULL default '',
|
||||||
|
`PrincipalID` VARCHAR(255) NOT NULL default '',
|
||||||
|
`SelectedRoleID` char(36) NOT NULL default '',
|
||||||
|
`Contribution` int(11) NOT NULL default '0',
|
||||||
|
`ListInProfile` int(4) NOT NULL default '1',
|
||||||
|
`AcceptNotices` int(4) NOT NULL default '1',
|
||||||
|
`AccessToken` char(36) NOT NULL default '',
|
||||||
|
PRIMARY KEY (`GroupID`,`PrincipalID`),
|
||||||
|
KEY `PrincipalID` (`PrincipalID`)
|
||||||
|
) ENGINE=MyISAM;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE TABLE `os_groups_roles` (
|
||||||
|
`GroupID` char(36) NOT NULL default '',
|
||||||
|
`RoleID` char(36) NOT NULL default '',
|
||||||
|
`Name` varchar(255) NOT NULL default '',
|
||||||
|
`Description` varchar(255) NOT NULL default '',
|
||||||
|
`Title` varchar(255) NOT NULL default '',
|
||||||
|
`Powers` bigint(20) unsigned NOT NULL default '0',
|
||||||
|
PRIMARY KEY (`GroupID`,`RoleID`),
|
||||||
|
KEY `GroupID` (`GroupID`)
|
||||||
|
) ENGINE=MyISAM;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE TABLE `os_groups_rolemembership` (
|
||||||
|
`GroupID` char(36) NOT NULL default '',
|
||||||
|
`RoleID` char(36) NOT NULL default '',
|
||||||
|
`PrincipalID` VARCHAR(255) NOT NULL default '',
|
||||||
|
PRIMARY KEY (`GroupID`,`RoleID`,`PrincipalID`),
|
||||||
|
KEY `PrincipalID` (`PrincipalID`)
|
||||||
|
) ENGINE=MyISAM;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE TABLE `os_groups_invites` (
|
||||||
|
`InviteID` char(36) NOT NULL default '',
|
||||||
|
`GroupID` char(36) NOT NULL default '',
|
||||||
|
`RoleID` char(36) NOT NULL default '',
|
||||||
|
`PrincipalID` VARCHAR(255) NOT NULL default '',
|
||||||
|
`TMStamp` timestamp NOT NULL,
|
||||||
|
PRIMARY KEY (`InviteID`),
|
||||||
|
UNIQUE KEY `PrincipalGroup` (`GroupID`,`PrincipalID`)
|
||||||
|
) ENGINE=MyISAM;
|
||||||
|
|
||||||
|
|
||||||
|
CREATE TABLE `os_groups_notices` (
|
||||||
|
`GroupID` char(36) NOT NULL default '',
|
||||||
|
`NoticeID` char(36) NOT NULL default '',
|
||||||
|
`TMStamp` int(10) unsigned NOT NULL default '0',
|
||||||
|
`FromName` varchar(255) NOT NULL default '',
|
||||||
|
`Subject` varchar(255) NOT NULL default '',
|
||||||
|
`Message` text NOT NULL,
|
||||||
|
`HasAttachment` int(4) NOT NULL default '0',
|
||||||
|
`AttachmentType` int(4) NOT NULL default '0',
|
||||||
|
`AttachmentName` varchar(128) NOT NULL default '',
|
||||||
|
`AttachmentItemID` char(36) NOT NULL default '',
|
||||||
|
`AttachmentOwnerID` varchar(255) NOT NULL default '',
|
||||||
|
PRIMARY KEY (`NoticeID`),
|
||||||
|
KEY `GroupID` (`GroupID`),
|
||||||
|
KEY `TMStamp` (`TMStamp`)
|
||||||
|
) ENGINE=MyISAM;
|
||||||
|
|
||||||
|
CREATE TABLE `os_groups_principals` (
|
||||||
|
`PrincipalID` VARCHAR(255) NOT NULL default '',
|
||||||
|
`ActiveGroupID` char(36) NOT NULL default '',
|
||||||
|
PRIMARY KEY (`PrincipalID`)
|
||||||
|
) ENGINE=MyISAM;
|
||||||
|
|
||||||
|
COMMIT;
|
||||||
|
|
||||||
|
:VERSION 2 # --------------------------
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
INSERT INTO `os_groups_groups` SELECT * from `diva_groups_groups`;
|
||||||
|
DROP TABLE `diva_groups_groups`;
|
||||||
|
INSERT INTO `os_groups_membership` SELECT * from `diva_groups_membership`;
|
||||||
|
DROP TABLE `diva_groups_membership`;
|
||||||
|
INSERT INTO `os_groups_roles` SELECT * from `diva_groups_roles`;
|
||||||
|
DROP TABLE `diva_groups_roles`;
|
||||||
|
INSERT INTO `os_groups_rolemembership` SELECT * from `diva_groups_rolemembership`;
|
||||||
|
DROP TABLE `diva_groups_rolemembership`;
|
||||||
|
INSERT INTO `os_groups_invites` SELECT * from `diva_groups_invites`;
|
||||||
|
DROP TABLE `diva_groups_invites`;
|
||||||
|
INSERT INTO `os_groups_notices` SELECT * from `diva_groups_notices`;
|
||||||
|
DROP TABLE `diva_groups_notices`;
|
||||||
|
INSERT INTO `os_groups_principals` SELECT * from `diva_groups_principals`;
|
||||||
|
DROP TABLE `diva_groups_principals`;
|
||||||
|
|
||||||
|
DELETE FROM `migrations` WHERE name='diva_im_Store';
|
||||||
|
|
||||||
|
COMMIT;
|
|
@ -0,0 +1,67 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) Contributors, http://opensimulator.org/
|
||||||
|
* See CONTRIBUTORS.TXT for a full list of copyright holders.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
* * Neither the name of the OpenSimulator Project nor the
|
||||||
|
* names of its contributors may be used to endorse or promote products
|
||||||
|
* derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
|
||||||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
|
||||||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using log4net;
|
||||||
|
using OpenMetaverse;
|
||||||
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Data;
|
||||||
|
|
||||||
|
namespace OpenSim.Data.Null
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Not a proper generic data handler yet - probably needs to actually store the data as well instead of relying
|
||||||
|
/// on descendent classes
|
||||||
|
/// </summary>
|
||||||
|
public class NullGenericDataHandler
|
||||||
|
{
|
||||||
|
protected List<T> Get<T>(string[] fields, string[] vals, List<T> inputEntities)
|
||||||
|
{
|
||||||
|
List<T> entities = inputEntities;
|
||||||
|
|
||||||
|
for (int i = 0; i < fields.Length; i++)
|
||||||
|
{
|
||||||
|
entities
|
||||||
|
= entities.Where(
|
||||||
|
e =>
|
||||||
|
{
|
||||||
|
FieldInfo fi = typeof(T).GetField(fields[i]);
|
||||||
|
if (fi == null)
|
||||||
|
throw new NotImplementedException(string.Format("No field {0} for val {1}", fields[i], vals[i]));
|
||||||
|
|
||||||
|
return fi.GetValue(e).ToString() == vals[i];
|
||||||
|
}
|
||||||
|
).ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
return entities;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -222,5 +222,13 @@ namespace OpenSim.Data.Null
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public bool VerifyAgent(UUID agentId, UUID secureSessionID)
|
||||||
|
{
|
||||||
|
if (Instance != this)
|
||||||
|
return Instance.VerifyAgent(agentId, secureSessionID);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -113,11 +113,14 @@ namespace OpenSim.Data.Null
|
||||||
// Find region data
|
// Find region data
|
||||||
List<RegionData> ret = new List<RegionData>();
|
List<RegionData> ret = new List<RegionData>();
|
||||||
|
|
||||||
foreach (RegionData r in m_regionData.Values)
|
lock (m_regionData)
|
||||||
{
|
{
|
||||||
// m_log.DebugFormat("[NULL REGION DATA]: comparing {0} to {1}", cleanName, r.RegionName.ToLower());
|
foreach (RegionData r in m_regionData.Values)
|
||||||
|
{
|
||||||
|
// m_log.DebugFormat("[NULL REGION DATA]: comparing {0} to {1}", cleanName, r.RegionName.ToLower());
|
||||||
if (queryMatch(r.RegionName.ToLower()))
|
if (queryMatch(r.RegionName.ToLower()))
|
||||||
ret.Add(r);
|
ret.Add(r);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret.Count > 0)
|
if (ret.Count > 0)
|
||||||
|
@ -133,10 +136,13 @@ namespace OpenSim.Data.Null
|
||||||
|
|
||||||
List<RegionData> ret = new List<RegionData>();
|
List<RegionData> ret = new List<RegionData>();
|
||||||
|
|
||||||
foreach (RegionData r in m_regionData.Values)
|
lock (m_regionData)
|
||||||
{
|
{
|
||||||
if (r.posX == posX && r.posY == posY)
|
foreach (RegionData r in m_regionData.Values)
|
||||||
ret.Add(r);
|
{
|
||||||
|
if (r.posX == posX && r.posY == posY)
|
||||||
|
ret.Add(r);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret.Count > 0)
|
if (ret.Count > 0)
|
||||||
|
@ -150,8 +156,11 @@ namespace OpenSim.Data.Null
|
||||||
if (m_useStaticInstance && Instance != this)
|
if (m_useStaticInstance && Instance != this)
|
||||||
return Instance.Get(regionID, scopeID);
|
return Instance.Get(regionID, scopeID);
|
||||||
|
|
||||||
if (m_regionData.ContainsKey(regionID))
|
lock (m_regionData)
|
||||||
return m_regionData[regionID];
|
{
|
||||||
|
if (m_regionData.ContainsKey(regionID))
|
||||||
|
return m_regionData[regionID];
|
||||||
|
}
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -163,10 +172,13 @@ namespace OpenSim.Data.Null
|
||||||
|
|
||||||
List<RegionData> ret = new List<RegionData>();
|
List<RegionData> ret = new List<RegionData>();
|
||||||
|
|
||||||
foreach (RegionData r in m_regionData.Values)
|
lock (m_regionData)
|
||||||
{
|
{
|
||||||
if (r.posX >= startX && r.posX <= endX && r.posY >= startY && r.posY <= endY)
|
foreach (RegionData r in m_regionData.Values)
|
||||||
ret.Add(r);
|
{
|
||||||
|
if (r.posX >= startX && r.posX <= endX && r.posY >= startY && r.posY <= endY)
|
||||||
|
ret.Add(r);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -180,7 +192,10 @@ namespace OpenSim.Data.Null
|
||||||
// m_log.DebugFormat(
|
// m_log.DebugFormat(
|
||||||
// "[NULL REGION DATA]: Storing region {0} {1}, scope {2}", data.RegionName, data.RegionID, data.ScopeID);
|
// "[NULL REGION DATA]: Storing region {0} {1}, scope {2}", data.RegionName, data.RegionID, data.ScopeID);
|
||||||
|
|
||||||
m_regionData[data.RegionID] = data;
|
lock (m_regionData)
|
||||||
|
{
|
||||||
|
m_regionData[data.RegionID] = data;
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -190,10 +205,13 @@ namespace OpenSim.Data.Null
|
||||||
if (m_useStaticInstance && Instance != this)
|
if (m_useStaticInstance && Instance != this)
|
||||||
return Instance.SetDataItem(regionID, item, value);
|
return Instance.SetDataItem(regionID, item, value);
|
||||||
|
|
||||||
if (!m_regionData.ContainsKey(regionID))
|
lock (m_regionData)
|
||||||
return false;
|
{
|
||||||
|
if (!m_regionData.ContainsKey(regionID))
|
||||||
|
return false;
|
||||||
|
|
||||||
m_regionData[regionID].Data[item] = value;
|
m_regionData[regionID].Data[item] = value;
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -205,10 +223,13 @@ namespace OpenSim.Data.Null
|
||||||
|
|
||||||
// m_log.DebugFormat("[NULL REGION DATA]: Deleting region {0}", regionID);
|
// m_log.DebugFormat("[NULL REGION DATA]: Deleting region {0}", regionID);
|
||||||
|
|
||||||
if (!m_regionData.ContainsKey(regionID))
|
lock (m_regionData)
|
||||||
return false;
|
{
|
||||||
|
if (!m_regionData.ContainsKey(regionID))
|
||||||
|
return false;
|
||||||
|
|
||||||
m_regionData.Remove(regionID);
|
m_regionData.Remove(regionID);
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -238,10 +259,13 @@ namespace OpenSim.Data.Null
|
||||||
|
|
||||||
List<RegionData> ret = new List<RegionData>();
|
List<RegionData> ret = new List<RegionData>();
|
||||||
|
|
||||||
foreach (RegionData r in m_regionData.Values)
|
lock (m_regionData)
|
||||||
{
|
{
|
||||||
if ((Convert.ToInt32(r.Data["flags"]) & regionFlags) != 0)
|
foreach (RegionData r in m_regionData.Values)
|
||||||
ret.Add(r);
|
{
|
||||||
|
if ((Convert.ToInt32(r.Data["flags"]) & regionFlags) != 0)
|
||||||
|
ret.Add(r);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
|
|
|
@ -26,66 +26,65 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
using OpenMetaverse;
|
using System.Collections;
|
||||||
using log4net;
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
|
using System.Threading;
|
||||||
|
using log4net;
|
||||||
|
using OpenMetaverse;
|
||||||
using OpenSim.Framework;
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Data;
|
||||||
|
|
||||||
namespace OpenSim.Region.Framework.Scenes.Scripting
|
namespace OpenSim.Data.Null
|
||||||
{
|
{
|
||||||
public class NullScriptHost : IScriptHost
|
public class NullXGroupData : NullGenericDataHandler, IXGroupData
|
||||||
{
|
{
|
||||||
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
// private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
private Vector3 m_pos = new Vector3(((int)Constants.RegionSize * 0.5f), ((int)Constants.RegionSize * 0.5f), 30);
|
private Dictionary<UUID, XGroup> m_groups = new Dictionary<UUID, XGroup>();
|
||||||
|
|
||||||
public string Name
|
public NullXGroupData(string connectionString, string realm) {}
|
||||||
|
|
||||||
|
public bool StoreGroup(XGroup group)
|
||||||
{
|
{
|
||||||
get { return "Object"; }
|
lock (m_groups)
|
||||||
set { }
|
{
|
||||||
|
m_groups[group.groupID] = group.Clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public string SitName
|
public XGroup[] GetGroups(string field, string val)
|
||||||
{
|
{
|
||||||
get { return String.Empty; }
|
return GetGroups(new string[] { field }, new string[] { val });
|
||||||
set { }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public string TouchName
|
public XGroup[] GetGroups(string[] fields, string[] vals)
|
||||||
{
|
{
|
||||||
get { return String.Empty; }
|
lock (m_groups)
|
||||||
set { }
|
{
|
||||||
|
List<XGroup> origGroups = Get<XGroup>(fields, vals, m_groups.Values.ToList());
|
||||||
|
|
||||||
|
return origGroups.Select(g => g.Clone()).ToArray();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public string Description
|
public bool DeleteGroups(string field, string val)
|
||||||
{
|
{
|
||||||
get { return String.Empty; }
|
return DeleteGroups(new string[] { field }, new string[] { val });
|
||||||
set { }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public UUID UUID
|
public bool DeleteGroups(string[] fields, string[] vals)
|
||||||
{
|
{
|
||||||
get { return UUID.Zero; }
|
lock (m_groups)
|
||||||
}
|
{
|
||||||
|
XGroup[] groupsToDelete = GetGroups(fields, vals);
|
||||||
|
Array.ForEach(groupsToDelete, g => m_groups.Remove(g.groupID));
|
||||||
|
}
|
||||||
|
|
||||||
public UUID OwnerID
|
return true;
|
||||||
{
|
|
||||||
get { return UUID.Zero; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public UUID CreatorID
|
|
||||||
{
|
|
||||||
get { return UUID.Zero; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public Vector3 AbsolutePosition
|
|
||||||
{
|
|
||||||
get { return m_pos; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SetText(string text, Vector3 color, double alpha)
|
|
||||||
{
|
|
||||||
m_log.Warn("Tried to SetText "+text+" on NullScriptHost");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -61,5 +61,5 @@ using System.Runtime.InteropServices;
|
||||||
// You can specify all the values or you can default the Revision and Build Numbers
|
// You can specify all the values or you can default the Revision and Build Numbers
|
||||||
// by using the '*' as shown below:
|
// by using the '*' as shown below:
|
||||||
|
|
||||||
[assembly : AssemblyVersion("0.7.5.*")]
|
[assembly : AssemblyVersion("0.7.6.*")]
|
||||||
[assembly : AssemblyFileVersion("0.6.5.0")]
|
|
||||||
|
|
|
@ -61,5 +61,5 @@ using System.Runtime.InteropServices;
|
||||||
// You can specify all the values or you can default the Revision and Build Numbers
|
// You can specify all the values or you can default the Revision and Build Numbers
|
||||||
// by using the '*' as shown below:
|
// by using the '*' as shown below:
|
||||||
|
|
||||||
[assembly : AssemblyVersion("0.7.5.*")]
|
[assembly : AssemblyVersion("0.7.6.*")]
|
||||||
[assembly : AssemblyFileVersion("0.6.5.0")]
|
|
||||||
|
|
|
@ -61,5 +61,5 @@ using System.Runtime.InteropServices;
|
||||||
// You can specify all the values or you can default the Revision and Build Numbers
|
// You can specify all the values or you can default the Revision and Build Numbers
|
||||||
// by using the '*' as shown below:
|
// by using the '*' as shown below:
|
||||||
|
|
||||||
[assembly : AssemblyVersion("0.7.5.*")]
|
[assembly : AssemblyVersion("0.7.6.*")]
|
||||||
[assembly : AssemblyFileVersion("0.6.5.0")]
|
|
||||||
|
|
|
@ -575,3 +575,20 @@ CREATE TABLE `regionenvironment` (
|
||||||
);
|
);
|
||||||
|
|
||||||
COMMIT;
|
COMMIT;
|
||||||
|
|
||||||
|
:VERSION 27
|
||||||
|
BEGIN;
|
||||||
|
ALTER TABLE prims ADD COLUMN DynAttrs TEXT;
|
||||||
|
COMMIT;
|
||||||
|
|
||||||
|
:VERSION 28
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
ALTER TABLE prims ADD COLUMN `PhysicsShapeType` tinyint(4) NOT NULL default '0';
|
||||||
|
ALTER TABLE prims ADD COLUMN `Density` double NOT NULL default '1000';
|
||||||
|
ALTER TABLE prims ADD COLUMN `GravityModifier` double NOT NULL default '1';
|
||||||
|
ALTER TABLE prims ADD COLUMN `Friction` double NOT NULL default '0.6';
|
||||||
|
ALTER TABLE prims ADD COLUMN `Restitution` double NOT NULL default '0.5';
|
||||||
|
|
||||||
|
COMMIT;
|
||||||
|
|
|
@ -46,7 +46,7 @@ namespace OpenSim.Data.SQLite
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class SQLiteAssetData : AssetDataBase
|
public class SQLiteAssetData : AssetDataBase
|
||||||
{
|
{
|
||||||
// private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
private const string SelectAssetSQL = "select * from assets where UUID=:UUID";
|
private const string SelectAssetSQL = "select * from assets where UUID=:UUID";
|
||||||
private const string SelectAssetMetadataSQL = "select Name, Description, Type, Temporary, asset_flags, UUID, CreatorID from assets limit :start, :count";
|
private const string SelectAssetMetadataSQL = "select Name, Description, Type, Temporary, asset_flags, UUID, CreatorID from assets limit :start, :count";
|
||||||
|
@ -133,6 +133,24 @@ namespace OpenSim.Data.SQLite
|
||||||
/// <param name="asset">Asset Base</param>
|
/// <param name="asset">Asset Base</param>
|
||||||
override public bool StoreAsset(AssetBase asset)
|
override public bool StoreAsset(AssetBase asset)
|
||||||
{
|
{
|
||||||
|
string assetName = asset.Name;
|
||||||
|
if (asset.Name.Length > 64)
|
||||||
|
{
|
||||||
|
assetName = asset.Name.Substring(0, 64);
|
||||||
|
m_log.WarnFormat(
|
||||||
|
"[ASSET DB]: Name '{0}' for asset {1} truncated from {2} to {3} characters on add",
|
||||||
|
asset.Name, asset.ID, asset.Name.Length, assetName.Length);
|
||||||
|
}
|
||||||
|
|
||||||
|
string assetDescription = asset.Description;
|
||||||
|
if (asset.Description.Length > 64)
|
||||||
|
{
|
||||||
|
assetDescription = asset.Description.Substring(0, 64);
|
||||||
|
m_log.WarnFormat(
|
||||||
|
"[ASSET DB]: Description '{0}' for asset {1} truncated from {2} to {3} characters on add",
|
||||||
|
asset.Description, asset.ID, asset.Description.Length, assetDescription.Length);
|
||||||
|
}
|
||||||
|
|
||||||
//m_log.Info("[ASSET DB]: Creating Asset " + asset.FullID.ToString());
|
//m_log.Info("[ASSET DB]: Creating Asset " + asset.FullID.ToString());
|
||||||
if (ExistsAsset(asset.FullID))
|
if (ExistsAsset(asset.FullID))
|
||||||
{
|
{
|
||||||
|
@ -143,8 +161,8 @@ namespace OpenSim.Data.SQLite
|
||||||
using (SqliteCommand cmd = new SqliteCommand(UpdateAssetSQL, m_conn))
|
using (SqliteCommand cmd = new SqliteCommand(UpdateAssetSQL, m_conn))
|
||||||
{
|
{
|
||||||
cmd.Parameters.Add(new SqliteParameter(":UUID", asset.FullID.ToString()));
|
cmd.Parameters.Add(new SqliteParameter(":UUID", asset.FullID.ToString()));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Name", asset.Name));
|
cmd.Parameters.Add(new SqliteParameter(":Name", assetName));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Description", asset.Description));
|
cmd.Parameters.Add(new SqliteParameter(":Description", assetDescription));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Type", asset.Type));
|
cmd.Parameters.Add(new SqliteParameter(":Type", asset.Type));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Local", asset.Local));
|
cmd.Parameters.Add(new SqliteParameter(":Local", asset.Local));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Temporary", asset.Temporary));
|
cmd.Parameters.Add(new SqliteParameter(":Temporary", asset.Temporary));
|
||||||
|
@ -164,8 +182,8 @@ namespace OpenSim.Data.SQLite
|
||||||
using (SqliteCommand cmd = new SqliteCommand(InsertAssetSQL, m_conn))
|
using (SqliteCommand cmd = new SqliteCommand(InsertAssetSQL, m_conn))
|
||||||
{
|
{
|
||||||
cmd.Parameters.Add(new SqliteParameter(":UUID", asset.FullID.ToString()));
|
cmd.Parameters.Add(new SqliteParameter(":UUID", asset.FullID.ToString()));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Name", asset.Name));
|
cmd.Parameters.Add(new SqliteParameter(":Name", assetName));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Description", asset.Description));
|
cmd.Parameters.Add(new SqliteParameter(":Description", assetDescription));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Type", asset.Type));
|
cmd.Parameters.Add(new SqliteParameter(":Type", asset.Type));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Local", asset.Local));
|
cmd.Parameters.Add(new SqliteParameter(":Local", asset.Local));
|
||||||
cmd.Parameters.Add(new SqliteParameter(":Temporary", asset.Temporary));
|
cmd.Parameters.Add(new SqliteParameter(":Temporary", asset.Temporary));
|
||||||
|
|
|
@ -1232,6 +1232,14 @@ namespace OpenSim.Data.SQLite
|
||||||
createCol(prims, "VolumeDetect", typeof(Int16));
|
createCol(prims, "VolumeDetect", typeof(Int16));
|
||||||
|
|
||||||
createCol(prims, "MediaURL", typeof(String));
|
createCol(prims, "MediaURL", typeof(String));
|
||||||
|
|
||||||
|
createCol(prims, "DynAttrs", typeof(String));
|
||||||
|
|
||||||
|
createCol(prims, "PhysicsShapeType", typeof(Byte));
|
||||||
|
createCol(prims, "Density", typeof(Double));
|
||||||
|
createCol(prims, "GravityModifier", typeof(Double));
|
||||||
|
createCol(prims, "Friction", typeof(Double));
|
||||||
|
createCol(prims, "Restitution", typeof(Double));
|
||||||
|
|
||||||
// Add in contraints
|
// Add in contraints
|
||||||
prims.PrimaryKey = new DataColumn[] { prims.Columns["UUID"] };
|
prims.PrimaryKey = new DataColumn[] { prims.Columns["UUID"] };
|
||||||
|
@ -1711,6 +1719,22 @@ namespace OpenSim.Data.SQLite
|
||||||
// m_log.DebugFormat("[SQLITE]: MediaUrl type [{0}]", row["MediaURL"].GetType());
|
// m_log.DebugFormat("[SQLITE]: MediaUrl type [{0}]", row["MediaURL"].GetType());
|
||||||
prim.MediaUrl = (string)row["MediaURL"];
|
prim.MediaUrl = (string)row["MediaURL"];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!(row["DynAttrs"] is System.DBNull))
|
||||||
|
{
|
||||||
|
//m_log.DebugFormat("[SQLITE]: DynAttrs type [{0}]", row["DynAttrs"].GetType());
|
||||||
|
prim.DynAttrs = DAMap.FromXml((string)row["DynAttrs"]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
prim.DynAttrs = new DAMap();
|
||||||
|
}
|
||||||
|
|
||||||
|
prim.PhysicsShapeType = Convert.ToByte(row["PhysicsShapeType"]);
|
||||||
|
prim.Density = Convert.ToSingle(row["Density"]);
|
||||||
|
prim.GravityModifier = Convert.ToSingle(row["GravityModifier"]);
|
||||||
|
prim.Friction = Convert.ToSingle(row["Friction"]);
|
||||||
|
prim.Restitution = Convert.ToSingle(row["Restitution"]);
|
||||||
|
|
||||||
return prim;
|
return prim;
|
||||||
}
|
}
|
||||||
|
@ -2133,6 +2157,17 @@ namespace OpenSim.Data.SQLite
|
||||||
row["VolumeDetect"] = 0;
|
row["VolumeDetect"] = 0;
|
||||||
|
|
||||||
row["MediaURL"] = prim.MediaUrl;
|
row["MediaURL"] = prim.MediaUrl;
|
||||||
|
|
||||||
|
if (prim.DynAttrs.Count > 0)
|
||||||
|
row["DynAttrs"] = prim.DynAttrs.ToXml();
|
||||||
|
else
|
||||||
|
row["DynAttrs"] = null;
|
||||||
|
|
||||||
|
row["PhysicsShapeType"] = prim.PhysicsShapeType;
|
||||||
|
row["Density"] = (double)prim.Density;
|
||||||
|
row["GravityModifier"] = (double)prim.GravityModifier;
|
||||||
|
row["Friction"] = (double)prim.Friction;
|
||||||
|
row["Restitution"] = (double)prim.Restitution;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
@ -2392,7 +2427,7 @@ namespace OpenSim.Data.SQLite
|
||||||
|
|
||||||
if (!(row["Media"] is System.DBNull))
|
if (!(row["Media"] is System.DBNull))
|
||||||
s.Media = PrimitiveBaseShape.MediaList.FromXml((string)row["Media"]);
|
s.Media = PrimitiveBaseShape.MediaList.FromXml((string)row["Media"]);
|
||||||
|
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -49,7 +49,7 @@ using OpenSim.Data.SQLite;
|
||||||
namespace OpenSim.Data.Tests
|
namespace OpenSim.Data.Tests
|
||||||
{
|
{
|
||||||
[TestFixture(Description = "Asset store tests (SQLite)")]
|
[TestFixture(Description = "Asset store tests (SQLite)")]
|
||||||
public class SQLiteAssetTests : AssetTests<SqliteConnection, SQLiteAssetData>
|
public class SQLiteAssetTests : AssetTests<SqliteConnection, SQLiteAssetData>
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -33,6 +33,7 @@ using NUnit.Framework;
|
||||||
using NUnit.Framework.Constraints;
|
using NUnit.Framework.Constraints;
|
||||||
using OpenMetaverse;
|
using OpenMetaverse;
|
||||||
using OpenSim.Framework;
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Tests.Common;
|
||||||
using log4net;
|
using log4net;
|
||||||
using System.Data;
|
using System.Data;
|
||||||
using System.Data.Common;
|
using System.Data.Common;
|
||||||
|
@ -43,6 +44,12 @@ namespace OpenSim.Data.Tests
|
||||||
/// <summary>This is a base class for testing any Data service for any DBMS.
|
/// <summary>This is a base class for testing any Data service for any DBMS.
|
||||||
/// Requires NUnit 2.5 or better (to support the generics).
|
/// Requires NUnit 2.5 or better (to support the generics).
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// FIXME: Should extend OpenSimTestCase but compile on mono 2.4.3 currently fails with
|
||||||
|
/// AssetTests`2 : System.MemberAccessException : Cannot create an instance of OpenSim.Data.Tests.AssetTests`2[TConn,TAssetData] because Type.ContainsGenericParameters is true.
|
||||||
|
/// and similar on EstateTests, InventoryTests and RegionTests.
|
||||||
|
/// Runs fine with mono 2.10.8.1, so easiest thing is to wait until min Mono version uplifts.
|
||||||
|
/// </remarks>
|
||||||
/// <typeparam name="TConn"></typeparam>
|
/// <typeparam name="TConn"></typeparam>
|
||||||
/// <typeparam name="TService"></typeparam>
|
/// <typeparam name="TService"></typeparam>
|
||||||
public class BasicDataServiceTest<TConn, TService>
|
public class BasicDataServiceTest<TConn, TService>
|
||||||
|
|
|
@ -36,6 +36,7 @@ using NUnit.Framework;
|
||||||
using NUnit.Framework.Constraints;
|
using NUnit.Framework.Constraints;
|
||||||
using OpenMetaverse;
|
using OpenMetaverse;
|
||||||
using OpenSim.Framework;
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Tests.Common;
|
||||||
|
|
||||||
namespace OpenSim.Data.Tests
|
namespace OpenSim.Data.Tests
|
||||||
{
|
{
|
||||||
|
@ -254,7 +255,7 @@ namespace OpenSim.Data.Tests
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class PropertyCompareConstraintTest
|
public class PropertyCompareConstraintTest : OpenSimTestCase
|
||||||
{
|
{
|
||||||
public class HasInt
|
public class HasInt
|
||||||
{
|
{
|
||||||
|
|
|
@ -34,6 +34,7 @@ using System.Text;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
using OpenMetaverse;
|
using OpenMetaverse;
|
||||||
using OpenSim.Framework;
|
using OpenSim.Framework;
|
||||||
|
using OpenSim.Tests.Common;
|
||||||
|
|
||||||
namespace OpenSim.Data.Tests
|
namespace OpenSim.Data.Tests
|
||||||
{
|
{
|
||||||
|
@ -158,7 +159,7 @@ namespace OpenSim.Data.Tests
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class PropertyScramblerTests
|
public class PropertyScramblerTests : OpenSimTestCase
|
||||||
{
|
{
|
||||||
[Test]
|
[Test]
|
||||||
public void TestScramble()
|
public void TestScramble()
|
||||||
|
|
|
@ -59,5 +59,4 @@ using System.Runtime.InteropServices;
|
||||||
// Revision
|
// Revision
|
||||||
//
|
//
|
||||||
|
|
||||||
[assembly : AssemblyVersion("0.7.5.*")]
|
[assembly : AssemblyVersion("0.7.6.*")]
|
||||||
[assembly : AssemblyFileVersion("0.6.5.0")]
|
|
||||||
|
|
|
@ -29,5 +29,5 @@ using System.Runtime.InteropServices;
|
||||||
// Build Number
|
// Build Number
|
||||||
// Revision
|
// Revision
|
||||||
//
|
//
|
||||||
[assembly: AssemblyVersion("0.7.5.*")]
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
|
||||||
|
|
|
@ -497,8 +497,6 @@ namespace OpenSim.Framework
|
||||||
/// </remarks>
|
/// </remarks>
|
||||||
public List<AvatarAttachment> GetAttachments()
|
public List<AvatarAttachment> GetAttachments()
|
||||||
{
|
{
|
||||||
|
|
||||||
|
|
||||||
lock (m_attachments)
|
lock (m_attachments)
|
||||||
{
|
{
|
||||||
List<AvatarAttachment> alist = new List<AvatarAttachment>();
|
List<AvatarAttachment> alist = new List<AvatarAttachment>();
|
||||||
|
@ -508,7 +506,8 @@ namespace OpenSim.Framework
|
||||||
alist.Add(new AvatarAttachment(attach));
|
alist.Add(new AvatarAttachment(attach));
|
||||||
}
|
}
|
||||||
return alist;
|
return alist;
|
||||||
} }
|
}
|
||||||
|
}
|
||||||
|
|
||||||
internal void AppendAttachment(AvatarAttachment attach)
|
internal void AppendAttachment(AvatarAttachment attach)
|
||||||
{
|
{
|
||||||
|
@ -562,45 +561,59 @@ namespace OpenSim.Framework
|
||||||
if (attachpoint == 0)
|
if (attachpoint == 0)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (item == UUID.Zero)
|
lock (m_attachments)
|
||||||
{
|
{
|
||||||
lock (m_attachments)
|
if (item == UUID.Zero)
|
||||||
{
|
{
|
||||||
if (m_attachments.ContainsKey(attachpoint))
|
if (m_attachments.ContainsKey(attachpoint))
|
||||||
{
|
{
|
||||||
m_attachments.Remove(attachpoint);
|
m_attachments.Remove(attachpoint);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// When a user logs in, the attachment item ids are pulled from persistence in the Avatars table. However,
|
||||||
|
// the asset ids are not saved. When the avatar enters a simulator the attachments are set again. If
|
||||||
|
// we simply perform an item check here then the asset ids (which are now present) are never set, and NPC attachments
|
||||||
|
// later fail unless the attachment is detached and reattached.
|
||||||
|
//
|
||||||
|
// Therefore, we will carry on with the set if the existing attachment has no asset id.
|
||||||
|
AvatarAttachment existingAttachment = GetAttachmentForItem(item);
|
||||||
|
if (existingAttachment != null)
|
||||||
|
{
|
||||||
|
// m_log.DebugFormat(
|
||||||
|
// "[AVATAR APPEARANCE]: Found existing attachment for {0}, asset {1} at point {2}",
|
||||||
|
// existingAttachment.ItemID, existingAttachment.AssetID, existingAttachment.AttachPoint);
|
||||||
|
|
||||||
|
if (existingAttachment.AssetID != UUID.Zero && existingAttachment.AttachPoint == (attachpoint & 0x7F))
|
||||||
|
{
|
||||||
|
m_log.DebugFormat(
|
||||||
|
"[AVATAR APPEARANCE]: Ignoring attempt to attach an already attached item {0} at point {1}",
|
||||||
|
item, attachpoint);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Remove it here so that the later append does not add a second attachment but we still update
|
||||||
|
// the assetID
|
||||||
|
DetachAttachment(existingAttachment.ItemID);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
// check if this is an append or a replace, 0x80 marks it as an append
|
||||||
}
|
if ((attachpoint & 0x80) > 0)
|
||||||
|
{
|
||||||
// When a user logs in, the attachment item ids are pulled from persistence in the Avatars table. However,
|
// strip the append bit
|
||||||
// the asset ids are not saved. When the avatar enters a simulator the attachments are set again. If
|
int point = attachpoint & 0x7F;
|
||||||
// we simply perform an item check here then the asset ids (which are now present) are never set, and NPC attachments
|
AppendAttachment(new AvatarAttachment(point, item, asset));
|
||||||
// later fail unless the attachment is detached and reattached.
|
}
|
||||||
//
|
else
|
||||||
// Therefore, we will carry on with the set if the existing attachment has no asset id.
|
{
|
||||||
AvatarAttachment existingAttachment = GetAttachmentForItem(item);
|
ReplaceAttachment(new AvatarAttachment(attachpoint,item, asset));
|
||||||
if (existingAttachment != null
|
}
|
||||||
&& existingAttachment.AssetID != UUID.Zero
|
|
||||||
&& existingAttachment.AttachPoint == (attachpoint & 0x7F))
|
|
||||||
{
|
|
||||||
// m_log.DebugFormat("[AVATAR APPEARANCE] attempt to attach an already attached item {0}",item);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// check if this is an append or a replace, 0x80 marks it as an append
|
|
||||||
if ((attachpoint & 0x80) > 0)
|
|
||||||
{
|
|
||||||
// strip the append bit
|
|
||||||
int point = attachpoint & 0x7F;
|
|
||||||
AppendAttachment(new AvatarAttachment(point, item, asset));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
ReplaceAttachment(new AvatarAttachment(attachpoint,item, asset));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -649,6 +662,10 @@ namespace OpenSim.Framework
|
||||||
int index = kvp.Value.FindIndex(delegate(AvatarAttachment a) { return a.ItemID == itemID; });
|
int index = kvp.Value.FindIndex(delegate(AvatarAttachment a) { return a.ItemID == itemID; });
|
||||||
if (index >= 0)
|
if (index >= 0)
|
||||||
{
|
{
|
||||||
|
// m_log.DebugFormat(
|
||||||
|
// "[AVATAR APPEARANCE]: Detaching attachment {0}, index {1}, point {2}",
|
||||||
|
// m_attachments[kvp.Key][index].ItemID, index, m_attachments[kvp.Key][index].AttachPoint);
|
||||||
|
|
||||||
// Remove it from the list of attachments at that attach point
|
// Remove it from the list of attachments at that attach point
|
||||||
m_attachments[kvp.Key].RemoveAt(index);
|
m_attachments[kvp.Key].RemoveAt(index);
|
||||||
|
|
||||||
|
|
|
@ -61,5 +61,5 @@ using System.Runtime.InteropServices;
|
||||||
// You can specify all the values or you can default the Revision and Build Numbers
|
// You can specify all the values or you can default the Revision and Build Numbers
|
||||||
// by using the '*' as shown below:
|
// by using the '*' as shown below:
|
||||||
|
|
||||||
[assembly : AssemblyVersion("0.7.5.*")]
|
[assembly : AssemblyVersion("0.7.6.*")]
|
||||||
[assembly : AssemblyFileVersion("0.6.5.0")]
|
|
||||||
|
|
|
@ -65,23 +65,27 @@ namespace OpenSim.Framework.Configuration.HTTP
|
||||||
byte[] buf = new byte[8192];
|
byte[] buf = new byte[8192];
|
||||||
HttpWebRequest request =
|
HttpWebRequest request =
|
||||||
(HttpWebRequest) WebRequest.Create(remoteConfigSettings.baseConfigURL + configFileName);
|
(HttpWebRequest) WebRequest.Create(remoteConfigSettings.baseConfigURL + configFileName);
|
||||||
HttpWebResponse response = (HttpWebResponse) request.GetResponse();
|
using (HttpWebResponse response = (HttpWebResponse)request.GetResponse())
|
||||||
|
|
||||||
Stream resStream = response.GetResponseStream();
|
|
||||||
|
|
||||||
string tempString = null;
|
|
||||||
int count = 0;
|
|
||||||
|
|
||||||
do
|
|
||||||
{
|
{
|
||||||
count = resStream.Read(buf, 0, buf.Length);
|
using (Stream resStream = response.GetResponseStream())
|
||||||
if (count != 0)
|
|
||||||
{
|
{
|
||||||
tempString = Util.UTF8.GetString(buf, 0, count);
|
string tempString = null;
|
||||||
sb.Append(tempString);
|
int count = 0;
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
count = resStream.Read(buf, 0, buf.Length);
|
||||||
|
if (count != 0)
|
||||||
|
{
|
||||||
|
tempString = Util.UTF8.GetString(buf, 0, count);
|
||||||
|
sb.Append(tempString);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while (count > 0);
|
||||||
|
|
||||||
|
LoadDataFromString(sb.ToString());
|
||||||
}
|
}
|
||||||
} while (count > 0);
|
}
|
||||||
LoadDataFromString(sb.ToString());
|
|
||||||
}
|
}
|
||||||
catch (WebException)
|
catch (WebException)
|
||||||
{
|
{
|
||||||
|
|
|
@ -29,5 +29,5 @@ using System.Runtime.InteropServices;
|
||||||
// Build Number
|
// Build Number
|
||||||
// Revision
|
// Revision
|
||||||
//
|
//
|
||||||
[assembly: AssemblyVersion("0.7.5.*")]
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
|
||||||
|
|
|
@ -29,5 +29,5 @@ using System.Runtime.InteropServices;
|
||||||
// Build Number
|
// Build Number
|
||||||
// Revision
|
// Revision
|
||||||
//
|
//
|
||||||
[assembly: AssemblyVersion("0.7.5.*")]
|
[assembly: AssemblyVersion("0.7.6.*")]
|
||||||
[assembly: AssemblyFileVersion("1.0.0.0")]
|
|
||||||
|
|
|
@ -55,4 +55,4 @@ using System.Runtime.InteropServices;
|
||||||
// You can specify all values by your own or you can build default build and revision
|
// You can specify all values by your own or you can build default build and revision
|
||||||
// numbers with the '*' character (the default):
|
// numbers with the '*' character (the default):
|
||||||
|
|
||||||
[assembly : AssemblyVersion("0.7.5.*")]
|
[assembly : AssemblyVersion("0.7.6.*")]
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue