* 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