/* * 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.IO; using System.Reflection; using System.Text; using System.Xml; using System.Xml.Schema; using System.Xml.Serialization; using log4net; using OpenMetaverse; using OpenMetaverse.StructuredData; namespace OpenSim.Framework { /// /// This class stores and retrieves dynamic attributes. /// /// /// Modules that want to use dynamic attributes need to do so in a private data store /// which is accessed using a unique name. DAMap provides access to the data stores, /// each of which is an OSDMap. Modules are free to store any type of data they want /// within their data store. However, avoid storing large amounts of data because that /// would slow down database access. /// public class DAMap : IXmlSerializable { // private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private static readonly int MIN_NAMESPACE_LENGTH = 4; private OSDMap m_map = new OSDMap(); // WARNING: this is temporary for experimentation only, it will be removed!!!! public OSDMap TopLevelMap { get { return m_map; } set { m_map = value; } } public XmlSchema GetSchema() { return null; } public static DAMap FromXml(string rawXml) { DAMap map = new DAMap(); map.ReadXml(rawXml); return map; } public void ReadXml(XmlReader reader) { ReadXml(reader.ReadInnerXml()); } public void ReadXml(string rawXml) { // System.Console.WriteLine("Trying to deserialize [{0}]", rawXml); lock (this) { m_map = (OSDMap)OSDParser.DeserializeLLSDXml(rawXml); SanitiseMap(this); } } public void WriteXml(XmlWriter writer) { writer.WriteRaw(ToXml()); } public string ToXml() { lock (this) return OSDParser.SerializeLLSDXmlString(m_map); } public void CopyFrom(DAMap other) { // Deep copy string data = null; lock (other) { if (other.CountNamespaces > 0) { data = OSDParser.SerializeLLSDXmlString(other.m_map); } } lock (this) { if (data == null) Clear(); else m_map = (OSDMap)OSDParser.DeserializeLLSDXml(data); } } /// /// Sanitise the map to remove any namespaces or stores that are not OSDMap. /// /// /// public static void SanitiseMap(DAMap daMap) { List keysToRemove = null; // Hard-coded special case that needs to be removed in the future. Normally, modules themselves should // handle reading data from old locations bool osMaterialsMigrationRequired = false; OSDMap namespacesMap = daMap.m_map; foreach (string key in namespacesMap.Keys) { // Console.WriteLine("Processing ns {0}", key); if (!(namespacesMap[key] is OSDMap)) { if (keysToRemove == null) keysToRemove = new List(); keysToRemove.Add(key); } else if (key == "OS:Materials") { osMaterialsMigrationRequired = true; } } if (keysToRemove != null) { foreach (string key in keysToRemove) { // Console.WriteLine ("Removing bad ns {0}", key); namespacesMap.Remove(key); } } // Hard-coded special case that needs to be removed in the future. Normally, modules themselves should // handle reading data from old locations if (osMaterialsMigrationRequired) daMap.SetStore("OpenSim", "Materials", (OSDMap)namespacesMap["OS:Materials"]); foreach (OSD nsOsd in namespacesMap.Values) { OSDMap nsOsdMap = (OSDMap)nsOsd; keysToRemove = null; foreach (string key in nsOsdMap.Keys) { if (!(nsOsdMap[key] is OSDMap)) { if (keysToRemove == null) keysToRemove = new List(); keysToRemove.Add(key); } } if (keysToRemove != null) foreach (string key in keysToRemove) nsOsdMap.Remove(key); } } /// /// Get the number of namespaces /// public int CountNamespaces { get { lock (this) { return m_map.Count; } } } /// /// Get the number of stores. /// public int CountStores { get { int count = 0; lock (this) { foreach (OSD osdNamespace in m_map) { count += ((OSDMap)osdNamespace).Count; } } return count; } } /// /// Retrieve a Dynamic Attribute store /// /// namespace for the store - use "OpenSim" for in-core modules /// name of the store within the namespace /// an OSDMap representing the stored data, or null if not found public OSDMap GetStore(string ns, string storeName) { OSD namespaceOsd; lock (this) { if (m_map.TryGetValue(ns, out namespaceOsd)) { OSD store; if (((OSDMap)namespaceOsd).TryGetValue(storeName, out store)) return (OSDMap)store; } } return null; } /// /// Saves a Dynamic attribute store /// /// namespace for the store - use "OpenSim" for in-core modules /// name of the store within the namespace /// an OSDMap representing the data to store public void SetStore(string ns, string storeName, OSDMap store) { ValidateNamespace(ns); OSDMap nsMap; lock (this) { if (!m_map.ContainsKey(ns)) { nsMap = new OSDMap(); m_map[ns] = nsMap; } nsMap = (OSDMap)m_map[ns]; // m_log.DebugFormat("[DA MAP]: Setting store to {0}:{1}", ns, storeName); nsMap[storeName] = store; } } /// /// Validate the key used for storing separate data stores. /// /// public static void ValidateNamespace(string ns) { if (ns.Length < MIN_NAMESPACE_LENGTH) throw new Exception("Minimum namespace length is " + MIN_NAMESPACE_LENGTH); } public bool ContainsStore(string ns, string storeName) { OSD namespaceOsd; lock (this) { if (m_map.TryGetValue(ns, out namespaceOsd)) { return ((OSDMap)namespaceOsd).ContainsKey(storeName); } } return false; } public bool TryGetStore(string ns, string storeName, out OSDMap store) { OSD namespaceOsd; lock (this) { if (m_map.TryGetValue(ns, out namespaceOsd)) { OSD storeOsd; bool result = ((OSDMap)namespaceOsd).TryGetValue(storeName, out storeOsd); store = (OSDMap)storeOsd; return result; } } store = null; return false; } public void Clear() { lock (this) m_map.Clear(); } public bool RemoveStore(string ns, string storeName) { OSD namespaceOsd; lock (this) { if (m_map.TryGetValue(ns, out namespaceOsd)) { OSDMap namespaceOsdMap = (OSDMap)namespaceOsd; namespaceOsdMap.Remove(storeName); // Don't keep empty namespaces around if (namespaceOsdMap.Count <= 0) m_map.Remove(ns); } } return false; } } }