From 6a8d8f54e88a21e6cfd78dc7981cdeec2f18094d Mon Sep 17 00:00:00 2001 From: Tleiades Hax Date: Tue, 30 Oct 2007 22:42:34 +0000 Subject: Step one on the long march towards grid based inventory. Introduction of an InevntoryServer --- OpenSim/Framework/General/InventoryConfig.cs | 67 ++++++++ OpenSim/Framework/General/InventoryItemBase.cs | 30 +++- OpenSim/Framework/General/NetworkServersInfo.cs | 6 +- OpenSim/Grid/InventoryServer/InventoryManager.cs | 170 +++++++++++++++------ OpenSim/Grid/InventoryServer/Main.cs | 55 +++++-- OpenSim/Region/Application/OpenSimMain.cs | 3 +- .../ClientStack/ClientView.ProcessPackets.cs | 2 +- .../Communications/OGS1/CommunicationsOGS1.cs | 4 +- .../Communications/OGS1/OGS1InventoryService.cs | 24 ++- OpenSim/Region/Environment/Modules/ChatModule.cs | 2 +- .../Region/Environment/Modules/WorldCommModule.cs | 2 +- OpenSim/Region/Environment/Scenes/ScenePresence.cs | 2 +- bin/Inventory_Default.xml | 62 ++++++++ bin/Inventory_Library.xml | 144 +++++++++++++++++ prebuild.xml | 29 ++++ 15 files changed, 531 insertions(+), 71 deletions(-) create mode 100644 OpenSim/Framework/General/InventoryConfig.cs create mode 100644 bin/Inventory_Default.xml create mode 100644 bin/Inventory_Library.xml diff --git a/OpenSim/Framework/General/InventoryConfig.cs b/OpenSim/Framework/General/InventoryConfig.cs new file mode 100644 index 0000000..9ba3e07 --- /dev/null +++ b/OpenSim/Framework/General/InventoryConfig.cs @@ -0,0 +1,67 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace OpenSim.Framework +{ + /// + /// UserConfig -- For User Server Configuration + /// + public class InventoryConfig + { + public string DefaultStartupMsg = ""; + public string UserServerURL = ""; + public string UserSendKey = ""; + public string UserRecvKey = ""; + + public string DatabaseProvider = ""; + public static uint DefaultHttpPort = 8004; + + public int HttpPort = 8004; + + private ConfigurationMember configMember; + + public InventoryConfig(string description, string filename) + { + configMember = new ConfigurationMember(filename, description, this.loadConfigurationOptions, this.handleIncomingConfiguration); + configMember.performConfigurationRetrieve(); + } + + public void loadConfigurationOptions() + { + configMember.addConfigurationOption("default_startup_message", ConfigurationOption.ConfigurationTypes.TYPE_STRING_NOT_EMPTY, "Default Startup Message", "Welcome to OGS", false); + configMember.addConfigurationOption("default_user_server", ConfigurationOption.ConfigurationTypes.TYPE_STRING_NOT_EMPTY, "Default User Server URI", "http://127.0.0.1:" + UserConfig.DefaultHttpPort.ToString(), false); + configMember.addConfigurationOption("user_send_key", ConfigurationOption.ConfigurationTypes.TYPE_STRING, "Key to send to user server", "null", false); + configMember.addConfigurationOption("user_recv_key", ConfigurationOption.ConfigurationTypes.TYPE_STRING, "Key to expect from user server", "null", false); + configMember.addConfigurationOption("database_provider", ConfigurationOption.ConfigurationTypes.TYPE_STRING, "DLL for database provider", "OpenSim.Framework.Data.MySQL.dll", false); + configMember.addConfigurationOption("http_port", ConfigurationOption.ConfigurationTypes.TYPE_INT32, "Http Listener port", DefaultHttpPort.ToString(), false); + } + + public bool handleIncomingConfiguration(string configuration_key, object configuration_result) + { + switch (configuration_key) + { + case "default_startup_message": + this.DefaultStartupMsg = (string)configuration_result; + break; + case "default_user_server": + this.UserServerURL = (string)configuration_result; + break; + case "user_send_key": + this.UserSendKey = (string)configuration_result; + break; + case "user_recv_key": + this.UserRecvKey = (string)configuration_result; + break; + case "database_provider": + this.DatabaseProvider = (string)configuration_result; + break; + case "http_port": + HttpPort = (int)configuration_result; + break; + } + + return true; + } + } +} diff --git a/OpenSim/Framework/General/InventoryItemBase.cs b/OpenSim/Framework/General/InventoryItemBase.cs index 45700ae..f782913 100644 --- a/OpenSim/Framework/General/InventoryItemBase.cs +++ b/OpenSim/Framework/General/InventoryItemBase.cs @@ -25,6 +25,9 @@ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * */ +using System; +using System.Xml.Serialization; +using System.Collections; using System.Collections.Generic; using libsecondlife; @@ -242,4 +245,29 @@ namespace OpenSim.Framework /// The id of the folder void deleteInventoryFolder(LLUUID folder); } -} \ No newline at end of file + + /* + * .Net has some issues, serializing a dictionary, so we cannot reuse the InventoryFolder + * class defined in Communications.Framework.Communications.Caches. So we serialize/deserialize + * into this simpler class, and then use that. + */ + [XmlRoot(ElementName = "inventory", IsNullable = true)] + public class SerializableInventory + { + [XmlRoot(ElementName = "folder", IsNullable = true)] + public class SerializableFolder : InventoryFolderBase + { + [XmlArray(ElementName = "folders", IsNullable = true)] + [XmlArrayItem(ElementName = "folder", IsNullable = true, Type = typeof(SerializableFolder))] + public ArrayList SubFolders; + + [XmlArray(ElementName = "items", IsNullable = true)] + [XmlArrayItem(ElementName = "item", IsNullable = true, Type = typeof(InventoryItemBase))] + public ArrayList Items; + } + + [XmlElement(ElementName = "folder", IsNullable = true)] + public SerializableFolder root; + } + +} diff --git a/OpenSim/Framework/General/NetworkServersInfo.cs b/OpenSim/Framework/General/NetworkServersInfo.cs index 98d489e..aa8aa2b 100644 --- a/OpenSim/Framework/General/NetworkServersInfo.cs +++ b/OpenSim/Framework/General/NetworkServersInfo.cs @@ -43,6 +43,8 @@ namespace OpenSim.Framework public string UserRecvKey = ""; public bool isSandbox; + public string InventoryURL = ""; + public static int DefaultHttpListenerPort = 9000; public int HttpListenerPort = DefaultHttpListenerPort; @@ -91,6 +93,8 @@ namespace OpenSim.Framework UserSendKey = config.Configs["Network"].GetString("user_send_key", "null"); UserRecvKey = config.Configs["Network"].GetString("user_recv_key", "null"); AssetURL = config.Configs["Network"].GetString("asset_server_url", AssetURL); + InventoryURL = config.Configs["Network"].GetString("inventory_server_url", + "http://127.0.0.1:" + InventoryConfig.DefaultHttpPort.ToString()); } } -} \ No newline at end of file +} diff --git a/OpenSim/Grid/InventoryServer/InventoryManager.cs b/OpenSim/Grid/InventoryServer/InventoryManager.cs index 016b8bb..ca9ebf4 100644 --- a/OpenSim/Grid/InventoryServer/InventoryManager.cs +++ b/OpenSim/Grid/InventoryServer/InventoryManager.cs @@ -26,37 +26,36 @@ * */ using System; -using System.Collections; -using System.Collections.Generic; +using System.IO; using System.Text; -using OpenGrid.Framework.Data; -using libsecondlife; using System.Reflection; - +using System.Collections; +using System.Collections.Generic; using System.Xml; -using Nwc.XmlRpc; -using OpenSim.Framework.Sims; -using OpenSim.Framework.Inventory; -using OpenSim.Framework.Utilities; +using System.Xml.Serialization; +using libsecondlife; -using System.Security.Cryptography; +using OpenSim.Framework; +using OpenSim.Framework.Console; +using OpenSim.Framework.Servers; -namespace OpenGridServices.InventoryServer +namespace OpenSim.Grid.InventoryServer { - class InventoryManager + + public class InventoryManager { - Dictionary _plugins = new Dictionary(); + IInventoryData _databasePlugin; /// /// Adds a new inventory server plugin - user servers will be requested in the order they were loaded. /// /// The filename to the inventory server plugin DLL - public void AddPlugin(string FileName) + public void AddDatabasePlugin(string FileName) { - OpenSim.Framework.Console.MainConsole.Instance.Verbose( "Invenstorage: Attempting to load " + FileName); + MainLog.Instance.Verbose(OpenInventory_Main.LogName, "Invenstorage: Attempting to load " + FileName); Assembly pluginAssembly = Assembly.LoadFrom(FileName); - OpenSim.Framework.Console.MainConsole.Instance.Verbose( "Invenstorage: Found " + pluginAssembly.GetTypes().Length + " interfaces."); + MainLog.Instance.Verbose(OpenInventory_Main.LogName, "Invenstorage: Found " + pluginAssembly.GetTypes().Length + " interfaces."); foreach (Type pluginType in pluginAssembly.GetTypes()) { if (!pluginType.IsAbstract) @@ -67,8 +66,9 @@ namespace OpenGridServices.InventoryServer { IInventoryData plug = (IInventoryData)Activator.CreateInstance(pluginAssembly.GetType(pluginType.ToString())); plug.Initialise(); - this._plugins.Add(plug.getName(), plug); - OpenSim.Framework.Console.MainConsole.Instance.Verbose( "Invenstorage: Added IUserData Interface"); + _databasePlugin = plug; + MainLog.Instance.Verbose(OpenInventory_Main.LogName, "Invenstorage: Added IInventoryData Interface"); + break; } typeInterface = null; @@ -78,48 +78,132 @@ namespace OpenGridServices.InventoryServer pluginAssembly = null; } - public List getRootFolders(LLUUID user) + protected static SerializableInventory loadInventoryFromXmlFile(string fileName) + { + FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read); + XmlReader reader = new XmlTextReader(fs); + XmlSerializer x = new XmlSerializer(typeof(SerializableInventory)); + SerializableInventory inventory = (SerializableInventory)x.Deserialize(reader); + fs.Close(); + fs.Dispose(); + return inventory; + } + + protected static void saveInventoryToStream(SerializableInventory inventory, Stream s) { - foreach (KeyValuePair kvp in _plugins) + XmlTextWriter writer = new XmlTextWriter(s, Encoding.UTF8); + writer.Formatting = Formatting.Indented; + XmlSerializer x = new XmlSerializer(typeof(SerializableInventory)); + x.Serialize(writer, inventory); + } + + protected static bool fixupFolder(SerializableInventory.SerializableFolder f, SerializableInventory.SerializableFolder parent) + { + bool modified = false; + + // ensure we have a valid folder id + if (f.folderID == LLUUID.Zero) { - try - { - return kvp.Value.getUserRootFolders(user); - } - catch (Exception e) - { - OpenSim.Framework.Console.MainConsole.Instance.Notice("Unable to get root folders via " + kvp.Key + " (" + e.ToString() + ")"); - } + f.folderID = LLUUID.Random(); + modified = true; + } + + // ensure we have valid agent id + if (f.agentID == LLUUID.Zero) + { + if (parent != null) + f.agentID = parent.agentID; + else + f.agentID = f.folderID; + modified = true; + } + + if (f.parentID == LLUUID.Zero && parent != null) + { + f.parentID = parent.folderID; + modified = true; } - return null; + + + foreach (SerializableInventory.SerializableFolder child in f.SubFolders) + { + modified |= fixupFolder(child, f); + } + + return modified; } - public XmlRpcResponse XmlRpcInventoryRequest(XmlRpcRequest request) + protected static bool fixupInventory(SerializableInventory inventory) { - XmlRpcResponse response = new XmlRpcResponse(); - Hashtable requestData = (Hashtable)request.Params[0]; + return fixupFolder(inventory.root, null); + } - Hashtable responseData = new Hashtable(); + public class GetInventory : BaseStreamHandler + { + private SerializableInventory _inventory; + private InventoryManager _manager; + public GetInventory(InventoryManager manager) + : base("GET", "/inventory") + { + _manager = manager; - // Stuff happens here + _inventory = loadInventoryFromXmlFile("Inventory_Library.xml"); + if (fixupInventory(_inventory)) + { + FileStream fs = new FileStream("Inventory_Library.xml", FileMode.Truncate, FileAccess.Write); + saveInventoryToStream(_inventory, fs); + fs.Flush(); + fs.Close(); + MainLog.Instance.Debug(OpenInventory_Main.LogName, "Modified"); + } + } - if (requestData.ContainsKey("Access-type")) + private void CreateDefaultInventory(LLUUID userID) { - if (requestData["access-type"] == "rootfolders") + } + + private byte[] GetUserInventory(LLUUID userID) + { + MainLog.Instance.Notice(OpenInventory_Main.LogName, "Getting Inventory for user {0}", userID.ToStringHyphenated()); + byte[] result = new byte[] { }; + + InventoryFolderBase fb = _manager._databasePlugin.getUserRootFolder(userID); + if (fb == null) { -// responseData["rootfolders"] = + MainLog.Instance.Notice(OpenInventory_Main.LogName, "Inventory not found for user {0}, creating new", userID.ToStringHyphenated()); + CreateDefaultInventory(userID); } + + return result; } - else + + override public byte[] Handle(string path, Stream request) { - responseData["error"] = "No access-type specified."; - } + byte[] result = new byte[] { }; + string[] parms = path.Split(new char[] { '/' }, StringSplitOptions.RemoveEmptyEntries); + if (parms.Length >= 1) + { + if (string.Compare(parms[1], "library", true) == 0) + { - // Stuff stops happening here + MemoryStream ms = new MemoryStream(); + saveInventoryToStream(_inventory, ms); - response.Value = responseData; - return response; + result = ms.GetBuffer(); + Array.Resize(ref result, (int)ms.Length); + } + else if (string.Compare(parms[1], "user", true) == 0) + { + if (parms.Length >= 2) + { + result = GetUserInventory(new LLUUID(parms[2])); + } + } + } + return result; + } } + } } diff --git a/OpenSim/Grid/InventoryServer/Main.cs b/OpenSim/Grid/InventoryServer/Main.cs index 97addb0..1bc396b 100644 --- a/OpenSim/Grid/InventoryServer/Main.cs +++ b/OpenSim/Grid/InventoryServer/Main.cs @@ -31,48 +31,71 @@ using System.Collections.Generic; using System.Reflection; using System.IO; using System.Text; + using libsecondlife; -using OpenSim.Framework.User; -using OpenSim.Framework.Sims; -using OpenSim.Framework.Inventory; -using OpenSim.Framework.Interfaces; + +using OpenSim.Framework; using OpenSim.Framework.Console; -using OpenSim.Servers; -using OpenSim.Framework.Utilities; +using OpenSim.Framework.Servers; + +using InventoryManager = OpenSim.Grid.InventoryServer.InventoryManager; -namespace OpenGridServices.InventoryServer +namespace OpenSim.Grid.InventoryServer { - public class OpenInventory_Main : BaseServer, conscmd_callback + public class OpenInventory_Main : conscmd_callback { - ConsoleBase m_console; + LogBase m_console; InventoryManager m_inventoryManager; + InventoryConfig m_config; + public const string LogName = "INVENTORY"; + + [STAThread] public static void Main(string[] args) { + OpenInventory_Main theServer = new OpenInventory_Main(); + theServer.Startup(); + + theServer.Work(); } public OpenInventory_Main() { - m_console = new ConsoleBase("opengrid-inventory-console.log", "OpenInventory", this, false); - MainConsole.Instance = m_console; + m_console = new LogBase("opengrid-inventory-console.log", LogName, this, true); + MainLog.Instance = m_console; } public void Startup() { - MainConsole.Instance.Notice("Initialising inventory manager..."); + MainLog.Instance.Notice("Initialising inventory manager..."); + m_config = new InventoryConfig(LogName, (Path.Combine(Util.configDir(), "InventoryServer_Config.xml"))); + m_inventoryManager = new InventoryManager(); + m_inventoryManager.AddDatabasePlugin(m_config.DatabaseProvider); + MainLog.Instance.Notice(LogName, "Starting HTTP server ..."); + BaseHttpServer httpServer = new BaseHttpServer(m_config.HttpPort); - MainConsole.Instance.Notice("Starting HTTP server"); - BaseHttpServer httpServer = new BaseHttpServer(8004); + httpServer.AddStreamHandler(new InventoryManager.GetInventory(m_inventoryManager)); - httpServer.AddXmlRPCHandler("rootfolders", m_inventoryManager.XmlRpcInventoryRequest); - //httpServer.AddRestHandler("GET","/rootfolders/",Rest + httpServer.Start(); + MainLog.Instance.Notice(LogName, "Started HTTP server"); + } + + private void Work() + { + m_console.Notice("Enter help for a list of commands\n"); + + while (true) + { + m_console.MainLogPrompt(); + } } public void RunCmd(string cmd, string[] cmdparams) { switch (cmd) { + case "quit": case "shutdown": m_console.Close(); Environment.Exit(0); diff --git a/OpenSim/Region/Application/OpenSimMain.cs b/OpenSim/Region/Application/OpenSimMain.cs index 7245482..6be067f 100644 --- a/OpenSim/Region/Application/OpenSimMain.cs +++ b/OpenSim/Region/Application/OpenSimMain.cs @@ -175,6 +175,7 @@ namespace OpenSim config.Set("user_send_key", "null"); config.Set("user_recv_key", "null"); config.Set("asset_server_url", "http://127.0.0.1:" + AssetConfig.DefaultHttpPort.ToString()); + config.Set("inventory_server_url", "http://127.0.0.1:" + InventoryConfig.DefaultHttpPort.ToString()); } } @@ -774,4 +775,4 @@ namespace OpenSim #endregion } -} \ No newline at end of file +} diff --git a/OpenSim/Region/ClientStack/ClientView.ProcessPackets.cs b/OpenSim/Region/ClientStack/ClientView.ProcessPackets.cs index 7d1780c..89e5465 100644 --- a/OpenSim/Region/ClientStack/ClientView.ProcessPackets.cs +++ b/OpenSim/Region/ClientStack/ClientView.ProcessPackets.cs @@ -770,4 +770,4 @@ namespace OpenSim.Region.ClientStack OutPacket(logReply); } } -} \ No newline at end of file +} diff --git a/OpenSim/Region/Communications/OGS1/CommunicationsOGS1.cs b/OpenSim/Region/Communications/OGS1/CommunicationsOGS1.cs index 941cc30..1281a44 100644 --- a/OpenSim/Region/Communications/OGS1/CommunicationsOGS1.cs +++ b/OpenSim/Region/Communications/OGS1/CommunicationsOGS1.cs @@ -42,8 +42,8 @@ namespace OpenSim.Region.Communications.OGS1 m_gridService = gridInterComms; m_interRegion = gridInterComms; - m_inventoryService = new OGS1InventoryService(); + m_inventoryService = new OGS1InventoryService(serversInfo.InventoryURL); m_userService = new OGS1UserServices(this); } } -} \ No newline at end of file +} diff --git a/OpenSim/Region/Communications/OGS1/OGS1InventoryService.cs b/OpenSim/Region/Communications/OGS1/OGS1InventoryService.cs index 49fdee9..85df353 100644 --- a/OpenSim/Region/Communications/OGS1/OGS1InventoryService.cs +++ b/OpenSim/Region/Communications/OGS1/OGS1InventoryService.cs @@ -25,7 +25,9 @@ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * */ - +using System; +using System.IO; +using System.Xml.Serialization; using System.Collections.Generic; using libsecondlife; using OpenSim.Framework; @@ -36,8 +38,11 @@ namespace OpenSim.Region.Communications.OGS1 { public class OGS1InventoryService : IInventoryServices { - public OGS1InventoryService() + string _inventoryServerUrl; + + public OGS1InventoryService(string inventoryServerUrl) { + _inventoryServerUrl = inventoryServerUrl; } #region IInventoryServices Members @@ -45,6 +50,19 @@ namespace OpenSim.Region.Communications.OGS1 public void RequestInventoryForUser(LLUUID userID, InventoryFolderInfo folderCallBack, InventoryItemInfo itemCallBack) { + //TODO! Uncomment when all is done + //SerializableInventory userInventory = null; + + //RestClient inventoryServer = new RestClient(_inventoryServerUrl); + //inventoryServer.AddResourcePath("inventory"); + //inventoryServer.AddResourcePath("user"); + //inventoryServer.AddResourcePath(userID.ToStringHyphenated()); + + //using (Stream userInventoryStream = inventoryServer.Request()) + //{ + // XmlSerializer x = new XmlSerializer(typeof(SerializableInventory)); + // userInventory = (SerializableInventory)x.Deserialize(userInventoryStream); + //} } public void AddNewInventoryFolder(LLUUID userID, InventoryFolderImpl folder) @@ -70,4 +88,4 @@ namespace OpenSim.Region.Communications.OGS1 #endregion } -} \ No newline at end of file +} diff --git a/OpenSim/Region/Environment/Modules/ChatModule.cs b/OpenSim/Region/Environment/Modules/ChatModule.cs index 9d4187a..99b69e1 100644 --- a/OpenSim/Region/Environment/Modules/ChatModule.cs +++ b/OpenSim/Region/Environment/Modules/ChatModule.cs @@ -431,4 +431,4 @@ namespace OpenSim.Region.Environment.Modules m_tcp.Close(); } } -} \ No newline at end of file +} diff --git a/OpenSim/Region/Environment/Modules/WorldCommModule.cs b/OpenSim/Region/Environment/Modules/WorldCommModule.cs index 7a631d7..5af2ce3 100644 --- a/OpenSim/Region/Environment/Modules/WorldCommModule.cs +++ b/OpenSim/Region/Environment/Modules/WorldCommModule.cs @@ -490,4 +490,4 @@ namespace OpenSim.Region.Environment.Modules return m_id; } } -} \ No newline at end of file +} diff --git a/OpenSim/Region/Environment/Scenes/ScenePresence.cs b/OpenSim/Region/Environment/Scenes/ScenePresence.cs index 5155b41..3b73893 100644 --- a/OpenSim/Region/Environment/Scenes/ScenePresence.cs +++ b/OpenSim/Region/Environment/Scenes/ScenePresence.cs @@ -1059,4 +1059,4 @@ namespace OpenSim.Region.Environment.Scenes SendOurAppearance(m_controllingClient); } } -} \ No newline at end of file +} diff --git a/bin/Inventory_Default.xml b/bin/Inventory_Default.xml new file mode 100644 index 0000000..898e241 --- /dev/null +++ b/bin/Inventory_Default.xml @@ -0,0 +1,62 @@ + + + + My Inventory + + 9 + + + Animations + 20 + + + Body Parts + 13 + + + Clothing + 5 + + + Gestures + 21 + + + Landmarks + 3 + + + Lost And Found + 16 + + + Notecards + 7 + + + Objects + 6 + + + Photo Album + 15 + + + Scripts + 10 + + + Sounds + 1 + + + Textures + 0 + + + Trash + 14 + + + + diff --git a/bin/Inventory_Library.xml b/bin/Inventory_Library.xml new file mode 100644 index 0000000..327b8da --- /dev/null +++ b/bin/Inventory_Library.xml @@ -0,0 +1,144 @@ + + + + Library + + + + 9 + 0 + + + Animations + + + + 20 + 0 + + + + + Body Parts + + + + 13 + 0 + + + + + Clothing + + + + 5 + 0 + + + + + Gestures + + + + 21 + 0 + + + + + Landmarks + + + + 3 + 0 + + + + + Lost And Found + + + + 3 + 0 + + + + + Notecards + + + + 7 + 0 + + + + + Objects + + + + 6 + 0 + + + + + Photo Album + + + + 15 + 0 + + + + + Scripts + + + + 10 + 0 + + + + + Sounds + + + + 1 + 0 + + + + + Textures + + + + 0 + 0 + + + + + Accessories + + + + 8 + 0 + + + + + + + \ No newline at end of file diff --git a/prebuild.xml b/prebuild.xml index bc1cc78..79f8d3a 100644 --- a/prebuild.xml +++ b/prebuild.xml @@ -846,6 +846,34 @@ + + + + ../../../bin/ + + + + + ../../../bin/ + + + + ../../../bin/ + + + + + + + + + + + + + + + @@ -1054,3 +1082,4 @@ + -- cgit v1.1