From eeb5381bbbf58190824d047d845569564368c708 Mon Sep 17 00:00:00 2001
From: Dr Scofield
Date: Thu, 18 Sep 2008 15:50:52 +0000
Subject: and this actually adds the appearance code itself (and not just the
check-in message)
---
.../Rest/Inventory/RestAppearanceServices.cs | 826 +++++++++++++++++++++
OpenSim/ApplicationPlugins/Rest/rest.xsd | 276 +++++++
2 files changed, 1102 insertions(+)
create mode 100644 OpenSim/ApplicationPlugins/Rest/Inventory/RestAppearanceServices.cs
create mode 100644 OpenSim/ApplicationPlugins/Rest/rest.xsd
(limited to 'OpenSim')
diff --git a/OpenSim/ApplicationPlugins/Rest/Inventory/RestAppearanceServices.cs b/OpenSim/ApplicationPlugins/Rest/Inventory/RestAppearanceServices.cs
new file mode 100644
index 0000000..bc706f0
--- /dev/null
+++ b/OpenSim/ApplicationPlugins/Rest/Inventory/RestAppearanceServices.cs
@@ -0,0 +1,826 @@
+/*
+ * 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 OpenSim 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.Threading;
+using System.Xml;
+using System.Drawing;
+using OpenSim.Framework;
+using OpenSim.Framework.Servers;
+using OpenSim.Framework.Communications;
+using OpenSim.Framework.Communications.Cache;
+using OpenMetaverse;
+using OpenMetaverse.Imaging;
+using Nini.Config;
+
+namespace OpenSim.ApplicationPlugins.Rest.Inventory
+{
+ public class RestAppearanceServices : IRest
+ {
+ private static readonly int PARM_USERID = 0;
+ private static readonly int PARM_PATH = 1;
+
+ private bool enabled = false;
+ private string qPrefix = "appearance";
+
+ ///
+ /// The constructor makes sure that the service prefix is absolute
+ /// and the registers the service handler and the allocator.
+ ///
+
+ public RestAppearanceServices()
+ {
+ Rest.Log.InfoFormat("{0} User appearance services initializing", MsgId);
+ Rest.Log.InfoFormat("{0} Using REST Implementation Version {1}", MsgId, Rest.Version);
+
+ // If a relative path was specified for the handler's domain,
+ // add the standard prefix to make it absolute, e.g. /admin
+
+ if (!qPrefix.StartsWith(Rest.UrlPathSeparator))
+ {
+ Rest.Log.InfoFormat("{0} Domain is relative, adding absolute prefix", MsgId);
+ qPrefix = String.Format("{0}{1}{2}", Rest.Prefix, Rest.UrlPathSeparator, qPrefix);
+ Rest.Log.InfoFormat("{0} Domain is now <{1}>", MsgId, qPrefix);
+ }
+
+ // Register interface using the absolute URI.
+
+ Rest.Plugin.AddPathHandler(DoAppearance,qPrefix,Allocate);
+
+ // Activate if everything went OK
+
+ enabled = true;
+
+ Rest.Log.InfoFormat("{0} User appearance services initialization complete", MsgId);
+ }
+
+ ///
+ /// Post-construction, pre-enabled initialization opportunity
+ /// Not currently exploited.
+ ///
+
+ public void Initialize()
+ {
+ }
+
+ ///
+ /// Called by the plug-in to halt service processing. Local processing is
+ /// disabled.
+ ///
+
+ public void Close()
+ {
+ enabled = false;
+ Rest.Log.InfoFormat("{0} User appearance services closing down", MsgId);
+ }
+
+ ///
+ /// This property is declared locally because it is used a lot and
+ /// brevity is nice.
+ ///
+
+ internal string MsgId
+ {
+ get { return Rest.MsgId; }
+ }
+
+ #region Interface
+
+ ///
+ /// The plugin (RestHandler) calls this method to allocate the request
+ /// state carrier for a new request. It is destroyed when the request
+ /// completes. All request-instance specific state is kept here. This
+ /// is registered when this service provider is registered.
+ ///
+ /// Inbound HTTP request information
+ /// Outbound HTTP request information
+ /// REST service domain prefix
+ /// A RequestData instance suitable for this service
+
+ private RequestData Allocate(OSHttpRequest request, OSHttpResponse response, string prefix)
+ {
+ return (RequestData) new AppearanceRequestData(request, response, prefix);
+ }
+
+ ///
+ /// This method is registered with the handler when this service provider
+ /// is initialized. It is called whenever the plug-in identifies this service
+ /// provider as the best match for a given request.
+ /// It handles all aspects of inventory REST processing, i.e. /admin/inventory
+ ///
+ /// A consolidated HTTP request work area
+
+ private void DoAppearance(RequestData hdata)
+ {
+ AppearanceRequestData rdata = (AppearanceRequestData) hdata;
+
+ Rest.Log.DebugFormat("{0} DoAppearance ENTRY", MsgId);
+
+ // If we're disabled, do nothing.
+
+ if (!enabled)
+ {
+ return;
+ }
+
+ // Now that we know this is a serious attempt to
+ // access inventory data, we should find out who
+ // is asking, and make sure they are authorized
+ // to do so. We need to validate the caller's
+ // identity before revealing anything about the
+ // status quo. Authenticate throws an exception
+ // via Fail if no identity information is present.
+ //
+ // With the present HTTP server we can't use the
+ // builtin authentication mechanisms because they
+ // would be enforced for all in-bound requests.
+ // Instead we look at the headers ourselves and
+ // handle authentication directly.
+
+ try
+ {
+ if (!rdata.IsAuthenticated)
+ {
+ rdata.Fail(Rest.HttpStatusCodeNotAuthorized,String.Format("user \"{0}\" could not be authenticated", rdata.userName));
+ }
+ }
+ catch (RestException e)
+ {
+ if (e.statusCode == Rest.HttpStatusCodeNotAuthorized)
+ {
+ Rest.Log.WarnFormat("{0} User not authenticated", MsgId);
+ Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId, rdata.request.Headers.Get("Authorization"));
+ }
+ else
+ {
+ Rest.Log.ErrorFormat("{0} User authentication failed", MsgId);
+ Rest.Log.DebugFormat("{0} Authorization header: {1}", MsgId, rdata.request.Headers.Get("Authorization"));
+ }
+ throw (e);
+ }
+
+ Rest.Log.DebugFormat("{0} Authenticated {1}", MsgId, rdata.userName);
+
+ // We can only get here if we are authorized
+ //
+ // The requestor may have specified an UUID or
+ // a conjoined FirstName LastName string. We'll
+ // try both. If we fail with the first, UUID,
+ // attempt, we try the other. As an example, the
+ // URI for a valid inventory request might be:
+ //
+ // http://:/admin/inventory/Arthur Dent
+ //
+ // Indicating that this is an inventory request for
+ // an avatar named Arthur Dent. This is ALL that is
+ // required to designate a GET for an entire
+ // inventory.
+ //
+
+
+ // Do we have at least a user agent name?
+
+ if (rdata.Parameters.Length < 1)
+ {
+ Rest.Log.WarnFormat("{0} Appearance: No user agent identifier specified", MsgId);
+ rdata.Fail(Rest.HttpStatusCodeBadRequest, "no user identity specified");
+ }
+
+ // The first parameter MUST be the agent identification, either an UUID
+ // or a space-separated First-name Last-Name specification. We check for
+ // an UUID first, if anyone names their character using a valid UUID
+ // that identifies another existing avatar will cause this a problem...
+
+ try
+ {
+ rdata.uuid = new UUID(rdata.Parameters[PARM_USERID]);
+ Rest.Log.DebugFormat("{0} UUID supplied", MsgId);
+ rdata.userProfile = Rest.UserServices.GetUserProfile(rdata.uuid);
+ }
+ catch
+ {
+ string[] names = rdata.Parameters[PARM_USERID].Split(Rest.CA_SPACE);
+ if (names.Length == 2)
+ {
+ Rest.Log.DebugFormat("{0} Agent Name supplied [2]", MsgId);
+ rdata.userProfile = Rest.UserServices.GetUserProfile(names[0],names[1]);
+ }
+ else
+ {
+ Rest.Log.WarnFormat("{0} A Valid UUID or both first and last names must be specified", MsgId);
+ rdata.Fail(Rest.HttpStatusCodeBadRequest, "invalid user identity");
+ }
+ }
+
+ // If the user profile is null then either the server is broken, or the
+ // user is not known. We always assume the latter case.
+
+ if (rdata.userProfile != null)
+ {
+ Rest.Log.DebugFormat("{0} User profile obtained for agent {1} {2}",
+ MsgId, rdata.userProfile.FirstName, rdata.userProfile.SurName);
+ }
+ else
+ {
+ Rest.Log.WarnFormat("{0} No user profile for {1}", MsgId, rdata.path);
+ rdata.Fail(Rest.HttpStatusCodeNotFound, "unrecognized user identity");
+ }
+
+ // If we get to here, then we have effectively validated the user's
+
+ switch (rdata.method)
+ {
+ case Rest.HEAD : // Do the processing, set the status code, suppress entity
+ DoGet(rdata);
+ rdata.buffer = null;
+ break;
+
+ case Rest.GET : // Do the processing, set the status code, return entity
+ DoGet(rdata);
+ break;
+
+ case Rest.PUT : // Update named element
+ DoUpdate(rdata);
+ break;
+
+ case Rest.POST : // Add new information to identified context.
+ DoExtend(rdata);
+ break;
+
+ case Rest.DELETE : // Delete information
+ DoDelete(rdata);
+ break;
+
+ default :
+ Rest.Log.WarnFormat("{0} Method {1} not supported for {2}",
+ MsgId, rdata.method, rdata.path);
+ rdata.Fail(Rest.HttpStatusCodeMethodNotAllowed,
+ String.Format("{0} not supported", rdata.method));
+ break;
+ }
+ }
+
+ #endregion Interface
+
+ #region method-specific processing
+
+ ///
+ /// This method implements GET processing for user's appearance.
+ ///
+ /// HTTP service request work area
+
+ private void DoGet(AppearanceRequestData rdata)
+ {
+
+ rdata.userAppearance = Rest.AvatarServices.GetUserAppearance(rdata.userProfile.ID);
+
+ if (rdata.userAppearance == null)
+ {
+ rdata.Fail(Rest.HttpStatusCodeNoContent,"appearance data not found");
+ }
+
+ rdata.initXmlWriter();
+
+ FormatUserAppearance(rdata);
+
+ // Indicate a successful request
+
+ rdata.Complete();
+
+ // Send the response to the user. The body will be implicitly
+ // constructed from the result of the XML writer.
+
+ rdata.Respond(String.Format("Appearance {0} Normal completion", rdata.method));
+ }
+
+ ///
+ /// POST adds NEW information to the user profile database.
+ /// This effectively resets the appearance before applying those
+ /// characteristics supplied in the request.
+ ///
+
+ private void DoExtend(AppearanceRequestData rdata)
+ {
+ bool created = false;
+ bool modified = false;
+ string newnode = String.Empty;
+
+ AvatarAppearance old = Rest.AvatarServices.GetUserAppearance(rdata.userProfile.ID);
+ rdata.userAppearance = new AvatarAppearance();
+
+ rdata.userAppearance.Owner = old.Owner;
+
+ if (GetUserAppearance(rdata))
+ {
+ modified = rdata.userAppearance != null;
+ created = !modified;
+ Rest.AvatarServices.UpdateUserAppearance(rdata.userProfile.ID, rdata.userAppearance);
+ }
+
+ if (created)
+ {
+ newnode = String.Format("{0} {1}", rdata.userProfile.FirstName,
+ rdata.userProfile.SurName);
+ // Must include a location header with a URI that identifies the new resource.
+ rdata.AddHeader(Rest.HttpHeaderLocation,String.Format("http://{0}{1}:{2}{3}{4}",
+ rdata.hostname,rdata.port,rdata.path,Rest.UrlPathSeparator, newnode));
+ rdata.Complete(Rest.HttpStatusCodeCreated);
+ }
+ else
+ {
+ if (modified)
+ {
+ rdata.Complete(Rest.HttpStatusCodeOK);
+ }
+ else
+ {
+ rdata.Complete(Rest.HttpStatusCodeNoContent);
+ }
+ }
+
+ rdata.Respond(String.Format("Appearance {0} : Normal completion", rdata.method));
+
+ }
+
+ ///
+ /// This updates the user's appearance. not all aspects need to be provided,
+ /// only those supplied will be changed.
+ ///
+
+ private void DoUpdate(AppearanceRequestData rdata)
+ {
+
+ bool created = false;
+ bool modified = false;
+
+
+ rdata.userAppearance = Rest.AvatarServices.GetUserAppearance(rdata.userProfile.ID);
+
+ if (GetUserAppearance(rdata))
+ {
+ modified = rdata.userAppearance != null;
+ created = !modified;
+ Rest.AvatarServices.UpdateUserAppearance(rdata.userProfile.ID, rdata.userAppearance);
+ }
+
+ if (created)
+ {
+ rdata.Complete(Rest.HttpStatusCodeCreated);
+ }
+ else
+ {
+ if (modified)
+ {
+ rdata.Complete(Rest.HttpStatusCodeOK);
+ }
+ else
+ {
+ rdata.Complete(Rest.HttpStatusCodeNoContent);
+ }
+ }
+
+ rdata.Respond(String.Format("Appearance {0} : Normal completion", rdata.method));
+
+ }
+
+ ///
+ /// Delete the specified user's appearance. This actually performs a reset
+ /// to the default avatar appearance, if the info is already there.
+ /// Existing ownership is preserved. All prior updates are lost and can not
+ /// be recovered.
+ ///
+
+ private void DoDelete(AppearanceRequestData rdata)
+ {
+
+ AvatarAppearance old = Rest.AvatarServices.GetUserAppearance(rdata.userProfile.ID);
+
+ if (old != null)
+ {
+ rdata.userAppearance = new AvatarAppearance();
+
+ rdata.userAppearance.Owner = old.Owner;
+
+ Rest.AvatarServices.UpdateUserAppearance(rdata.userProfile.ID, rdata.userAppearance);
+
+ rdata.Complete();
+ }
+ else
+ {
+ rdata.Complete(Rest.HttpStatusCodeNoContent);
+ }
+
+ rdata.Respond(String.Format("Appearance {0} : Normal completion", rdata.method));
+
+ }
+
+#endregion method-specific processing
+
+ private bool GetUserAppearance(AppearanceRequestData rdata)
+ {
+
+ XmlReader xml;
+ bool indata = false;
+
+ rdata.initXmlReader();
+ xml = rdata.reader;
+
+ while (xml.Read())
+ {
+ switch (xml.NodeType)
+ {
+ case XmlNodeType.Element :
+ switch (xml.Name)
+ {
+ case "Appearance" :
+ if (xml.MoveToAttribute("Height"))
+ {
+ rdata.userAppearance.AvatarHeight = (float) Convert.ToDouble(xml.Value);
+ indata = true;
+ }
+ break;
+ case "Body" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.BodyItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.BodyAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Skin" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.SkinItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.SkinAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Hair" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.HairItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.HairAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Eyes" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.EyesItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.EyesAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Shirt" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.ShirtItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.ShirtAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Pants" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.PantsItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.PantsAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Shoes" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.ShoesItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.ShoesAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Socks" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.SocksItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.SocksAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Jacket" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.JacketItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.JacketAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Gloves" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.GlovesItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.GlovesAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "UnderShirt" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.UnderShirtItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.UnderShirtAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "UnderPants" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.UnderPantsItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.UnderPantsAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Skirt" :
+ if (xml.MoveToAttribute("Item"))
+ {
+ rdata.userAppearance.SkirtItem = xml.Value;
+ indata = true;
+ }
+ if (xml.MoveToAttribute("Asset"))
+ {
+ rdata.userAppearance.SkirtAsset = xml.Value;
+ indata = true;
+ }
+ break;
+ case "Attachment" :
+ {
+
+ int ap;
+ UUID asset;
+ UUID item;
+
+ if (xml.MoveToAttribute("AtPoint"))
+ {
+ ap = Convert.ToInt32(xml.Value);
+ if (xml.MoveToAttribute("Asset"))
+ {
+ asset = new UUID(xml.Value);
+ if (xml.MoveToAttribute("Asset"))
+ {
+ item = new UUID(xml.Value);
+ rdata.userAppearance.SetAttachment(ap, item, asset);
+ indata = true;
+ }
+ }
+ }
+ }
+ break;
+ case "Texture" :
+ if (xml.MoveToAttribute("Default"))
+ {
+ rdata.userAppearance.Texture = new Primitive.TextureEntry(new UUID(xml.Value));
+ indata = true;
+ }
+ break;
+ case "Face" :
+ {
+ uint index;
+ if (xml.MoveToAttribute("Index"))
+ {
+ index = Convert.ToUInt32(xml.Value);
+ if (xml.MoveToAttribute("Id"))
+ {
+ rdata.userAppearance.Texture.CreateFace(index).TextureID = new UUID(xml.Value);
+ indata = true;
+ }
+ }
+ }
+ break;
+ case "VisualParameters" :
+ {
+ xml.ReadContentAsBase64(rdata.userAppearance.VisualParams,
+ 0,rdata.userAppearance.VisualParams.Length);
+ indata = true;
+ }
+ break;
+ }
+ break;
+ }
+ }
+
+ return indata;
+
+ }
+
+ private void FormatUserAppearance(AppearanceRequestData rdata)
+ {
+
+ rdata.writer.WriteStartElement("Appearance");
+ rdata.writer.WriteAttributeString("Height", rdata.userAppearance.AvatarHeight.ToString());
+
+ rdata.writer.WriteStartElement("Body");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.BodyItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.BodyAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Skin");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.SkinItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.SkinAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Hair");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.HairItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.HairAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Eyes");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.EyesItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.EyesAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Shirt");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.ShirtItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.ShirtAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Pants");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.PantsItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.PantsAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Shoes");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.ShoesItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.ShoesAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Socks");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.SocksItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.SocksAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Jacket");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.JacketItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.JacketAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Gloves");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.GlovesItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.GlovesAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("UnderShirt");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.UnderShirtItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.UnderShirtAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("UnderPants");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.UnderPantsItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.UnderPantsAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ rdata.writer.WriteStartElement("Skirt");
+ rdata.writer.WriteAttributeString("Item",rdata.userAppearance.SkirtItem.ToString());
+ rdata.writer.WriteAttributeString("Asset",rdata.userAppearance.SkirtAsset.ToString());
+ rdata.writer.WriteEndElement();
+
+ Hashtable attachments = rdata.userAppearance.GetAttachments();
+
+ rdata.writer.WriteStartElement("Attachments");
+
+ for (int i=0; i
+ /// These are the inventory specific request/response state
+ /// extensions.
+ ///
+
+ internal UUID uuid = UUID.Zero;
+ internal UserProfileData userProfile = null;
+ internal AvatarAppearance userAppearance = null;
+
+ internal AppearanceRequestData(OSHttpRequest request, OSHttpResponse response, string prefix)
+ : base(request, response, prefix)
+ {
+ }
+
+ }
+
+ #endregion Appearance RequestData extension
+
+ }
+}
diff --git a/OpenSim/ApplicationPlugins/Rest/rest.xsd b/OpenSim/ApplicationPlugins/Rest/rest.xsd
new file mode 100644
index 0000000..4dc0ae4
--- /dev/null
+++ b/OpenSim/ApplicationPlugins/Rest/rest.xsd
@@ -0,0 +1,276 @@
+
+
+
+
+ Open Simulator Export/Import XML schema
+ August 2008
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
--
cgit v1.1