From 530cc2488461a4ef68a06eaba42698fcdc09f459 Mon Sep 17 00:00:00 2001 From: Sean Dague Date: Tue, 19 Feb 2008 19:16:21 +0000 Subject: From: Michael Osias This patch implements the llSendRemoteData command and fixes mantis 552, and possibly 586. --- .../Environment/Modules/ScriptsHttpRequests.cs | 31 +- .../Region/Environment/Modules/WorldCommModule.cs | 72 ++- OpenSim/Region/Environment/Modules/XMLRPCModule.cs | 482 ++++++++++++++++----- 3 files changed, 442 insertions(+), 143 deletions(-) (limited to 'OpenSim/Region/Environment/Modules') diff --git a/OpenSim/Region/Environment/Modules/ScriptsHttpRequests.cs b/OpenSim/Region/Environment/Modules/ScriptsHttpRequests.cs index e3de13b..f38f354 100644 --- a/OpenSim/Region/Environment/Modules/ScriptsHttpRequests.cs +++ b/OpenSim/Region/Environment/Modules/ScriptsHttpRequests.cs @@ -184,13 +184,15 @@ namespace OpenSim.Region.Environment.Modules public void StopHttpRequest(uint m_localID, LLUUID m_itemID) { - lock (HttpListLock) - { - HttpRequestClass tmpReq; - if (m_pendingRequests.TryGetValue(m_itemID, out tmpReq)) + if(m_pendingRequests != null) { + lock (HttpListLock) { - tmpReq.Stop(); - m_pendingRequests.Remove(m_itemID); + HttpRequestClass tmpReq; + if (m_pendingRequests.TryGetValue(m_itemID, out tmpReq)) + { + tmpReq.Stop(); + m_pendingRequests.Remove(m_itemID); + } } } } @@ -216,7 +218,6 @@ namespace OpenSim.Region.Environment.Modules { if (tmpReq.finished) { - m_pendingRequests.Remove(luid); return tmpReq; } } @@ -224,6 +225,21 @@ namespace OpenSim.Region.Environment.Modules } return null; } + + public void RemoveCompletedRequest(LLUUID id) + { + lock (HttpListLock) + { + HttpRequestClass tmpReq; + if (m_pendingRequests.TryGetValue(id, out tmpReq)) + { + tmpReq.Stop(); + tmpReq = null; + m_pendingRequests.Remove(id); + } + } + } + } // @@ -269,6 +285,7 @@ namespace OpenSim.Region.Environment.Modules httpThread.Name = "HttpRequestThread"; httpThread.Priority = ThreadPriority.BelowNormal; httpThread.IsBackground = true; + finished = false; httpThread.Start(); } diff --git a/OpenSim/Region/Environment/Modules/WorldCommModule.cs b/OpenSim/Region/Environment/Modules/WorldCommModule.cs index a426eda..b6d4e3d 100644 --- a/OpenSim/Region/Environment/Modules/WorldCommModule.cs +++ b/OpenSim/Region/Environment/Modules/WorldCommModule.cs @@ -84,8 +84,8 @@ namespace OpenSim.Region.Environment.Modules m_scene = scene; m_scene.RegisterModuleInterface(this); m_listenerManager = new ListenerManager(); - m_pending = new Queue(); m_scene.EventManager.OnNewClient += NewClient; + m_pending = new Queue(); } public void PostInitialise() @@ -139,9 +139,12 @@ namespace OpenSim.Region.Environment.Modules public void DeleteListener(LLUUID itemID) { - lock (ListLock) + if (m_listenerManager != null) { - m_listenerManager.DeleteListener(itemID); + lock (ListLock) + { + m_listenerManager.DeleteListener(itemID); + } } } @@ -167,14 +170,14 @@ namespace OpenSim.Region.Environment.Modules // If they are in proximity, then if they are // listeners, if so add them to the pending queue - foreach (LLUUID eb in m_scene.Entities.Keys) + foreach (ListenerInfo li in m_listenerManager.GetListeners()) { EntityBase sPart; - m_scene.Entities.TryGetValue(eb, out sPart); + m_scene.Entities.TryGetValue(li.GetHostID(), out sPart); // Dont process if this message is from itself! - if (eb.ToString().Equals(sourceItemID) || + if (li.GetHostID().ToString().Equals(sourceItemID) || sPart.UUID.ToString().Equals(sourceItemID)) continue; @@ -196,7 +199,10 @@ namespace OpenSim.Region.Environment.Modules ); if (isListener != null) { - m_pending.Enqueue(isListener); + lock (CommListLock) + { + m_pending.Enqueue(isListener); + } } } break; @@ -210,7 +216,10 @@ namespace OpenSim.Region.Environment.Modules ); if (isListener != null) { - m_pending.Enqueue(isListener); + lock (CommListLock) + { + m_pending.Enqueue(isListener); + } } } break; @@ -223,14 +232,17 @@ namespace OpenSim.Region.Environment.Modules ); if (isListener != null) { - m_pending.Enqueue(isListener); + lock (CommListLock) + { + m_pending.Enqueue(isListener); + } } } break; case ChatTypeEnum.Broadcast: ListenerInfo isListen = - m_listenerManager.IsListenerMatch(sourceItemID, eb, channel, name, msg); + m_listenerManager.IsListenerMatch(sourceItemID, li.GetItemID(), channel, name, msg); if (isListen != null) { ListenerInfo isListener = m_listenerManager.IsListenerMatch( @@ -238,19 +250,24 @@ namespace OpenSim.Region.Environment.Modules ); if (isListener != null) { - m_pending.Enqueue(isListener); + lock (CommListLock) + { + m_pending.Enqueue(isListener); + } } } break; } } - ; } } public bool HasMessages() { - return (m_pending.Count > 0); + if (m_pending != null) + return (m_pending.Count > 0); + else + return false; } public ListenerInfo GetNextMessage() @@ -264,6 +281,17 @@ namespace OpenSim.Region.Environment.Modules return li; } + + public uint PeekNextMessageLocalID() + { + return m_pending.Peek().GetLocalID(); + } + + public LLUUID PeekNextMessageItemID() + { + return m_pending.Peek().GetItemID(); + } + } // hostID: the ID of the ScenePart @@ -280,8 +308,7 @@ namespace OpenSim.Region.Environment.Modules m_listeners = new Dictionary(); } - public int AddListener(uint localID, LLUUID itemID, LLUUID hostID, int channel, string name, string id, - string msg) + public int AddListener(uint localID, LLUUID itemID, LLUUID hostID, int channel, string name, string id, string msg) { if (m_listeners.Count < m_MaxListeners) { @@ -410,6 +437,11 @@ namespace OpenSim.Region.Environment.Modules } return null; } + + public Dictionary.ValueCollection GetListeners() + { + return m_listeners.Values; + } } public class ListenerInfo @@ -425,21 +457,20 @@ namespace OpenSim.Region.Environment.Modules private string m_message; // The message private bool m_active; // Listener is active or not - public ListenerInfo(uint localID, int handle, LLUUID ItemID, LLUUID hostID, int channel, string name, LLUUID id, - string message) + public ListenerInfo(uint localID, int handle, LLUUID ItemID, LLUUID hostID, int channel, string name, LLUUID id, string message) { Initialise(localID, handle, ItemID, hostID, channel, name, id, message); } public ListenerInfo(uint localID, int handle, LLUUID ItemID, LLUUID hostID, int channel, string name, LLUUID id, - string message, LLUUID sourceItemID) + string message, LLUUID sourceItemID) { Initialise(localID, handle, ItemID, hostID, channel, name, id, message); m_sourceItemID = sourceItemID; } private void Initialise(uint localID, int handle, LLUUID ItemID, LLUUID hostID, int channel, string name, - LLUUID id, string message) + LLUUID id, string message) { m_handle = handle; m_channel = channel; @@ -511,5 +542,6 @@ namespace OpenSim.Region.Environment.Modules { return m_id; } + } -} \ No newline at end of file +} diff --git a/OpenSim/Region/Environment/Modules/XMLRPCModule.cs b/OpenSim/Region/Environment/Modules/XMLRPCModule.cs index edd1df8..48a4c7c 100644 --- a/OpenSim/Region/Environment/Modules/XMLRPCModule.cs +++ b/OpenSim/Region/Environment/Modules/XMLRPCModule.cs @@ -78,19 +78,20 @@ namespace OpenSim.Region.Environment.Modules private static readonly log4net.ILog m_log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); private Scene m_scene; - private Queue rpcQueue = new Queue(); private object XMLRPCListLock = new object(); private string m_name = "XMLRPCModule"; private int RemoteReplyScriptWait = 300; - private int RemoteReplyScriptTimeout = 900; + private int RemoteReplyScriptTimeout = 9000; private int m_remoteDataPort = 0; private List m_scenes = new List(); // private Dictionary m_openChannels; - // - private Dictionary m_pendingResponse; + private Dictionary m_rpcPending; + private Dictionary m_rpcPendingResponses; + + private Dictionary m_pendingSRDResponses; public void Initialise(Scene scene, IConfigSource config) { @@ -115,13 +116,15 @@ namespace OpenSim.Region.Environment.Modules if (IsEnabled()) { m_openChannels = new Dictionary(); - m_pendingResponse = new Dictionary(); + m_rpcPending = new Dictionary(); + m_rpcPendingResponses = new Dictionary(); + m_pendingSRDResponses = new Dictionary(); // Start http server // Attach xmlrpc handlers m_log.Info("[REMOTE_DATA]: " + "Starting XMLRPC Server on port " + m_remoteDataPort + " for llRemoteData commands."); - BaseHttpServer httpServer = new BaseHttpServer((uint) m_remoteDataPort); + BaseHttpServer httpServer = new BaseHttpServer((uint)m_remoteDataPort); httpServer.AddXmlRPCHandler("llRemoteData", XmlRpcRemoteData); httpServer.Start(); } @@ -188,18 +191,28 @@ namespace OpenSim.Region.Environment.Modules return channel; } - public void DeleteChannel(LLUUID itemID) + // Delete channels based on itemID + // for when a script is deleted + public void DeleteChannels(LLUUID itemID) { - foreach (RPCChannelInfo li in m_openChannels.Values) + if (m_openChannels != null) { + ArrayList tmp = new ArrayList(); - if (li.GetItemID().Equals(itemID)) + lock (XMLRPCListLock) { + foreach (RPCChannelInfo li in m_openChannels.Values) + { + if (li.GetItemID().Equals(itemID)) + { + tmp.Add(itemID); + } + } - m_openChannels.Remove(li.GetChannelID()); - return; - + System.Collections.IEnumerator tmpEnumerator = tmp.GetEnumerator(); + while ( tmpEnumerator.MoveNext() ) + m_openChannels.Remove((LLUUID)tmpEnumerator.Current); } } @@ -218,15 +231,12 @@ namespace OpenSim.Region.Environment.Modules RPCRequestInfo rpcInfo; LLUUID message_key = new LLUUID(message_id); - if (m_pendingResponse.TryGetValue(message_key, out rpcInfo)) + if (m_rpcPendingResponses.TryGetValue(message_key, out rpcInfo)) { - rpcInfo.SetRetval(sdata); + rpcInfo.SetStrRetval(sdata); + rpcInfo.SetIntRetval(idata); rpcInfo.SetProcessed(true); - - lock (XMLRPCListLock) - { - m_pendingResponse.Remove(message_key); - } + m_rpcPendingResponses.Remove(message_key); } } @@ -248,18 +258,18 @@ namespace OpenSim.Region.Environment.Modules { XmlRpcResponse response = new XmlRpcResponse(); - Hashtable requestData = (Hashtable) request.Params[0]; + Hashtable requestData = (Hashtable)request.Params[0]; bool GoodXML = (requestData.Contains("Channel") && requestData.Contains("IntValue") && requestData.Contains("StringValue")); if (GoodXML) { - LLUUID channel = new LLUUID((string) requestData["Channel"]); + LLUUID channel = new LLUUID((string)requestData["Channel"]); RPCChannelInfo rpcChanInfo; if (m_openChannels.TryGetValue(channel, out rpcChanInfo)) { - string intVal = (string) requestData["IntValue"]; - string strVal = (string) requestData["StringValue"]; + string intVal = (string)requestData["IntValue"]; + string strVal = (string)requestData["StringValue"]; RPCRequestInfo rpcInfo; @@ -268,7 +278,7 @@ namespace OpenSim.Region.Environment.Modules rpcInfo = new RPCRequestInfo(rpcChanInfo.GetLocalID(), rpcChanInfo.GetItemID(), channel, strVal, intVal); - rpcQueue.Enqueue(rpcInfo); + m_rpcPending.Add(rpcInfo.GetMessageID(), rpcInfo); } int timeoutCtr = 0; @@ -280,16 +290,20 @@ namespace OpenSim.Region.Environment.Modules } if (rpcInfo.IsProcessed()) { - response.Value = rpcInfo.GetRetval(); + Hashtable param = new Hashtable(); + param["StringValue"] = rpcInfo.GetStrRetval(); + param["IntValue"] = Convert.ToString(rpcInfo.GetIntRetval()); + + ArrayList parameters = new ArrayList(); + parameters.Add(param); + + response.Value = parameters; rpcInfo = null; } else { response.SetFault(-1, "Script timeout"); - lock (XMLRPCListLock) - { - m_pendingResponse.Remove(rpcInfo.GetMessageID()); - } + rpcInfo = null; } } else @@ -303,129 +317,365 @@ namespace OpenSim.Region.Environment.Modules public bool hasRequests() { - return (rpcQueue.Count > 0); - } - - public RPCRequestInfo GetNextRequest() - { lock (XMLRPCListLock) { - RPCRequestInfo rpcInfo = rpcQueue.Dequeue(); - m_pendingResponse.Add(rpcInfo.GetMessageID(), rpcInfo); - return rpcInfo; + if (m_rpcPending != null) + return (m_rpcPending.Count > 0); + else + return false; } } - } - - /************************************************************** - * - * Class RPCRequestInfo - * - * Holds details about incoming requests until they are picked - * from the queue by LSLLongCmdHandler - * ***********************************************************/ - public class RPCRequestInfo - { - private string m_StrVal; - private string m_IntVal; - private bool m_processed; - private string m_resp; - private uint m_localID; - private LLUUID m_ItemID; - private LLUUID m_MessageID; - private LLUUID m_ChannelKey; - - public RPCRequestInfo(uint localID, LLUUID itemID, LLUUID channelKey, string strVal, string intVal) + public RPCRequestInfo GetNextCompletedRequest() { - m_localID = localID; - m_StrVal = strVal; - m_IntVal = intVal; - m_ItemID = itemID; - m_ChannelKey = channelKey; - m_MessageID = LLUUID.Random(); - m_processed = false; - m_resp = String.Empty; - } + if (m_rpcPending != null) + { + lock (XMLRPCListLock) + { + foreach (LLUUID luid in m_rpcPending.Keys) + { + RPCRequestInfo tmpReq; - public bool IsProcessed() - { - return m_processed; - } + if (m_rpcPending.TryGetValue(luid, out tmpReq)) + { - public LLUUID GetChannelKey() - { - return m_ChannelKey; + if (!tmpReq.IsProcessed()) return tmpReq; + } + } + } + } + return null; } - public void SetProcessed(bool processed) + public void RemoveCompletedRequest(LLUUID id) { - m_processed = processed; + lock (XMLRPCListLock) + { + RPCRequestInfo tmp; + if (m_rpcPending.TryGetValue(id, out tmp)) + { + m_rpcPending.Remove(id); + m_rpcPendingResponses.Add(id, tmp); + } + else + { + Console.WriteLine("UNABLE TO REMOVE COMPLETED REQUEST"); + } + } } - public void SetRetval(string resp) + public LLUUID SendRemoteData(uint localID, LLUUID itemID, string channel, string dest, int idata, string sdata) { - m_resp = resp; - } - public string GetRetval() - { - return m_resp; - } + SendRemoteDataRequest req = new SendRemoteDataRequest( + localID, itemID, channel, dest, idata, sdata + ); + m_pendingSRDResponses.Add(req.GetReqID(), req); + return req.process(); - public uint GetLocalID() - { - return m_localID; } - public LLUUID GetItemID() + public SendRemoteDataRequest GetNextCompletedSRDRequest() { - return m_ItemID; - } + if (m_pendingSRDResponses != null) + { + lock (XMLRPCListLock) + { + foreach (LLUUID luid in m_pendingSRDResponses.Keys) + { + SendRemoteDataRequest tmpReq; - public string GetStrVal() - { - return m_StrVal; + if (m_pendingSRDResponses.TryGetValue(luid, out tmpReq)) + { + if (tmpReq.finished) + return tmpReq; + } + } + } + } + return null; } - public int GetIntValue() + public void RemoveCompletedSRDRequest(LLUUID id) { - return int.Parse(m_IntVal); + lock (XMLRPCListLock) + { + SendRemoteDataRequest tmpReq; + if (m_pendingSRDResponses.TryGetValue(id, out tmpReq)) + { + m_pendingSRDResponses.Remove(id); + } + } } - public LLUUID GetMessageID() + public void CancelSRDRequests(LLUUID itemID) { - return m_MessageID; + if (m_pendingSRDResponses != null) + { + lock (XMLRPCListLock) + { + foreach (SendRemoteDataRequest li in m_pendingSRDResponses.Values) + { + if (li.m_itemID.Equals(itemID)) + m_pendingSRDResponses.Remove(li.GetReqID()); + } + } + } } } + /************************************************************** + * + * Class RPCRequestInfo + * + * Holds details about incoming requests until they are picked + * from the queue by LSLLongCmdHandler + * ***********************************************************/ - public class RPCChannelInfo - { - private LLUUID m_itemID; - private uint m_localID; - private LLUUID m_ChannelKey; - - public RPCChannelInfo(uint localID, LLUUID itemID, LLUUID channelID) + public class RPCRequestInfo { - m_ChannelKey = channelID; - m_localID = localID; - m_itemID = itemID; - } + private string m_StrVal; + private string m_IntVal; + private bool m_processed; + private string m_respStr; + private int m_respInt; + private uint m_localID; + private LLUUID m_ItemID; + private LLUUID m_MessageID; + private LLUUID m_ChannelKey; + + public RPCRequestInfo(uint localID, LLUUID itemID, LLUUID channelKey, string strVal, string intVal) + { + m_localID = localID; + m_StrVal = strVal; + m_IntVal = intVal; + m_ItemID = itemID; + m_ChannelKey = channelKey; + m_MessageID = LLUUID.Random(); + m_processed = false; + m_respStr = String.Empty; + m_respInt = 0; + } - public LLUUID GetItemID() - { - return m_itemID; + public bool IsProcessed() + { + return m_processed; + } + + public LLUUID GetChannelKey() + { + return m_ChannelKey; + } + + public void SetProcessed(bool processed) + { + m_processed = processed; + } + + public void SetStrRetval(string resp) + { + m_respStr = resp; + } + + public string GetStrRetval() + { + return m_respStr; + } + public void SetIntRetval(int resp) + { + m_respInt = resp; + } + + public int GetIntRetval() + { + return m_respInt; + } + public uint GetLocalID() + { + return m_localID; + } + + public LLUUID GetItemID() + { + return m_ItemID; + } + + public string GetStrVal() + { + return m_StrVal; + } + + public int GetIntValue() + { + return int.Parse(m_IntVal); + } + + public LLUUID GetMessageID() + { + return m_MessageID; + } } - public LLUUID GetChannelID() + public class RPCChannelInfo { - return m_ChannelKey; + private LLUUID m_itemID; + private uint m_localID; + private LLUUID m_ChannelKey; + + public RPCChannelInfo(uint localID, LLUUID itemID, LLUUID channelID) + { + m_ChannelKey = channelID; + m_localID = localID; + m_itemID = itemID; + } + + public LLUUID GetItemID() + { + return m_itemID; + } + + public LLUUID GetChannelID() + { + return m_ChannelKey; + } + + public uint GetLocalID() + { + return m_localID; + } + } - public uint GetLocalID() + public class SendRemoteDataRequest { - return m_localID; + + public LLUUID reqID; + public string destURL; + public string channel; + public string sdata; + public int idata; + public bool finished; + public string response_sdata; + public int response_idata; + public XmlRpcRequest request; + private Thread httpThread; + public LLUUID m_itemID; + public uint m_localID; + private static readonly log4net.ILog m_log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); + + public SendRemoteDataRequest(uint localID, LLUUID itemID, string channel, string dest, int idata, string sdata) + { + + this.channel = channel; + this.destURL = dest; + this.idata = idata; + this.sdata = sdata; + m_itemID = itemID; + m_localID = localID; + + reqID = LLUUID.Random(); + + } + + public LLUUID process() + { + httpThread = new Thread(SendRequest); + httpThread.Name = "HttpRequestThread"; + httpThread.Priority = ThreadPriority.BelowNormal; + httpThread.IsBackground = true; + finished = false; + httpThread.Start(); + + return reqID; + + } + + /* + * TODO: More work on the response codes. Right now + * returning 200 for success or 499 for exception + */ + + public void SendRequest() + { + + Hashtable param = new Hashtable(); + + // Check if channel is an LLUUID + // if not, use as method name + LLUUID parseUID; + string mName = "llRemoteData"; + if( (channel != null) && (channel != "") ) + if( !LLUUID.TryParse(channel, out parseUID) ) + mName = channel; + else + param["Channel"] = channel; + + param["StringValue"] = sdata; + param["IntValue"] = Convert.ToString(idata); + + ArrayList parameters = new ArrayList(); + parameters.Add(param); + XmlRpcRequest req = new XmlRpcRequest(mName, parameters); + try + { + XmlRpcResponse resp = req.Send(destURL, 30000); + if (resp != null) + { + Hashtable respParms; + if(resp.Value.GetType().Equals(Type.GetType("System.Collections.Hashtable"))) { + respParms = (Hashtable)resp.Value; + } + else { + ArrayList respData = (ArrayList)resp.Value; + respParms = (Hashtable)respData[0]; + } + if (respParms != null) + { + if (respParms.Contains("StringValue")) + { + sdata = (string)respParms["StringValue"]; + } + if (respParms.Contains("IntValue")) + { + idata = Convert.ToInt32((string)respParms["IntValue"]); + } + if (respParms.Contains("faultString")) + { + sdata = (string)respParms["faultString"]; + } + if (respParms.Contains("faultCode")) + { + idata = Convert.ToInt32(respParms["faultCode"]); + } + } + } + } + catch (System.Net.WebException we) + { + sdata = we.Message; + m_log.Warn("[SendRemoteDataRequest]: Request failed"); + m_log.Warn(we.StackTrace); + } + + finished = true; + + } + + public void Stop() + { + try + { + httpThread.Abort(); + } + catch (Exception) + { + } + } + + public LLUUID GetReqID() + { + return reqID; + } + } - } -} +} \ No newline at end of file -- cgit v1.1