/* * 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.Net; using System.Reflection; using System.Text; using log4net; using Nini.Config; using Mono.Addins; using OpenMetaverse; using OpenMetaverse.StructuredData; using OpenSim.Framework; using OpenSim.Framework.Servers.HttpServer; using OpenSim.Region.Framework.Interfaces; using OpenSim.Region.Framework.Scenes; // using OpenSim.Services.Interfaces; using Caps = OpenSim.Framework.Capabilities.Caps; namespace OpenSim.Region.ClientStack.Linden { /// /// SimulatorFeatures capability. /// /// /// This is required for uploading Mesh. /// Since is accepts an open-ended response, we also send more information /// for viewers that care to interpret it. /// /// NOTE: Part of this code was adapted from the Aurora project, specifically /// the normal part of the response in the capability handler. /// [Extension(Path = "/OpenSim/RegionModules", NodeName = "RegionModule", Id = "SimulatorFeaturesModule")] public class SimulatorFeaturesModule : INonSharedRegionModule, ISimulatorFeaturesModule { private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); public event SimulatorFeaturesRequestDelegate OnSimulatorFeaturesRequest; private Scene m_scene; /// /// Simulator features /// private OSDMap m_features = new OSDMap(); private bool m_ExportSupported = false; private bool m_doScriptSyntax; static private object m_scriptSyntaxLock = new object(); static private UUID m_scriptSyntaxID = UUID.Zero; static private byte[] m_scriptSyntaxXML = null; #region ISharedRegionModule Members public void Initialise(IConfigSource source) { IConfig config = source.Configs["SimulatorFeatures"]; m_doScriptSyntax = true; if (config != null) { m_ExportSupported = config.GetBoolean("ExportSupported", m_ExportSupported); m_doScriptSyntax = config.GetBoolean("ScriptSyntax", m_doScriptSyntax); } ReadScriptSyntax(); AddDefaultFeatures(); } public void AddRegion(Scene s) { m_scene = s; m_scene.EventManager.OnRegisterCaps += RegisterCaps; m_scene.RegisterModuleInterface(this); } public void RemoveRegion(Scene s) { m_scene.EventManager.OnRegisterCaps -= RegisterCaps; } public void RegionLoaded(Scene s) { GetGridExtraFeatures(s); } public void Close() { } public string Name { get { return "SimulatorFeaturesModule"; } } public Type ReplaceableInterface { get { return null; } } #endregion /// /// Add default features /// /// /// TODO: These should be added from other modules rather than hardcoded. /// private void AddDefaultFeatures() { lock (m_features) { m_features["MeshRezEnabled"] = true; m_features["MeshUploadEnabled"] = true; m_features["MeshXferEnabled"] = true; m_features["BakesOnMeshEnabled"] = true; m_features["PhysicsMaterialsEnabled"] = true; OSDMap typesMap = new OSDMap(); typesMap["convex"] = true; typesMap["none"] = true; typesMap["prim"] = true; m_features["PhysicsShapeTypes"] = typesMap; if(m_doScriptSyntax && m_scriptSyntaxID != UUID.Zero) m_features["LSLSyntaxId"] = OSD.FromUUID(m_scriptSyntaxID); OSDMap meshAnim = new OSDMap(); meshAnim["AnimatedObjectMaxTris"] = OSD.FromInteger(150000); meshAnim["MaxAgentAnimatedObjectAttachments"] = OSD.FromInteger(2); m_features["AnimatedObjects"] = meshAnim; m_features["MaxAgentAttachments"] = OSD.FromInteger(Constants.MaxAgentAttachments); m_features["MaxAgentGroupsBasic"] = OSD.FromInteger(Constants.MaxAgentGroups); m_features["MaxAgentGroupsPremium"] = OSD.FromInteger(Constants.MaxAgentGroups); // Extra information for viewers that want to use it // TODO: Take these out of here into their respective modules, like map-server-url OSDMap extrasMap; if(m_features.ContainsKey("OpenSimExtras")) { extrasMap = (OSDMap)m_features["OpenSimExtras"]; } else extrasMap = new OSDMap(); extrasMap["AvatarSkeleton"] = true; extrasMap["AnimationSet"] = true; extrasMap["MinSimHeight"] = Constants.MinSimulationHeight; extrasMap["MaxSimHeight"] = Constants.MaxSimulationHeight; extrasMap["MinHeightmap"] = Constants.MinTerrainHeightmap; extrasMap["MaxHeightmap"] = Constants.MaxTerrainHeightmap; if (m_ExportSupported) extrasMap["ExportSupported"] = true; if (extrasMap.Count > 0) m_features["OpenSimExtras"] = extrasMap; } } public void RegisterCaps(UUID agentID, Caps caps) { caps.RegisterSimpleHandler("SimulatorFeatures", new SimpleStreamHandler("/" + UUID.Random(), delegate (IOSHttpRequest request, IOSHttpResponse response) { HandleSimulatorFeaturesRequest(request, response, agentID); })); if (m_doScriptSyntax && m_scriptSyntaxID != UUID.Zero && m_scriptSyntaxXML != null) { caps.RegisterSimpleHandler("LSLSyntax", new SimpleStreamHandler("/" + UUID.Random(), HandleSyntaxRequest)); } } public void AddFeature(string name, OSD value) { lock (m_features) m_features[name] = value; } public void AddOpenSimExtraFeature(string name, OSD value) { lock (m_features) { OSDMap extrasMap; if (m_features.TryGetValue("OpenSimExtras", out OSD extra)) extrasMap = extra as OSDMap; else { extrasMap = new OSDMap(); } extrasMap[name] = value; m_features["OpenSimExtras"] = extrasMap; } } public bool RemoveFeature(string name) { lock (m_features) return m_features.Remove(name); } public bool TryGetFeature(string name, out OSD value) { lock (m_features) return m_features.TryGetValue(name, out value); } public bool TryGetOpenSimExtraFeature(string name, out OSD value) { value = null; lock (m_features) { if (!m_features.TryGetValue("OpenSimExtras", out OSD extra)) return false; if(!(extra is OSDMap)) return false; return (extra as OSDMap).TryGetValue(name, out value); } } public OSDMap GetFeatures() { lock (m_features) return new OSDMap(m_features); } private OSDMap DeepCopy() { // This isn't the cheapest way of doing this but the rate // of occurrence is low (on sim entry only) and it's a sure // way to get a true deep copy. OSD copy = OSDParser.DeserializeLLSDXml(OSDParser.SerializeLLSDXmlString(m_features)); return (OSDMap)copy; } private void HandleSimulatorFeaturesRequest(IOSHttpRequest request, IOSHttpResponse response, UUID agentID) { // m_log.DebugFormat("[SIMULATOR FEATURES MODULE]: SimulatorFeatures request"); if (request.HttpMethod != "GET") { response.StatusCode = (int)HttpStatusCode.NotFound; return; } ScenePresence sp = m_scene.GetScenePresence(agentID); if (sp == null) { response.StatusCode = (int)HttpStatusCode.ServiceUnavailable; response.AddHeader("Retry-After", "5"); return; } OSDMap copy = DeepCopy(); // Let's add the agentID to the destination guide, if it is expecting that. if (copy.ContainsKey("OpenSimExtras") && ((OSDMap)(copy["OpenSimExtras"])).ContainsKey("destination-guide-url")) ((OSDMap)copy["OpenSimExtras"])["destination-guide-url"] = Replace(((OSDMap)copy["OpenSimExtras"])["destination-guide-url"], "[USERID]", agentID.ToString()); OnSimulatorFeaturesRequest?.Invoke(agentID, ref copy); //Send back data response.RawBuffer = Util.UTF8.GetBytes(OSDParser.SerializeLLSDXmlString(copy)); response.StatusCode = (int)HttpStatusCode.OK; } private void HandleSyntaxRequest(IOSHttpRequest request, IOSHttpResponse response) { if (request.HttpMethod != "GET" || m_scriptSyntaxXML == null) { response.StatusCode = (int)HttpStatusCode.NotFound; return; } response.RawBuffer = m_scriptSyntaxXML; response.StatusCode = (int)HttpStatusCode.OK; } /// /// Gets the grid extra features. /// /// /// The URI Robust uses to handle the get_extra_features request /// private void GetGridExtraFeatures(Scene scene) { Dictionary extraFeatures = scene.GridService.GetExtraFeatures(); if (extraFeatures.ContainsKey("Result") && extraFeatures["Result"] != null && extraFeatures["Result"].ToString() == "Failure") { m_log.WarnFormat("[SIMULATOR FEATURES MODULE]: Unable to retrieve grid-wide features"); return; } GridInfo ginfo = scene.SceneGridInfo; lock (m_features) { OSDMap extrasMap; if (m_features.TryGetValue("OpenSimExtras", out OSD extra)) extrasMap = extra as OSDMap; else { extrasMap = new OSDMap(); } foreach (string key in extraFeatures.Keys) { string val = (string)extraFeatures[key]; switch(key) { case "GridName": ginfo.GridName = val; break; case "GridNick": ginfo.GridNick = val; break; case "GridURL": ginfo.GridUrl = val; break; case "GridURLAlias": string[] vals = val.Split(','); if(vals.Length > 0) ginfo.GridUrlAlias = vals; break; case "search-server-url": ginfo.SearchURL = val; break; case "destination-guide-url": ginfo.DestinationGuideURL = val; break; /* keep this local to avoid issues with diferent modules case "currency-base-uri": ginfo.EconomyURL = val; break; */ default: extrasMap[key] = val; if (key == "ExportSupported") { bool.TryParse(val, out m_ExportSupported); } break; } } m_features["OpenSimExtras"] = extrasMap; } } private string Replace(string url, string substring, string replacement) { if (!String.IsNullOrEmpty(url) && url.Contains(substring)) return url.Replace(substring, replacement); return url; } private void ReadScriptSyntax() { lock(m_scriptSyntaxLock) { if(!m_doScriptSyntax || m_scriptSyntaxID != UUID.Zero) return; if(!File.Exists("ScriptSyntax.xml")) return; try { using (StreamReader sr = File.OpenText("ScriptSyntax.xml")) { StringBuilder sb = new StringBuilder(400*1024); string s=""; char[] trimc = new char[] {' ','\t', '\n', '\r'}; s = sr.ReadLine(); if(s == null) return; s = s.Trim(trimc); UUID id; if(!UUID.TryParse(s,out id)) return; while ((s = sr.ReadLine()) != null) { s = s.Trim(trimc); if (String.IsNullOrEmpty(s) || s.StartsWith("