/* * 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.Region.ScriptEngine.Shared.Api.Plugins { public class Dataserver { private ObjectJobEngine m_WorkPool; public AsyncCommandManager m_CmdManager; public int DataserverRequestsCount { get { lock (DataserverRequests) return DataserverRequests.Count; } } private Dictionary DataserverRequests = new Dictionary(); public Dataserver(AsyncCommandManager CmdManager) { m_CmdManager = CmdManager; m_WorkPool = new ObjectJobEngine(ProcessActions, "ScriptDataServer", 1000, 4); } private class DataserverRequest { public uint localID; public UUID itemID; public UUID ID; public string handle; public DateTime startTime; public Action action; } public string RequestWithImediatePost(uint localID, UUID itemID, string reply) { string ID = UUID.Random().ToString(); m_CmdManager.m_ScriptEngine.PostObjectEvent(localID, new EventParams("dataserver", new Object[] { new LSL_Types.LSLString(ID), new LSL_Types.LSLString(reply)}, new DetectParams[0])); return ID; } //legacy public UUID RegisterRequest(uint localID, UUID itemID, string identifier) { lock (DataserverRequests) { if (DataserverRequests.ContainsKey(identifier)) return UUID.Zero; DataserverRequest ds = new DataserverRequest() { localID = localID, itemID = itemID, ID = UUID.Random(), handle = identifier, startTime = DateTime.UtcNow, action = null }; DataserverRequests[identifier] = ds; return ds.ID; } } // action, if provided, is executed async // its code pattern should be: //Action act = eventID => //{ // need operations to get reply string // m_AsyncCommands.DataserverPlugin.DataserverReply(eventID, reply); //} // eventID is the event id, provided by this on Invoque // see ProcessActions below // temporary don't use public UUID RegisterRequest(uint localID, UUID itemID, string identifier, Action action) { lock (DataserverRequests) { if (DataserverRequests.ContainsKey(identifier)) return UUID.Zero; DataserverRequest ds = new DataserverRequest() { localID = localID, itemID = itemID, ID = UUID.Random(), handle = identifier, startTime = DateTime.UtcNow, action = action }; DataserverRequests[identifier] = ds; if (action != null) m_WorkPool.Enqueue(identifier); return ds.ID; } } public UUID RegisterRequest(uint localID, UUID itemID, Action action) { lock (DataserverRequests) { string identifier = UUID.Random().ToString(); DataserverRequest ds = new DataserverRequest() { localID = localID, itemID = itemID, ID = UUID.Random(), handle = identifier, startTime = DateTime.MaxValue, action = action }; DataserverRequests[identifier] = ds; if (action != null) m_WorkPool.Enqueue(identifier); return ds.ID; } } public void ProcessActions(object st) { string id = st as string; if(string.IsNullOrEmpty(id)) return; DataserverRequest ds = null; lock (DataserverRequests) { if (!DataserverRequests.TryGetValue(id, out ds)) return; } if (ds == null || ds.action == null) return; try { ds.action.Invoke(ds.handle); } catch { } ds.action = null; lock (DataserverRequests) { DataserverRequests.Remove(id); } } //legacy ? public void DataserverReply(string identifier, string reply) { DataserverRequest ds; lock (DataserverRequests) { if (!DataserverRequests.TryGetValue(identifier, out ds)) return; DataserverRequests.Remove(identifier); } m_CmdManager.m_ScriptEngine.PostObjectEvent(ds.localID, new EventParams("dataserver", new Object[] { new LSL_Types.LSLString(ds.ID.ToString()), new LSL_Types.LSLString(reply) }, new DetectParams[0])); } public void RemoveEvents(uint localID, UUID itemID) { lock (DataserverRequests) { List toremove = new List(DataserverRequests.Count); foreach (DataserverRequest ds in DataserverRequests.Values) { if (ds.itemID.Equals(itemID)) toremove.Add(ds.handle); } foreach (string s in toremove) { DataserverRequests.Remove(s); } } } public void ExpireRequests() { lock (DataserverRequests) { List toremove = new List(DataserverRequests.Count); DateTime expirebase = DateTime.UtcNow.AddSeconds(-30); foreach (DataserverRequest ds in DataserverRequests.Values) { if (ds.action == null && ds.startTime < expirebase) toremove.Add(ds.handle); } foreach (string s in toremove) { DataserverRequests.Remove(s); } } } } }