Filling in the blanks: The "meat" of the REST console
parent
bf8e07606f
commit
002940dd5d
|
@ -26,6 +26,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
|
using System.Xml;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
|
@ -369,6 +370,65 @@ namespace OpenSim.Framework.Console
|
||||||
|
|
||||||
return new string[0];
|
return new string[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public XmlElement GetXml(XmlDocument doc)
|
||||||
|
{
|
||||||
|
XmlElement root = doc.CreateElement("", "HelpTree", "");
|
||||||
|
|
||||||
|
ProcessTreeLevel(tree, root, doc);
|
||||||
|
|
||||||
|
return root;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ProcessTreeLevel(Dictionary<string, object> level, XmlElement xml, XmlDocument doc)
|
||||||
|
{
|
||||||
|
foreach (KeyValuePair<string, object> kvp in level)
|
||||||
|
{
|
||||||
|
if (kvp.Value is Dictionary<string, Object>)
|
||||||
|
{
|
||||||
|
XmlElement next = doc.CreateElement("", "Level", "");
|
||||||
|
next.SetAttribute("Name", kvp.Key);
|
||||||
|
|
||||||
|
xml.AppendChild(next);
|
||||||
|
|
||||||
|
ProcessTreeLevel((Dictionary<string, object>)kvp.Value, next, doc);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CommandInfo c = (CommandInfo)kvp.Value;
|
||||||
|
|
||||||
|
XmlElement cmd = doc.CreateElement("", "Command", "");
|
||||||
|
|
||||||
|
XmlElement e;
|
||||||
|
|
||||||
|
e = doc.CreateElement("", "Module", "");
|
||||||
|
cmd.AppendChild(e);
|
||||||
|
e.AppendChild(doc.CreateTextNode(c.module));
|
||||||
|
|
||||||
|
e = doc.CreateElement("", "Shared", "");
|
||||||
|
cmd.AppendChild(e);
|
||||||
|
e.AppendChild(doc.CreateTextNode(c.shared.ToString()));
|
||||||
|
|
||||||
|
e = doc.CreateElement("", "HelpText", "");
|
||||||
|
cmd.AppendChild(e);
|
||||||
|
e.AppendChild(doc.CreateTextNode(c.help_text));
|
||||||
|
|
||||||
|
e = doc.CreateElement("", "LongHelp", "");
|
||||||
|
cmd.AppendChild(e);
|
||||||
|
e.AppendChild(doc.CreateTextNode(c.long_help));
|
||||||
|
|
||||||
|
e = doc.CreateElement("", "Description", "");
|
||||||
|
cmd.AppendChild(e);
|
||||||
|
e.AppendChild(doc.CreateTextNode(c.descriptive_help));
|
||||||
|
|
||||||
|
xml.AppendChild(cmd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void FromXml(XmlElement root)
|
||||||
|
{
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class Parser
|
public class Parser
|
||||||
|
|
|
@ -26,30 +26,43 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
|
using System.Xml;
|
||||||
|
using System.Collections;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
|
using OpenMetaverse;
|
||||||
using Nini.Config;
|
using Nini.Config;
|
||||||
using OpenSim.Framework.Servers.HttpServer;
|
using OpenSim.Framework.Servers.HttpServer;
|
||||||
using log4net;
|
using log4net;
|
||||||
|
|
||||||
namespace OpenSim.Framework.Console
|
namespace OpenSim.Framework.Console
|
||||||
{
|
{
|
||||||
|
public class ConsoleConnection
|
||||||
|
{
|
||||||
|
public int last;
|
||||||
|
public long lastLineSeen;
|
||||||
|
}
|
||||||
|
|
||||||
// A console that uses REST interfaces
|
// A console that uses REST interfaces
|
||||||
//
|
//
|
||||||
public class RemoteConsole : CommandConsole
|
public class RemoteConsole : CommandConsole
|
||||||
{
|
{
|
||||||
// private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
||||||
|
|
||||||
// private IHttpServer m_Server = null;
|
private IHttpServer m_Server = null;
|
||||||
// private IConfigSource m_Config = null;
|
private IConfigSource m_Config = null;
|
||||||
|
|
||||||
private List<string> m_Scrollback = new List<string>();
|
private List<string> m_Scrollback = new List<string>();
|
||||||
private ManualResetEvent m_DataEvent = new ManualResetEvent(false);
|
private ManualResetEvent m_DataEvent = new ManualResetEvent(false);
|
||||||
private List<string> m_InputData = new List<string>();
|
private List<string> m_InputData = new List<string>();
|
||||||
private uint m_LineNumber = 1;
|
private long m_LineNumber = 0;
|
||||||
|
private Dictionary<UUID, ConsoleConnection> m_Connections =
|
||||||
|
new Dictionary<UUID, ConsoleConnection>();
|
||||||
|
private string m_UserName = String.Empty;
|
||||||
|
private string m_Password = String.Empty;
|
||||||
|
|
||||||
public RemoteConsole(string defaultPrompt) : base(defaultPrompt)
|
public RemoteConsole(string defaultPrompt) : base(defaultPrompt)
|
||||||
{
|
{
|
||||||
|
@ -57,12 +70,23 @@ namespace OpenSim.Framework.Console
|
||||||
|
|
||||||
public void ReadConfig(IConfigSource config)
|
public void ReadConfig(IConfigSource config)
|
||||||
{
|
{
|
||||||
// m_Config = config;
|
m_Config = config;
|
||||||
|
|
||||||
|
IConfig netConfig = m_Config.Configs["Network"];
|
||||||
|
if (netConfig == null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_UserName = netConfig.GetString("ConsoleUser", String.Empty);
|
||||||
|
m_Password = netConfig.GetString("ConsolePass", String.Empty);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SetServer(IHttpServer server)
|
public void SetServer(IHttpServer server)
|
||||||
{
|
{
|
||||||
// m_Server = server;
|
m_Server = server;
|
||||||
|
|
||||||
|
m_Server.AddHTTPHandler("/StartSession/", HandleHttpStartSession);
|
||||||
|
m_Server.AddHTTPHandler("/CloseSession/", HandleHttpCloseSession);
|
||||||
|
m_Server.AddHTTPHandler("/SessionCommand/", HandleHttpSessionCommand);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void Output(string text, string level)
|
public override void Output(string text, string level)
|
||||||
|
@ -71,16 +95,14 @@ namespace OpenSim.Framework.Console
|
||||||
{
|
{
|
||||||
while (m_Scrollback.Count >= 1000)
|
while (m_Scrollback.Count >= 1000)
|
||||||
m_Scrollback.RemoveAt(0);
|
m_Scrollback.RemoveAt(0);
|
||||||
m_Scrollback.Add(String.Format("{0}", m_LineNumber)+":"+level+":"+text);
|
|
||||||
m_LineNumber++;
|
m_LineNumber++;
|
||||||
|
m_Scrollback.Add(String.Format("{0}", m_LineNumber)+":"+level+":"+text);
|
||||||
}
|
}
|
||||||
System.Console.Write(text);
|
System.Console.Write(text);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override string ReadLine(string p, bool isCommand, bool e)
|
public override string ReadLine(string p, bool isCommand, bool e)
|
||||||
{
|
{
|
||||||
System.Console.Write("{0}", prompt);
|
|
||||||
|
|
||||||
m_DataEvent.WaitOne();
|
m_DataEvent.WaitOne();
|
||||||
|
|
||||||
lock (m_InputData)
|
lock (m_InputData)
|
||||||
|
@ -115,5 +137,299 @@ namespace OpenSim.Framework.Console
|
||||||
return cmdinput;
|
return cmdinput;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void DoExpire()
|
||||||
|
{
|
||||||
|
List<UUID> expired = new List<UUID>();
|
||||||
|
|
||||||
|
lock (m_Connections)
|
||||||
|
{
|
||||||
|
foreach (KeyValuePair<UUID, ConsoleConnection> kvp in m_Connections)
|
||||||
|
{
|
||||||
|
if (System.Environment.TickCount - kvp.Value.last > 500000)
|
||||||
|
expired.Add(kvp.Key);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (UUID id in expired)
|
||||||
|
{
|
||||||
|
System.Console.WriteLine("Expired {0}", id.ToString());
|
||||||
|
CloseConnection(id);
|
||||||
|
m_Connections.Remove(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Hashtable HandleHttpStartSession(Hashtable request)
|
||||||
|
{
|
||||||
|
DoExpire();
|
||||||
|
|
||||||
|
Hashtable post = DecodePostString(request["body"].ToString());
|
||||||
|
Hashtable reply = new Hashtable();
|
||||||
|
|
||||||
|
reply["str_response_string"] = "";
|
||||||
|
reply["int_response_code"] = 401;
|
||||||
|
reply["content_type"] = "text/plain";
|
||||||
|
|
||||||
|
if (m_UserName == String.Empty)
|
||||||
|
return reply;
|
||||||
|
|
||||||
|
if (post["USER"] == null || post["PASS"] == null)
|
||||||
|
return reply;
|
||||||
|
|
||||||
|
if (m_UserName != post["USER"].ToString() ||
|
||||||
|
m_Password != post["PASS"].ToString())
|
||||||
|
{
|
||||||
|
return reply;
|
||||||
|
}
|
||||||
|
|
||||||
|
ConsoleConnection c = new ConsoleConnection();
|
||||||
|
c.last = System.Environment.TickCount;
|
||||||
|
c.lastLineSeen = 0;
|
||||||
|
|
||||||
|
UUID sessionID = UUID.Random();
|
||||||
|
|
||||||
|
lock (m_Connections)
|
||||||
|
{
|
||||||
|
m_Connections[sessionID] = c;
|
||||||
|
}
|
||||||
|
|
||||||
|
string uri = "/ReadResponses/" + sessionID.ToString() + "/";
|
||||||
|
|
||||||
|
m_Server.AddPollServiceHTTPHandler(uri, HandleHttpCloseSession,
|
||||||
|
new PollServiceEventArgs(HasEvents, GetEvents, NoEvents,
|
||||||
|
sessionID));
|
||||||
|
|
||||||
|
XmlDocument xmldoc = new XmlDocument();
|
||||||
|
XmlNode xmlnode = xmldoc.CreateNode(XmlNodeType.XmlDeclaration,
|
||||||
|
"", "");
|
||||||
|
|
||||||
|
xmldoc.AppendChild(xmlnode);
|
||||||
|
XmlElement rootElement = xmldoc.CreateElement("", "ConsoleSession",
|
||||||
|
"");
|
||||||
|
|
||||||
|
xmldoc.AppendChild(rootElement);
|
||||||
|
|
||||||
|
XmlElement id = xmldoc.CreateElement("", "SessionID", "");
|
||||||
|
id.AppendChild(xmldoc.CreateTextNode(sessionID.ToString()));
|
||||||
|
|
||||||
|
rootElement.AppendChild(id);
|
||||||
|
rootElement.AppendChild(MainConsole.Instance.Commands.GetXml(xmldoc));
|
||||||
|
|
||||||
|
reply["str_response_string"] = xmldoc.InnerXml;
|
||||||
|
reply["int_response_code"] = 200;
|
||||||
|
reply["content_type"] = "text/xml";
|
||||||
|
|
||||||
|
return reply;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Hashtable HandleHttpCloseSession(Hashtable request)
|
||||||
|
{
|
||||||
|
DoExpire();
|
||||||
|
|
||||||
|
Hashtable post = DecodePostString(request["body"].ToString());
|
||||||
|
Hashtable reply = new Hashtable();
|
||||||
|
|
||||||
|
reply["str_response_string"] = "";
|
||||||
|
reply["int_response_code"] = 404;
|
||||||
|
reply["content_type"] = "text/plain";
|
||||||
|
|
||||||
|
if (post["ID"] == null)
|
||||||
|
return reply;
|
||||||
|
|
||||||
|
UUID id;
|
||||||
|
if (!UUID.TryParse(post["ID"].ToString(), out id))
|
||||||
|
return reply;
|
||||||
|
|
||||||
|
lock (m_Connections)
|
||||||
|
{
|
||||||
|
if (m_Connections.ContainsKey(id))
|
||||||
|
{
|
||||||
|
CloseConnection(id);
|
||||||
|
m_Connections.Remove(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
XmlDocument xmldoc = new XmlDocument();
|
||||||
|
XmlNode xmlnode = xmldoc.CreateNode(XmlNodeType.XmlDeclaration,
|
||||||
|
"", "");
|
||||||
|
|
||||||
|
xmldoc.AppendChild(xmlnode);
|
||||||
|
XmlElement rootElement = xmldoc.CreateElement("", "ConsoleSession",
|
||||||
|
"");
|
||||||
|
|
||||||
|
xmldoc.AppendChild(rootElement);
|
||||||
|
|
||||||
|
XmlElement res = xmldoc.CreateElement("", "Result", "");
|
||||||
|
res.AppendChild(xmldoc.CreateTextNode("OK"));
|
||||||
|
|
||||||
|
rootElement.AppendChild(res);
|
||||||
|
|
||||||
|
reply["str_response_string"] = xmldoc.InnerXml;
|
||||||
|
reply["int_response_code"] = 200;
|
||||||
|
reply["content_type"] = "text/plain";
|
||||||
|
|
||||||
|
return reply;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Hashtable HandleHttpSessionCommand(Hashtable request)
|
||||||
|
{
|
||||||
|
DoExpire();
|
||||||
|
|
||||||
|
Hashtable post = DecodePostString(request["body"].ToString());
|
||||||
|
Hashtable reply = new Hashtable();
|
||||||
|
|
||||||
|
reply["str_response_string"] = "";
|
||||||
|
reply["int_response_code"] = 404;
|
||||||
|
reply["content_type"] = "text/plain";
|
||||||
|
|
||||||
|
if (post["ID"] == null)
|
||||||
|
return reply;
|
||||||
|
|
||||||
|
UUID id;
|
||||||
|
if (!UUID.TryParse(post["ID"].ToString(), out id))
|
||||||
|
return reply;
|
||||||
|
|
||||||
|
if (post["COMMAND"] == null || post["COMMAND"].ToString() == String.Empty)
|
||||||
|
return reply;
|
||||||
|
|
||||||
|
lock (m_InputData)
|
||||||
|
{
|
||||||
|
m_DataEvent.Set();
|
||||||
|
m_InputData.Add(post["COMMAND"].ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
XmlDocument xmldoc = new XmlDocument();
|
||||||
|
XmlNode xmlnode = xmldoc.CreateNode(XmlNodeType.XmlDeclaration,
|
||||||
|
"", "");
|
||||||
|
|
||||||
|
xmldoc.AppendChild(xmlnode);
|
||||||
|
XmlElement rootElement = xmldoc.CreateElement("", "ConsoleSession",
|
||||||
|
"");
|
||||||
|
|
||||||
|
xmldoc.AppendChild(rootElement);
|
||||||
|
|
||||||
|
XmlElement res = xmldoc.CreateElement("", "Result", "");
|
||||||
|
res.AppendChild(xmldoc.CreateTextNode("OK"));
|
||||||
|
|
||||||
|
rootElement.AppendChild(res);
|
||||||
|
|
||||||
|
reply["str_response_string"] = xmldoc.InnerXml;
|
||||||
|
reply["int_response_code"] = 200;
|
||||||
|
reply["content_type"] = "text/plain";
|
||||||
|
|
||||||
|
return reply;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Hashtable DecodePostString(string data)
|
||||||
|
{
|
||||||
|
Hashtable result = new Hashtable();
|
||||||
|
|
||||||
|
string[] terms = data.Split(new char[] {'&'});
|
||||||
|
|
||||||
|
foreach (string term in terms)
|
||||||
|
{
|
||||||
|
string[] elems = term.Split(new char[] {'='});
|
||||||
|
if (elems.Length == 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
string name = System.Web.HttpUtility.UrlDecode(elems[0]);
|
||||||
|
string value = String.Empty;
|
||||||
|
|
||||||
|
if (elems.Length > 1)
|
||||||
|
value = System.Web.HttpUtility.UrlDecode(elems[1]);
|
||||||
|
|
||||||
|
result[name] = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void CloseConnection(UUID id)
|
||||||
|
{
|
||||||
|
string uri = "/ReadResponses/" + id.ToString() + "/";
|
||||||
|
|
||||||
|
m_Server.RemovePollServiceHTTPHandler("", uri);
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool HasEvents(UUID sessionID)
|
||||||
|
{
|
||||||
|
ConsoleConnection c = null;
|
||||||
|
|
||||||
|
lock (m_Connections)
|
||||||
|
{
|
||||||
|
if (!m_Connections.ContainsKey(sessionID))
|
||||||
|
return false;
|
||||||
|
c = m_Connections[sessionID];
|
||||||
|
}
|
||||||
|
c.last = System.Environment.TickCount;
|
||||||
|
if (c.lastLineSeen < m_LineNumber)
|
||||||
|
return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Hashtable GetEvents(UUID sessionID, string request)
|
||||||
|
{
|
||||||
|
ConsoleConnection c = null;
|
||||||
|
|
||||||
|
lock (m_Connections)
|
||||||
|
{
|
||||||
|
if (!m_Connections.ContainsKey(sessionID))
|
||||||
|
return NoEvents();
|
||||||
|
c = m_Connections[sessionID];
|
||||||
|
}
|
||||||
|
c.last = System.Environment.TickCount;
|
||||||
|
if (c.lastLineSeen >= m_LineNumber)
|
||||||
|
return NoEvents();
|
||||||
|
|
||||||
|
Hashtable result = new Hashtable();
|
||||||
|
|
||||||
|
XmlDocument xmldoc = new XmlDocument();
|
||||||
|
XmlNode xmlnode = xmldoc.CreateNode(XmlNodeType.XmlDeclaration,
|
||||||
|
"", "");
|
||||||
|
|
||||||
|
xmldoc.AppendChild(xmlnode);
|
||||||
|
XmlElement rootElement = xmldoc.CreateElement("", "ConsoleSession",
|
||||||
|
"");
|
||||||
|
|
||||||
|
lock (m_Scrollback)
|
||||||
|
{
|
||||||
|
long startLine = m_LineNumber - m_Scrollback.Count;
|
||||||
|
|
||||||
|
for (long i = startLine ; i < m_LineNumber ; i++)
|
||||||
|
{
|
||||||
|
XmlElement res = xmldoc.CreateElement("", "Line", "");
|
||||||
|
long line = i + 1;
|
||||||
|
res.SetAttribute("Number", line.ToString());
|
||||||
|
res.AppendChild(xmldoc.CreateTextNode(m_Scrollback[(int)(i - startLine)]));
|
||||||
|
|
||||||
|
rootElement.AppendChild(res);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
c.lastLineSeen = m_LineNumber;
|
||||||
|
|
||||||
|
xmldoc.AppendChild(rootElement);
|
||||||
|
|
||||||
|
result["str_response_string"] = xmldoc.InnerXml;
|
||||||
|
result["int_response_code"] = 200;
|
||||||
|
result["content_type"] = "application/xml";
|
||||||
|
result["keepalive"] = false;
|
||||||
|
result["reusecontext"] = false;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Hashtable NoEvents()
|
||||||
|
{
|
||||||
|
Hashtable result = new Hashtable();
|
||||||
|
|
||||||
|
result["int_response_code"] = 502;
|
||||||
|
result["content_type"] = "text/plain";
|
||||||
|
result["keepalive"] = false;
|
||||||
|
result["reusecontext"] = false;
|
||||||
|
result["str_response_string"] = "Upstream error: ";
|
||||||
|
result["error_status_text"] = "Upstream error:";
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -77,16 +77,16 @@ namespace OpenSim.Server.Base
|
||||||
m_HttpServer = new BaseHttpServer(port);
|
m_HttpServer = new BaseHttpServer(port);
|
||||||
|
|
||||||
MainServer.Instance = m_HttpServer;
|
MainServer.Instance = m_HttpServer;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Initialise()
|
||||||
|
{
|
||||||
|
m_HttpServer.Start();
|
||||||
|
|
||||||
if (MainConsole.Instance is RemoteConsole)
|
if (MainConsole.Instance is RemoteConsole)
|
||||||
{
|
{
|
||||||
((RemoteConsole)MainConsole.Instance).SetServer(m_HttpServer);
|
((RemoteConsole)MainConsole.Instance).SetServer(m_HttpServer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void Initialise()
|
|
||||||
{
|
|
||||||
m_HttpServer.Start();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -105,9 +105,11 @@
|
||||||
|
|
||||||
<ReferencePath>../../../bin/</ReferencePath>
|
<ReferencePath>../../../bin/</ReferencePath>
|
||||||
<Reference name="System"/>
|
<Reference name="System"/>
|
||||||
|
<Reference name="System.Web"/>
|
||||||
<Reference name="log4net.dll"/>
|
<Reference name="log4net.dll"/>
|
||||||
<Reference name="Nini.dll"/>
|
<Reference name="Nini.dll"/>
|
||||||
<Reference name="OpenSim.Framework.Servers.HttpServer"/>
|
<Reference name="OpenSim.Framework.Servers.HttpServer"/>
|
||||||
|
<Reference name="OpenMetaverseTypes.dll"/>
|
||||||
<Files>
|
<Files>
|
||||||
<Match pattern="*.cs" recurse="true"/>
|
<Match pattern="*.cs" recurse="true"/>
|
||||||
</Files>
|
</Files>
|
||||||
|
|
Loading…
Reference in New Issue