* First draft resolution of mantis 777, 734, 389 - scripts do not save in non-home regions
* Should work in multi-region standalone and grid modes * This should also solve other non-home region caps issues (map requests, RC client inventory requests, etc) * We now pass CAPS information on to the destination region on region crossing, and set up a CAPS object when an agent becomes a master * Current limitation is that this will only work if your http_listener_port is 9000 * This is a very early code cut (lots of bad practice, hard coding and inefficiency). However, I wanted to get this out there for feedback and my own sanity. Next few patches will clean up the mess.0.6.0-stable
							parent
							
								
									f61ea1998e
								
							
						
					
					
						commit
						c1beb85315
					
				|  | @ -62,7 +62,12 @@ namespace OpenSim.Region.Capabilities | ||||||
|         private string m_httpListenerHostName; |         private string m_httpListenerHostName; | ||||||
|         private uint m_httpListenPort; |         private uint m_httpListenPort; | ||||||
| 
 | 
 | ||||||
|         private string m_capsObjectPath = "00001-"; |         /// <summary> | ||||||
|  |         /// This is the uuid portion of every CAPS path.  It is used to make capability urls private to the requester. | ||||||
|  |         /// </summary> | ||||||
|  |         private string m_capsObjectPath;         | ||||||
|  |         public string CapsObjectPath { get { return m_capsObjectPath; } } | ||||||
|  |          | ||||||
|         private string m_requestPath = "0000/"; |         private string m_requestPath = "0000/"; | ||||||
|         private string m_mapLayerPath = "0001/"; |         private string m_mapLayerPath = "0001/"; | ||||||
|         private string m_newInventory = "0002/"; |         private string m_newInventory = "0002/"; | ||||||
|  | @ -109,9 +114,12 @@ namespace OpenSim.Region.Capabilities | ||||||
|              |              | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|  |                 m_httpListener.RemoveStreamHandler("POST", capsBase + m_mapLayerPath); | ||||||
|                 m_httpListener.AddStreamHandler( |                 m_httpListener.AddStreamHandler( | ||||||
|                     new LLSDStreamhandler<LLSDMapRequest, LLSDMapLayerResponse>("POST", capsBase + m_mapLayerPath, |                     new LLSDStreamhandler<LLSDMapRequest, LLSDMapLayerResponse>("POST", capsBase + m_mapLayerPath, | ||||||
|                                                                                 GetMapLayer)); |                                                                                 GetMapLayer)); | ||||||
|  |                  | ||||||
|  |                 m_httpListener.RemoveStreamHandler("POST", capsBase + m_newInventory);                 | ||||||
|                 m_httpListener.AddStreamHandler( |                 m_httpListener.AddStreamHandler( | ||||||
|                     new LLSDStreamhandler<LLSDAssetUploadRequest, LLSDAssetUploadResponse>("POST", |                     new LLSDStreamhandler<LLSDAssetUploadRequest, LLSDAssetUploadResponse>("POST", | ||||||
|                                                                                            capsBase + m_newInventory, |                                                                                            capsBase + m_newInventory, | ||||||
|  | @ -142,6 +150,7 @@ namespace OpenSim.Region.Capabilities | ||||||
|         private void AddLegacyCapsHandler(BaseHttpServer httpListener, string path, RestMethod restMethod) |         private void AddLegacyCapsHandler(BaseHttpServer httpListener, string path, RestMethod restMethod) | ||||||
|         { |         { | ||||||
|             string capsBase = "/CAPS/" + m_capsObjectPath; |             string capsBase = "/CAPS/" + m_capsObjectPath; | ||||||
|  |             httpListener.RemoveStreamHandler("POST", capsBase + path);  | ||||||
|             httpListener.AddStreamHandler(new RestStreamHandler("POST", capsBase + path, restMethod)); |             httpListener.AddStreamHandler(new RestStreamHandler("POST", capsBase + path, restMethod)); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -63,5 +63,7 @@ namespace OpenSim.Framework | ||||||
|         RegionStatus Region_Status { get; set; } |         RegionStatus Region_Status { get; set; } | ||||||
| 
 | 
 | ||||||
|         ClientManager ClientManager { get; } |         ClientManager ClientManager { get; } | ||||||
|  |          | ||||||
|  |         string GetCapsPath(LLUUID agentId);        | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -70,13 +70,22 @@ namespace OpenSim.Framework.Servers | ||||||
|             m_port = port; |             m_port = port; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Add a stream handler to the http server.  If the handler already exists, then nothing happens. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="handler"></param> | ||||||
|         public void AddStreamHandler(IRequestHandler handler) |         public void AddStreamHandler(IRequestHandler handler) | ||||||
|         { |         { | ||||||
|             string httpMethod = handler.HttpMethod; |             string httpMethod = handler.HttpMethod; | ||||||
|             string path = handler.Path; |             string path = handler.Path; | ||||||
| 
 | 
 | ||||||
|             string handlerKey = GetHandlerKey(httpMethod, path); |             string handlerKey = GetHandlerKey(httpMethod, path); | ||||||
|             m_streamHandlers.Add(handlerKey, handler); |                                      | ||||||
|  |             if (!m_streamHandlers.ContainsKey(handlerKey)) | ||||||
|  |             { | ||||||
|  |                 //m_log.DebugFormat("[BASE HTTP SERVER]: Adding handler key {0}", handlerKey);                 | ||||||
|  |                 m_streamHandlers.Add(handlerKey, handler); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private static string GetHandlerKey(string httpMethod, string path) |         private static string GetHandlerKey(string httpMethod, string path) | ||||||
|  | @ -289,7 +298,7 @@ namespace OpenSim.Framework.Servers | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|                     System.Console.WriteLine("Handler not found for http request " + request.RawUrl); |                     m_log.ErrorFormat("[BASE HTTP SERVER] Handler not found for http request {0}", request.RawUrl); | ||||||
|                     responseString = "Error"; |                     responseString = "Error"; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  | @ -589,7 +598,11 @@ namespace OpenSim.Framework.Servers | ||||||
| 
 | 
 | ||||||
|         public void RemoveStreamHandler(string httpMethod, string path) |         public void RemoveStreamHandler(string httpMethod, string path) | ||||||
|         { |         { | ||||||
|             m_streamHandlers.Remove(GetHandlerKey(httpMethod, path)); |             string handlerKey = GetHandlerKey(httpMethod, path);             | ||||||
|  |              | ||||||
|  |             //m_log.DebugFormat("[BASE HTTP SERVER]: Removing handler key {0}", handlerKey); | ||||||
|  |              | ||||||
|  |             m_streamHandlers.Remove(handlerKey); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public void RemoveHTTPHandler(string httpMethod, string path) |         public void RemoveHTTPHandler(string httpMethod, string path) | ||||||
|  |  | ||||||
|  | @ -45,7 +45,6 @@ namespace OpenSim.Framework | ||||||
|         private static Random randomClass = new Random(); |         private static Random randomClass = new Random(); | ||||||
|         private static uint nextXferID = 5000; |         private static uint nextXferID = 5000; | ||||||
|         private static object XferLock = new object(); |         private static object XferLock = new object(); | ||||||
|         private static Dictionary<LLUUID, string> capsURLS = new Dictionary<LLUUID, string>(); |  | ||||||
| 
 | 
 | ||||||
|         // Get a list of invalid path characters (OS dependent) |         // Get a list of invalid path characters (OS dependent) | ||||||
|         private static string regexInvalidPathChars = "[" + new String(Path.GetInvalidPathChars()) + "]"; |         private static string regexInvalidPathChars = "[" + new String(Path.GetInvalidPathChars()) + "]"; | ||||||
|  | @ -422,27 +421,6 @@ namespace OpenSim.Framework | ||||||
|             return "."; |             return "."; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public static string GetCapsURL(LLUUID userID) |  | ||||||
|         { |  | ||||||
|             if (capsURLS.ContainsKey(userID)) |  | ||||||
|             { |  | ||||||
|                 return capsURLS[userID]; |  | ||||||
|             } |  | ||||||
|             return String.Empty; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static void SetCapsURL(LLUUID userID, string url) |  | ||||||
|         { |  | ||||||
|             if (capsURLS.ContainsKey(userID)) |  | ||||||
|             { |  | ||||||
|                 capsURLS[userID] = url; |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 capsURLS.Add(userID, url); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         // Nini (config) related Methods |         // Nini (config) related Methods | ||||||
|         public static IConfigSource ConvertDataRowToXMLConfig(DataRow row, string fileName) |         public static IConfigSource ConvertDataRowToXMLConfig(DataRow row, string fileName) | ||||||
|         { |         { | ||||||
|  |  | ||||||
|  | @ -984,7 +984,7 @@ namespace OpenSim.Region.ClientStack | ||||||
|             agentData.child = false; |             agentData.child = false; | ||||||
|             agentData.firstname = m_firstName; |             agentData.firstname = m_firstName; | ||||||
|             agentData.lastname = m_lastName; |             agentData.lastname = m_lastName; | ||||||
|             agentData.CapsPath = String.Empty; |             agentData.CapsPath = m_scene.GetCapsPath(m_agentId); | ||||||
|             return agentData; |             return agentData; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1649,53 +1649,11 @@ namespace OpenSim.Region.Environment.Scenes | ||||||
|         { |         { | ||||||
|             if (regionHandle == m_regInfo.RegionHandle) |             if (regionHandle == m_regInfo.RegionHandle) | ||||||
|             { |             { | ||||||
|                 if (agent.CapsPath != String.Empty) |                 capsPaths[agent.AgentID] = agent.CapsPath; | ||||||
|                 { |  | ||||||
|                     m_log.DebugFormat( |  | ||||||
|                         "[CONNECTION DEBUGGING]: Setting up CAPS handler for avatar {0} at {1} in {2}", |  | ||||||
|                         agent.AgentID, agent.CapsPath, RegionInfo.RegionName); |  | ||||||
|                      |                      | ||||||
|                     Caps cap = |                 if (!agent.child) | ||||||
|                         new Caps(AssetCache, m_httpListener, m_regInfo.ExternalHostName, m_httpListener.Port, |                 { | ||||||
|                                  agent.CapsPath, agent.AgentID, m_dumpAssetsToFile); |                     AddCapsHandler(agent.AgentID); | ||||||
| 
 |  | ||||||
|                     Util.SetCapsURL(agent.AgentID, |  | ||||||
|                                     "http://" + m_regInfo.ExternalHostName + ":" + m_httpListener.Port.ToString() + |  | ||||||
|                                     "/CAPS/" + agent.CapsPath + "0000/"); |  | ||||||
|                     cap.RegisterHandlers(); |  | ||||||
|                     if (agent.child) |  | ||||||
|                     { |  | ||||||
| 
 |  | ||||||
|                     } |  | ||||||
|                     cap.AddNewInventoryItem = AddInventoryItem; |  | ||||||
|                     cap.ItemUpdatedCall = CapsUpdateInventoryItemAsset; |  | ||||||
|                     cap.TaskScriptUpdatedCall = CapsUpdateTaskInventoryScriptAsset; |  | ||||||
|                     cap.CAPSFetchInventoryDescendents = CommsManager.UserProfileCacheService.HandleFetchInventoryDescendentsCAPS; |  | ||||||
| 
 |  | ||||||
|                     if (m_capsHandlers.ContainsKey(agent.AgentID)) |  | ||||||
|                     { |  | ||||||
|                         m_log.DebugFormat( |  | ||||||
|                             "[CONNECTION DEBUGGING]: Caps path already in use for avatar {0} in region {1}",  |  | ||||||
|                             agent.AgentID, RegionInfo.RegionName); |  | ||||||
|                          |  | ||||||
|                         try |  | ||||||
|                         { |  | ||||||
|                             m_capsHandlers[agent.AgentID] = cap; |  | ||||||
|                         } |  | ||||||
|                         catch (KeyNotFoundException) |  | ||||||
|                         { |  | ||||||
|                             m_log.DebugFormat( |  | ||||||
|                                 "[CONNECTION DEBUGGING]: Caught exception adding handler for avatar {0} at {1}",  |  | ||||||
|                                 agent.AgentID, RegionInfo.RegionName); |  | ||||||
|                              |  | ||||||
|                             // Fix for a potential race condition. |  | ||||||
|                             m_capsHandlers.Add(agent.AgentID, cap); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     else |  | ||||||
|                     { |  | ||||||
|                         m_capsHandlers.Add(agent.AgentID, cap); |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|  | @ -1716,6 +1674,56 @@ namespace OpenSim.Region.Environment.Scenes | ||||||
|                     "[CONNECTION DEBUGGING]: Skipping this region for welcoming avatar {0} [{1}] at {2}",  |                     "[CONNECTION DEBUGGING]: Skipping this region for welcoming avatar {0} [{1}] at {2}",  | ||||||
|                     agent.AgentID, regionHandle, RegionInfo.RegionName); |                     agent.AgentID, regionHandle, RegionInfo.RegionName); | ||||||
|             } |             } | ||||||
|  |         }                 | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Add a caps handler for the given agent. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="agentId"></param>    | ||||||
|  |         /// <param name="capsObjectPath"></param>      | ||||||
|  |         public void AddCapsHandler(LLUUID agentId) | ||||||
|  |         {            | ||||||
|  |             String capsObjectPath = GetCapsPath(agentId);             | ||||||
|  |              | ||||||
|  |             m_log.DebugFormat( | ||||||
|  |                 "[CONNECTION DEBUGGING]: Setting up CAPS handler for avatar {0} at {1} in {2}", | ||||||
|  |                 agentId, capsObjectPath, RegionInfo.RegionName); | ||||||
|  |              | ||||||
|  |             Caps cap = | ||||||
|  |                 new Caps(AssetCache, m_httpListener, m_regInfo.ExternalHostName, m_httpListener.Port, | ||||||
|  |                          capsObjectPath, agentId, m_dumpAssetsToFile); | ||||||
|  | 
 | ||||||
|  |             cap.RegisterHandlers(); | ||||||
|  | 
 | ||||||
|  |             cap.AddNewInventoryItem = AddInventoryItem; | ||||||
|  |             cap.ItemUpdatedCall = CapsUpdateInventoryItemAsset; | ||||||
|  |             cap.TaskScriptUpdatedCall = CapsUpdateTaskInventoryScriptAsset; | ||||||
|  |             cap.CAPSFetchInventoryDescendents = CommsManager.UserProfileCacheService.HandleFetchInventoryDescendentsCAPS; | ||||||
|  | 
 | ||||||
|  |             if (m_capsHandlers.ContainsKey(agentId)) | ||||||
|  |             { | ||||||
|  |                 m_log.DebugFormat( | ||||||
|  |                     "[CONNECTION DEBUGGING]: Caps path already in use for avatar {0} in region {1}",  | ||||||
|  |                     agentId, RegionInfo.RegionName); | ||||||
|  |                  | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     m_capsHandlers[agentId] = cap; | ||||||
|  |                 } | ||||||
|  |                 catch (KeyNotFoundException) | ||||||
|  |                 { | ||||||
|  |                     m_log.DebugFormat( | ||||||
|  |                         "[CONNECTION DEBUGGING]: Caught exception adding handler for avatar {0} at {1}",  | ||||||
|  |                         agentId, RegionInfo.RegionName); | ||||||
|  |                      | ||||||
|  |                     // Fix for a potential race condition. | ||||||
|  |                     m_capsHandlers.Add(agentId, cap); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 m_capsHandlers.Add(agentId, cap); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|  |  | ||||||
|  | @ -26,6 +26,7 @@ | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| using System; | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
| using libsecondlife; | using libsecondlife; | ||||||
| using OpenSim.Framework; | using OpenSim.Framework; | ||||||
| using OpenSim.Framework.Communications.Cache; | using OpenSim.Framework.Communications.Cache; | ||||||
|  | @ -196,5 +197,19 @@ namespace OpenSim.Region.Environment.Scenes | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         #endregion |         #endregion | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         /// XXX These two methods are very temporary | ||||||
|  |         /// </summary> | ||||||
|  |         protected Dictionary<LLUUID, String> capsPaths = new Dictionary<LLUUID, String>(); | ||||||
|  |         public string GetCapsPath(LLUUID agentId) | ||||||
|  |         { | ||||||
|  |             if (capsPaths.ContainsKey(agentId)) | ||||||
|  |             { | ||||||
|  |                 return capsPaths[agentId]; | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -483,7 +483,7 @@ namespace OpenSim.Region.Environment.Scenes | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///  |         /// Try to teleport an agent to a new region. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|         /// <param name="remoteClient"></param> |         /// <param name="remoteClient"></param> | ||||||
|         /// <param name="RegionHandle"></param> |         /// <param name="RegionHandle"></param> | ||||||
|  | @ -530,7 +530,11 @@ namespace OpenSim.Region.Environment.Scenes | ||||||
|                         m_commsProvider.InterRegion.ExpectAvatarCrossing(regionHandle, avatar.ControllingClient.AgentId, |                         m_commsProvider.InterRegion.ExpectAvatarCrossing(regionHandle, avatar.ControllingClient.AgentId, | ||||||
|                                                                      position, false); |                                                                      position, false); | ||||||
|                         AgentCircuitData circuitdata = avatar.ControllingClient.RequestClientInfo(); |                         AgentCircuitData circuitdata = avatar.ControllingClient.RequestClientInfo(); | ||||||
|                         string capsPath = Util.GetCapsURL(avatar.ControllingClient.AgentId); |                          | ||||||
|  |                         // TODO Should construct this behind a method | ||||||
|  |                         string capsPath =  | ||||||
|  |                             "http://" + reg.ExternalHostName + ":" + 9000 + "/CAPS/" + circuitdata.CapsPath + "0000/";     | ||||||
|  |                          | ||||||
|                         avatar.ControllingClient.SendRegionTeleport(regionHandle, 13, reg.ExternalEndPoint, 4, (1 << 4), |                         avatar.ControllingClient.SendRegionTeleport(regionHandle, 13, reg.ExternalEndPoint, 4, (1 << 4), | ||||||
|                                                                     capsPath); |                                                                     capsPath); | ||||||
|                         avatar.MakeChildAgent(); |                         avatar.MakeChildAgent(); | ||||||
|  |  | ||||||
|  | @ -545,6 +545,7 @@ namespace OpenSim.Region.Environment.Scenes | ||||||
| 
 | 
 | ||||||
|             m_scene.SwapRootAgentCount(false); |             m_scene.SwapRootAgentCount(false); | ||||||
|             m_scene.CommsManager.UserProfileCacheService.UpdateUserInventory(m_uuid); |             m_scene.CommsManager.UserProfileCacheService.UpdateUserInventory(m_uuid); | ||||||
|  |             m_scene.AddCapsHandler(m_uuid); | ||||||
|             //if (!m_gotAllObjectsInScene) |             //if (!m_gotAllObjectsInScene) | ||||||
|             //{ |             //{ | ||||||
|             m_scene.SendAllSceneObjectsToClient(this); |             m_scene.SendAllSceneObjectsToClient(this); | ||||||
|  | @ -1616,7 +1617,15 @@ namespace OpenSim.Region.Environment.Scenes | ||||||
|                 if (res) |                 if (res) | ||||||
|                 { |                 { | ||||||
|                     AgentCircuitData circuitdata = m_controllingClient.RequestClientInfo(); |                     AgentCircuitData circuitdata = m_controllingClient.RequestClientInfo(); | ||||||
|                     string capsPath = Util.GetCapsURL(m_controllingClient.AgentId); | 
 | ||||||
|  |                     // TODO Should construct this behind a method | ||||||
|  |                     string capsPath =  | ||||||
|  |                         "http://" + neighbourRegion.ExternalHostName + ":" + 9000 | ||||||
|  |                          + "/CAPS/" + circuitdata.CapsPath + "0000/"; | ||||||
|  |                      | ||||||
|  |                     m_log.DebugFormat( | ||||||
|  |                         "[CONNECTION DEBUGGING]: Sending new CAPS seed url {0} to avatar {1}", capsPath, m_uuid); | ||||||
|  |                      | ||||||
|                     m_controllingClient.CrossRegion(neighbourHandle, newpos, vel, neighbourRegion.ExternalEndPoint, |                     m_controllingClient.CrossRegion(neighbourHandle, newpos, vel, neighbourRegion.ExternalEndPoint, | ||||||
|                                                     capsPath); |                                                     capsPath); | ||||||
|                     MakeChildAgent(); |                     MakeChildAgent(); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 Justin Clarke Casey
						Justin Clarke Casey