From ba745a524d4f13552f8ed3827a7315b3000cbe53 Mon Sep 17 00:00:00 2001
From: Justin Clark-Casey (justincc)
Date: Wed, 2 Jul 2014 23:48:44 +0100
Subject: Actually call Close() for shared region modules when the simulator is
being shutdown.
Adds regression test for this case.
---
OpenSim/Region/Application/OpenSimBase.cs | 44 +++-
.../Region/CoreModules/Asset/CenomeAssetCache.cs | 9 +-
OpenSim/Region/Framework/Scenes/Scene.cs | 1 +
.../Scenes/Tests/SharedRegionModuleTests.cs | 249 +++++++++++++++++++++
4 files changed, 290 insertions(+), 13 deletions(-)
create mode 100644 OpenSim/Region/Framework/Scenes/Tests/SharedRegionModuleTests.cs
(limited to 'OpenSim/Region')
diff --git a/OpenSim/Region/Application/OpenSimBase.cs b/OpenSim/Region/Application/OpenSimBase.cs
index feb3cf8..3dc9907 100644
--- a/OpenSim/Region/Application/OpenSimBase.cs
+++ b/OpenSim/Region/Application/OpenSimBase.cs
@@ -71,6 +71,20 @@ namespace OpenSim
// OpenSim.ini Section name for ESTATES Settings
public const string ESTATE_SECTION_NAME = "Estates";
+ ///
+ /// Allow all plugin loading to be disabled for tests/debug.
+ ///
+ ///
+ /// true by default
+ ///
+ public bool EnableInitialPluginLoad { get; set; }
+
+ ///
+ /// Control whether we attempt to load an estate data service.
+ ///
+ /// For tests/debugging
+ public bool LoadEstateDataService { get; set; }
+
protected string proxyUrl;
protected int proxyOffset = 0;
@@ -96,7 +110,7 @@ namespace OpenSim
public ConsoleCommand CreateAccount = null;
- protected List m_plugins = new List();
+ public List m_plugins = new List();
///
/// The config information passed into the OpenSimulator region server.
@@ -135,6 +149,8 @@ namespace OpenSim
///
public OpenSimBase(IConfigSource configSource) : base()
{
+ EnableInitialPluginLoad = true;
+ LoadEstateDataService = true;
LoadConfigSettings(configSource);
}
@@ -240,20 +256,25 @@ namespace OpenSim
if (String.IsNullOrEmpty(module))
throw new Exception("Configuration file is missing the LocalServiceModule parameter in the [EstateDataStore] section");
- m_estateDataService = ServerUtils.LoadPlugin(module, new object[] { Config });
- if (m_estateDataService == null)
- throw new Exception(
- string.Format(
- "Could not load an IEstateDataService implementation from {0}, as configured in the LocalServiceModule parameter of the [EstateDataStore] config section.",
- module));
+ if (LoadEstateDataService)
+ {
+ m_estateDataService = ServerUtils.LoadPlugin(module, new object[] { Config });
+ if (m_estateDataService == null)
+ throw new Exception(
+ string.Format(
+ "Could not load an IEstateDataService implementation from {0}, as configured in the LocalServiceModule parameter of the [EstateDataStore] config section.",
+ module));
+ }
base.StartupSpecific();
- LoadPlugins();
+ if (EnableInitialPluginLoad)
+ LoadPlugins();
+
+ // We still want to post initalize any plugins even if loading has been disabled since a test may have
+ // inserted them manually.
foreach (IApplicationPlugin plugin in m_plugins)
- {
plugin.PostInitialise();
- }
if (m_console != null)
AddPluginCommands(m_console);
@@ -878,6 +899,9 @@ namespace OpenSim
try
{
SceneManager.Close();
+
+ foreach (IApplicationPlugin plugin in m_plugins)
+ plugin.Dispose();
}
catch (Exception e)
{
diff --git a/OpenSim/Region/CoreModules/Asset/CenomeAssetCache.cs b/OpenSim/Region/CoreModules/Asset/CenomeAssetCache.cs
index 9b0e1f4..ebec9d2 100644
--- a/OpenSim/Region/CoreModules/Asset/CenomeAssetCache.cs
+++ b/OpenSim/Region/CoreModules/Asset/CenomeAssetCache.cs
@@ -316,9 +316,12 @@ namespace OpenSim.Region.CoreModules.Asset
///
public void Close()
{
- m_enabled = false;
- m_cache.Clear();
- m_cache = null;
+ if (m_enabled)
+ {
+ m_enabled = false;
+ m_cache.Clear();
+ m_cache = null;
+ }
}
///
diff --git a/OpenSim/Region/Framework/Scenes/Scene.cs b/OpenSim/Region/Framework/Scenes/Scene.cs
index ca42d5b..3957ba6 100644
--- a/OpenSim/Region/Framework/Scenes/Scene.cs
+++ b/OpenSim/Region/Framework/Scenes/Scene.cs
@@ -1894,6 +1894,7 @@ namespace OpenSim.Region.Framework.Scenes
RegionInfo.RegionID,
RegionInfo.RegionLocX, RegionInfo.RegionLocY,
RegionInfo.RegionSizeX, RegionInfo.RegionSizeY);
+
if (error != String.Empty)
throw new Exception(error);
}
diff --git a/OpenSim/Region/Framework/Scenes/Tests/SharedRegionModuleTests.cs b/OpenSim/Region/Framework/Scenes/Tests/SharedRegionModuleTests.cs
new file mode 100644
index 0000000..d499f87
--- /dev/null
+++ b/OpenSim/Region/Framework/Scenes/Tests/SharedRegionModuleTests.cs
@@ -0,0 +1,249 @@
+/*
+ * 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 System.Net;
+using Mono.Addins;
+using Nini.Config;
+using NUnit.Framework;
+using OpenMetaverse;
+using OpenSim;
+using OpenSim.ApplicationPlugins.RegionModulesController;
+using OpenSim.Framework;
+using OpenSim.Region.CoreModules.ServiceConnectorsOut.Grid;
+using OpenSim.Region.Framework.Interfaces;
+using OpenSim.Tests.Common;
+
+namespace OpenSim.Region.Framework.Scenes.Tests
+{
+ public class SharedRegionModuleTests : OpenSimTestCase
+ {
+ [Test]
+ public void TestLifecycle()
+ {
+ TestHelpers.InMethod();
+// TestHelpers.EnableLogging();
+
+ UUID estateOwnerId = TestHelpers.ParseTail(0x1);
+ UUID regionId = TestHelpers.ParseTail(0x10);
+
+ IConfigSource configSource = new IniConfigSource();
+ configSource.AddConfig("Startup");
+ configSource.AddConfig("Modules");
+
+// // We use this to skip estate questions
+ // Turns out not to be needed is estate owner id is pre-set in region information.
+// IConfig estateConfig = configSource.AddConfig(OpenSimBase.ESTATE_SECTION_NAME);
+// estateConfig.Set("DefaultEstateOwnerName", "Zaphod Beeblebrox");
+// estateConfig.Set("DefaultEstateOwnerUUID", estateOwnerId);
+// estateConfig.Set("DefaultEstateOwnerEMail", "zaphod@galaxy.com");
+// estateConfig.Set("DefaultEstateOwnerPassword", "two heads");
+
+ // For grid servic
+ configSource.AddConfig("GridService");
+ configSource.Configs["Modules"].Set("GridServices", "LocalGridServicesConnector");
+ configSource.Configs["GridService"].Set("StorageProvider", "OpenSim.Data.Null.dll:NullRegionData");
+ configSource.Configs["GridService"].Set("LocalServiceModule", "OpenSim.Services.GridService.dll:GridService");
+ configSource.Configs["GridService"].Set("ConnectionString", "!static");
+
+ LocalGridServicesConnector gridService = new LocalGridServicesConnector();
+//
+ OpenSim sim = new OpenSim(configSource);
+
+ sim.SuppressExit = true;
+ sim.EnableInitialPluginLoad = false;
+ sim.LoadEstateDataService = false;
+ sim.NetServersInfo.HttpListenerPort = 0;
+
+ IRegistryCore reg = sim.ApplicationRegistry;
+
+ RegionInfo ri = new RegionInfo();
+ ri.RegionID = regionId;
+ ri.EstateSettings.EstateOwner = estateOwnerId;
+ ri.InternalEndPoint = new IPEndPoint(0, 0);
+
+ MockRegionModulesControllerPlugin rmcp = new MockRegionModulesControllerPlugin();
+ sim.m_plugins = new List() { rmcp };
+ reg.RegisterInterface(rmcp);
+
+ // XXX: Have to initialize directly for now
+ rmcp.Initialise(sim);
+
+ rmcp.AddNode(gridService);
+
+ TestSharedRegion tsr = new TestSharedRegion();
+ rmcp.AddNode(tsr);
+
+ // FIXME: Want to use the real one eventually but this is currently directly tied into Mono.Addins
+ // which has been written in such a way that makes it impossible to use for regression tests.
+// RegionModulesControllerPlugin rmcp = new RegionModulesControllerPlugin();
+// rmcp.LoadModulesFromAddins = false;
+//// reg.RegisterInterface(rmcp);
+// rmcp.Initialise(sim);
+// rmcp.PostInitialise();
+// TypeExtensionNode node = new TypeExtensionNode();
+// node.
+// rmcp.AddNode(node, configSource.Configs["Modules"], new Dictionary>());
+
+ sim.Startup();
+ IScene scene;
+ sim.CreateRegion(ri, out scene);
+
+ sim.Shutdown();
+
+ List co = tsr.CallOrder;
+ int expectedEventCount = 6;
+
+ Assert.AreEqual(
+ expectedEventCount,
+ co.Count,
+ "Expected {0} events but only got {1} ({2})",
+ expectedEventCount, co.Count, string.Join(",", co));
+ Assert.AreEqual("Initialise", co[0]);
+ Assert.AreEqual("PostInitialise", co[1]);
+ Assert.AreEqual("AddRegion", co[2]);
+ Assert.AreEqual("RegionLoaded", co[3]);
+ Assert.AreEqual("RemoveRegion", co[4]);
+ Assert.AreEqual("Close", co[5]);
+ }
+ }
+
+ class TestSharedRegion : ISharedRegionModule
+ {
+ // FIXME: Should really use MethodInfo
+ public List CallOrder = new List();
+
+ public string Name { get { return "TestSharedRegion"; } }
+
+ public Type ReplaceableInterface { get { return null; } }
+
+ public void PostInitialise()
+ {
+ CallOrder.Add("PostInitialise");
+ }
+
+ public void Initialise(IConfigSource source)
+ {
+ CallOrder.Add("Initialise");
+ }
+
+ public void Close()
+ {
+ CallOrder.Add("Close");
+ }
+
+ public void AddRegion(Scene scene)
+ {
+ CallOrder.Add("AddRegion");
+ }
+
+ public void RemoveRegion(Scene scene)
+ {
+ CallOrder.Add("RemoveRegion");
+ }
+
+ public void RegionLoaded(Scene scene)
+ {
+ CallOrder.Add("RegionLoaded");
+ }
+ }
+
+ class MockRegionModulesControllerPlugin : IRegionModulesController, IApplicationPlugin
+ {
+ // List of shared module instances, for adding to Scenes
+ private List m_sharedInstances = new List();
+
+ // Config access
+ private OpenSimBase m_openSim;
+
+ public string Version { get { return "0"; } }
+ public string Name { get { return "MockRegionModulesControllerPlugin"; } }
+
+ public void Initialise() {}
+
+ public void Initialise(OpenSimBase sim)
+ {
+ m_openSim = sim;
+ }
+
+ ///
+ /// Called when the application loading is completed
+ ///
+ public void PostInitialise()
+ {
+ foreach (ISharedRegionModule module in m_sharedInstances)
+ module.PostInitialise();
+ }
+
+ public void AddRegionToModules(Scene scene)
+ {
+ List sharedlist = new List();
+
+ foreach (ISharedRegionModule module in m_sharedInstances)
+ {
+ module.AddRegion(scene);
+ scene.AddRegionModule(module.Name, module);
+
+ sharedlist.Add(module);
+ }
+
+ foreach (ISharedRegionModule module in sharedlist)
+ {
+ module.RegionLoaded(scene);
+ }
+ }
+
+ public void RemoveRegionFromModules(Scene scene)
+ {
+ foreach (IRegionModuleBase module in scene.RegionModules.Values)
+ {
+// m_log.DebugFormat("[REGIONMODULE]: Removing scene {0} from module {1}",
+// scene.RegionInfo.RegionName, module.Name);
+ module.RemoveRegion(scene);
+ }
+
+ scene.RegionModules.Clear();
+ }
+
+ public void AddNode(ISharedRegionModule module)
+ {
+ m_sharedInstances.Add(module);
+ module.Initialise(m_openSim.ConfigSource.Source);
+ }
+
+ public void Dispose()
+ {
+ // We expect that all regions have been removed already
+ while (m_sharedInstances.Count > 0)
+ {
+ m_sharedInstances[0].Close();
+ m_sharedInstances.RemoveAt(0);
+ }
+ }
+ }
+}
\ No newline at end of file
--
cgit v1.1