From 27a0b3ecbdcf248b331742c7b2771d2a87dc8c3a Mon Sep 17 00:00:00 2001 From: teravus Date: Sun, 3 Feb 2013 06:49:17 -0500 Subject: [PATCH 1/8] Commit 1 in of this branch feature. This is one of many... --- .../TCPJSONStream/ClientAcceptedEventArgs.cs | 50 ++++++ .../TCPJSONStream/ClientNetworkContext.cs | 143 +++++++++++++++ .../TCPJSONStream/DisconnectedEventArgs.cs | 17 ++ .../TCPJSONStream/OpenSimWebSocketBase.cs | 73 ++++++++ .../TCPJSONStream/TCPJsonWebSocketServer.cs | 163 ++++++++++++++++++ prebuild.xml | 45 +++++ 6 files changed, 491 insertions(+) create mode 100644 OpenSim/Region/ClientStack/TCPJSONStream/ClientAcceptedEventArgs.cs create mode 100644 OpenSim/Region/ClientStack/TCPJSONStream/ClientNetworkContext.cs create mode 100644 OpenSim/Region/ClientStack/TCPJSONStream/DisconnectedEventArgs.cs create mode 100644 OpenSim/Region/ClientStack/TCPJSONStream/OpenSimWebSocketBase.cs create mode 100644 OpenSim/Region/ClientStack/TCPJSONStream/TCPJsonWebSocketServer.cs diff --git a/OpenSim/Region/ClientStack/TCPJSONStream/ClientAcceptedEventArgs.cs b/OpenSim/Region/ClientStack/TCPJSONStream/ClientAcceptedEventArgs.cs new file mode 100644 index 0000000000..a58eab13cc --- /dev/null +++ b/OpenSim/Region/ClientStack/TCPJSONStream/ClientAcceptedEventArgs.cs @@ -0,0 +1,50 @@ +using System; +using System.Net.Sockets; + +namespace OpenSim.Region.ClientStack.TCPJSONStream +{ + /// + /// Invoked when a client have been accepted by the + /// + /// + /// Can be used to revoke incoming connections + /// + public class ClientAcceptedEventArgs : EventArgs + { + private readonly Socket _socket; + private bool _revoke; + + /// + /// Initializes a new instance of the class. + /// + /// The socket. + public ClientAcceptedEventArgs(Socket socket) + { + _socket = socket; + } + + /// + /// Accepted socket. + /// + public Socket Socket + { + get { return _socket; } + } + + /// + /// Client should be revoked. + /// + public bool Revoked + { + get { return _revoke; } + } + + /// + /// Client may not be handled. + /// + public void Revoke() + { + _revoke = true; + } + } +} \ No newline at end of file diff --git a/OpenSim/Region/ClientStack/TCPJSONStream/ClientNetworkContext.cs b/OpenSim/Region/ClientStack/TCPJSONStream/ClientNetworkContext.cs new file mode 100644 index 0000000000..591f817b8f --- /dev/null +++ b/OpenSim/Region/ClientStack/TCPJSONStream/ClientNetworkContext.cs @@ -0,0 +1,143 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Net; +using System.Net.Sockets; +using System.Text; + +namespace OpenSim.Region.ClientStack.TCPJSONStream +{ + public class ClientNetworkContext + { + private Socket _socket; + private string _remoteAddress; + private string _remotePort; + private WebSocketConnectionStage _wsConnectionStatus = WebSocketConnectionStage.Accept; + private int _bytesLeft; + private NetworkStream _stream; + private byte[] _buffer; + public event EventHandler Disconnected = delegate { }; + + public ClientNetworkContext(IPEndPoint endPoint, int port, Stream stream, int buffersize, Socket sock) + { + _socket = sock; + _remoteAddress = endPoint.Address.ToString(); + _remotePort = port.ToString(); + _stream = stream as NetworkStream; + _buffer = new byte[buffersize]; + + + } + + public void BeginRead() + { + _wsConnectionStatus = WebSocketConnectionStage.Http; + try + { + _stream.BeginRead(_buffer, 0, _buffer.Length, OnReceive, _wsConnectionStatus); + } + catch (IOException err) + { + //m_log.Debug(err.ToString()); + } + } + + private void OnReceive(IAsyncResult ar) + { + try + { + int bytesRead = _stream.EndRead(ar); + if (bytesRead == 0) + { + + Disconnected(this, new DisconnectedEventArgs(SocketError.ConnectionReset)); + return; + } + + } + } + /// + /// send a whole buffer + /// + /// buffer to send + /// + public void Send(byte[] buffer) + { + if (buffer == null) + throw new ArgumentNullException("buffer"); + Send(buffer, 0, buffer.Length); + } + + /// + /// Send data using the stream + /// + /// Contains data to send + /// Start position in buffer + /// number of bytes to send + /// + /// + public void Send(byte[] buffer, int offset, int size) + { + + if (offset + size > buffer.Length) + throw new ArgumentOutOfRangeException("offset", offset, "offset + size is beyond end of buffer."); + + if (_stream != null && _stream.CanWrite) + { + try + { + _stream.Write(buffer, offset, size); + } + catch (IOException) + { + + } + } + + } + private void Reset() + { + if (_stream == null) + return; + _stream.Dispose(); + _stream = null; + if (_socket == null) + return; + if (_socket.Connected) + _socket.Disconnect(true); + _socket = null; + } + } + + public enum WebSocketConnectionStage + { + Reuse, + Accept, + Http, + WebSocket, + Closed + } + + public enum FrameOpCodesRFC6455 + { + Continue = 0x0, + Text = 0x1, + Binary = 0x2, + Close = 0x8, + Ping = 0x9, + Pong = 0xA + } + + public enum DataState + { + Empty = 0, + Waiting = 1, + Receiving = 2, + Complete = 3, + Closed = 4, + Ping = 5, + Pong = 6 + } + + +} diff --git a/OpenSim/Region/ClientStack/TCPJSONStream/DisconnectedEventArgs.cs b/OpenSim/Region/ClientStack/TCPJSONStream/DisconnectedEventArgs.cs new file mode 100644 index 0000000000..32880cc399 --- /dev/null +++ b/OpenSim/Region/ClientStack/TCPJSONStream/DisconnectedEventArgs.cs @@ -0,0 +1,17 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Net.Sockets; +using System.Text; + +namespace OpenSim.Region.ClientStack.TCPJSONStream +{ + public class DisconnectedEventArgs:EventArgs + { + public SocketError Error { get; private set; } + public DisconnectedEventArgs(SocketError err) + { + Error = err; + } + } +} diff --git a/OpenSim/Region/ClientStack/TCPJSONStream/OpenSimWebSocketBase.cs b/OpenSim/Region/ClientStack/TCPJSONStream/OpenSimWebSocketBase.cs new file mode 100644 index 0000000000..379438d384 --- /dev/null +++ b/OpenSim/Region/ClientStack/TCPJSONStream/OpenSimWebSocketBase.cs @@ -0,0 +1,73 @@ +/* + * 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.Net; +using Nini.Config; +using OpenSim.Framework; + +namespace OpenSim.Region.ClientStack.TCPJSONStream +{ + public sealed class TCPJsonWebSocketBase : IClientNetworkServer + { + private TCPJsonWebSocketServer m_tcpServer; + + public TCPJsonWebSocketBase() + { + } + + public void Initialise(IPAddress _listenIP, ref uint port, int proxyPortOffsetParm, bool allow_alternate_port, IConfigSource configSource, AgentCircuitManager authenticateClass) + { + m_tcpServer = new TCPJsonWebSocketServer(_listenIP,ref port, proxyPortOffsetParm, allow_alternate_port,configSource,authenticateClass); + } + + public void NetworkStop() + { + m_tcpServer.Stop(); + } + + public bool HandlesRegion(Location x) + { + return m_tcpServer.HandlesRegion(x); + } + + public void AddScene(IScene x) + { + m_tcpServer.AddScene(x); + } + + public void Start() + { + m_tcpServer.Start(); + } + + public void Stop() + { + m_tcpServer.Stop(); + } + } +} \ No newline at end of file diff --git a/OpenSim/Region/ClientStack/TCPJSONStream/TCPJsonWebSocketServer.cs b/OpenSim/Region/ClientStack/TCPJSONStream/TCPJsonWebSocketServer.cs new file mode 100644 index 0000000000..0713bf4403 --- /dev/null +++ b/OpenSim/Region/ClientStack/TCPJSONStream/TCPJsonWebSocketServer.cs @@ -0,0 +1,163 @@ +using System; +using System.Collections.Generic; +using System.Net; +using System.Net.Sockets; +using System.Reflection; +using System.Text; +using System.Threading; +using Nini.Config; +using OpenSim.Framework; +using OpenSim.Region.Framework.Scenes; +using log4net; + +namespace OpenSim.Region.ClientStack.TCPJSONStream +{ + public delegate void ExceptionHandler(object source, Exception exception); + + public class TCPJsonWebSocketServer + { + private readonly IPAddress _address; + private readonly int _port; + private readonly ManualResetEvent _shutdownEvent = new ManualResetEvent(false); + private TcpListener _listener; + private int _pendingAccepts; + private bool _shutdown; + private int _backlogAcceptQueueLength = 5; + private Scene m_scene; + private Location m_location; + private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); + + public event EventHandler Accepted = delegate { }; + + + public TCPJsonWebSocketServer(IPAddress _listenIP, ref uint port, int proxyPortOffsetParm, + bool allow_alternate_port, IConfigSource configSource, + AgentCircuitManager authenticateClass) + { + _address = _listenIP; + _port = (int)port; //Why is a uint passed in? + } + public void Stop() + { + _shutdown = true; + _listener.Stop(); + if (!_shutdownEvent.WaitOne()) + m_log.Error("[WEBSOCKETSERVER]: Failed to shutdown listener properly."); + _listener = null; + } + + public bool HandlesRegion(Location x) + { + return x == m_location; + } + + public void AddScene(IScene scene) + { + if (m_scene != null) + { + m_log.Debug("[WEBSOCKETSERVER]: AddScene() called but I already have a scene."); + return; + } + if (!(scene is Scene)) + { + m_log.Error("[WEBSOCKETSERVER]: AddScene() called with an unrecognized scene type " + scene.GetType()); + return; + } + + m_scene = (Scene)scene; + m_location = new Location(m_scene.RegionInfo.RegionHandle); + } + + public void Start() + { + _listener = new TcpListener(_address, _port); + _listener.Start(_backlogAcceptQueueLength); + Interlocked.Increment(ref _pendingAccepts); + _listener.BeginAcceptSocket(OnAccept, null); + } + + private void OnAccept(IAsyncResult ar) + { + bool beginAcceptCalled = false; + try + { + int count = Interlocked.Decrement(ref _pendingAccepts); + if (_shutdown) + { + if (count == 0) + _shutdownEvent.Set(); + return; + } + Interlocked.Increment(ref _pendingAccepts); + _listener.BeginAcceptSocket(OnAccept, null); + beginAcceptCalled = true; + Socket socket = _listener.EndAcceptSocket(ar); + if (!OnAcceptingSocket(socket)) + { + socket.Disconnect(true); + return; + } + ClientNetworkContext context = new ClientNetworkContext((IPEndPoint) socket.RemoteEndPoint, _port, + new NetworkStream(socket), 16384, socket); + HttpRequestParser parser; + context.BeginRead(); + + } + catch (Exception err) + { + if (ExceptionThrown == null) +#if DEBUG + throw; +#else + _logWriter.Write(this, LogPrio.Fatal, err.Message); + // we can't really do anything but close the connection +#endif + if (ExceptionThrown != null) + ExceptionThrown(this, err); + + if (!beginAcceptCalled) + RetryBeginAccept(); + + } + } + + private void RetryBeginAccept() + { + try + { + + _listener.BeginAcceptSocket(OnAccept, null); + } + catch (Exception err) + { + + if (ExceptionThrown == null) +#if DEBUG + throw; +#else + // we can't really do anything but close the connection +#endif + if (ExceptionThrown != null) + ExceptionThrown(this, err); + } + } + + private bool OnAcceptingSocket(Socket sock) + { + ClientAcceptedEventArgs args = new ClientAcceptedEventArgs(sock); + Accepted(this, args); + return !args.Revoked; + } + /// + /// Catch exceptions not handled by the listener. + /// + /// + /// Exceptions will be thrown during debug mode if this event is not used, + /// exceptions will be printed to console and suppressed during release mode. + /// + public event ExceptionHandler ExceptionThrown = delegate { }; + + + + } +} diff --git a/prebuild.xml b/prebuild.xml index 329ff73550..d8b4145698 100644 --- a/prebuild.xml +++ b/prebuild.xml @@ -1466,6 +1466,8 @@ + + @@ -1508,6 +1510,49 @@ + + + + ../../../../bin/ + + + + + ../../../../bin/ + + + + ../../../../bin/ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + From d18fbb98b7f51d46eb3e716c59a8e76bc772bad1 Mon Sep 17 00:00:00 2001 From: teravus Date: Sun, 3 Feb 2013 07:44:45 -0500 Subject: [PATCH 2/8] Adds the ability to load more then one IClientNetworkServer thereby allowing additional client stacks. Use comma separated values in clientstack_plugin in your config. --- OpenSim/Region/Application/OpenSimBase.cs | 37 +++++---- .../Region/ClientStack/ClientStackManager.cs | 83 ++++++++++++------- .../TCPJSONStream/ClientNetworkContext.cs | 3 + .../TCPJSONStream/TCPJsonWebSocketServer.cs | 2 +- bin/OpenSimDefaults.ini | 2 +- 5 files changed, 78 insertions(+), 49 deletions(-) diff --git a/OpenSim/Region/Application/OpenSimBase.cs b/OpenSim/Region/Application/OpenSimBase.cs index c3c87e786c..f5c06df3fb 100644 --- a/OpenSim/Region/Application/OpenSimBase.cs +++ b/OpenSim/Region/Application/OpenSimBase.cs @@ -316,7 +316,7 @@ namespace OpenSim /// /// /// - public IClientNetworkServer CreateRegion(RegionInfo regionInfo, bool portadd_flag, out IScene scene) + public List CreateRegion(RegionInfo regionInfo, bool portadd_flag, out IScene scene) { return CreateRegion(regionInfo, portadd_flag, false, out scene); } @@ -326,7 +326,7 @@ namespace OpenSim /// /// /// - public IClientNetworkServer CreateRegion(RegionInfo regionInfo, out IScene scene) + public List CreateRegion(RegionInfo regionInfo, out IScene scene) { return CreateRegion(regionInfo, false, true, out scene); } @@ -338,7 +338,7 @@ namespace OpenSim /// /// /// - public IClientNetworkServer CreateRegion(RegionInfo regionInfo, bool portadd_flag, bool do_post_init, out IScene mscene) + public List CreateRegion(RegionInfo regionInfo, bool portadd_flag, bool do_post_init, out IScene mscene) { int port = regionInfo.InternalEndPoint.Port; @@ -363,8 +363,8 @@ namespace OpenSim Util.XmlRpcCommand(proxyUrl, "AddPort", port, port + proxyOffset, regionInfo.ExternalHostName); } - IClientNetworkServer clientServer; - Scene scene = SetupScene(regionInfo, proxyOffset, Config, out clientServer); + List clientServers; + Scene scene = SetupScene(regionInfo, proxyOffset, Config, out clientServers); m_log.Info("[MODULES]: Loading Region's modules (old style)"); @@ -414,8 +414,11 @@ namespace OpenSim if (m_autoCreateClientStack) { - m_clientServers.Add(clientServer); - clientServer.Start(); + foreach (IClientNetworkServer clientserver in clientServers) + { + m_clientServers.Add(clientserver); + clientserver.Start(); + } } scene.EventManager.OnShutdown += delegate() { ShutdownRegion(scene); }; @@ -425,7 +428,7 @@ namespace OpenSim scene.Start(); scene.StartScripts(); - return clientServer; + return clientServers; } /// @@ -641,7 +644,7 @@ namespace OpenSim /// /// /// - protected Scene SetupScene(RegionInfo regionInfo, out IClientNetworkServer clientServer) + protected Scene SetupScene(RegionInfo regionInfo, out List clientServer) { return SetupScene(regionInfo, 0, null, out clientServer); } @@ -655,19 +658,20 @@ namespace OpenSim /// /// protected Scene SetupScene( - RegionInfo regionInfo, int proxyOffset, IConfigSource configSource, out IClientNetworkServer clientServer) + RegionInfo regionInfo, int proxyOffset, IConfigSource configSource, out List clientServer) { + List clientNetworkServers = null; + AgentCircuitManager circuitManager = new AgentCircuitManager(); IPAddress listenIP = regionInfo.InternalEndPoint.Address; //if (!IPAddress.TryParse(regionInfo.InternalEndPoint, out listenIP)) // listenIP = IPAddress.Parse("0.0.0.0"); uint port = (uint) regionInfo.InternalEndPoint.Port; - + IClientNetworkServer clientNetworkServer; if (m_autoCreateClientStack) { - clientServer - = m_clientStackManager.CreateServer( + clientNetworkServers = m_clientStackManager.CreateServers( listenIP, ref port, proxyOffset, regionInfo.m_allow_alternate_ports, configSource, circuitManager); } @@ -682,9 +686,12 @@ namespace OpenSim if (m_autoCreateClientStack) { - clientServer.AddScene(scene); + foreach (IClientNetworkServer clientnetserver in clientNetworkServers) + { + clientnetserver.AddScene(scene); + } } - + clientServer = clientNetworkServers; scene.LoadWorldMap(); scene.PhysicsScene = GetPhysicsScene(scene.RegionInfo.RegionName); diff --git a/OpenSim/Region/ClientStack/ClientStackManager.cs b/OpenSim/Region/ClientStack/ClientStackManager.cs index 84ea0b34db..299aabd787 100644 --- a/OpenSim/Region/ClientStack/ClientStackManager.cs +++ b/OpenSim/Region/ClientStack/ClientStackManager.cs @@ -26,6 +26,7 @@ */ using System; +using System.Collections.Generic; using System.Net; using System.Reflection; using log4net; @@ -38,39 +39,53 @@ namespace OpenSim.Region.ClientStack { private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); - private Type plugin; - private Assembly pluginAssembly; + private List plugin = new List(); + private List pluginAssembly = new List(); - public ClientStackManager(string dllName) + public ClientStackManager(string pDllName) { - m_log.Info("[CLIENTSTACK]: Attempting to load " + dllName); - - try + List clientstacks = new List(); + if (pDllName.Contains(",")) { - plugin = null; - pluginAssembly = Assembly.LoadFrom(dllName); + clientstacks = new List(pDllName.Split(',')); + } + else + { + clientstacks.Add(pDllName); + } + foreach (string dllName in clientstacks) + { + m_log.Info("[CLIENTSTACK]: Attempting to load " + dllName); - foreach (Type pluginType in pluginAssembly.GetTypes()) + try { - if (pluginType.IsPublic) - { - Type typeInterface = pluginType.GetInterface("IClientNetworkServer", true); + //plugin = null; + Assembly itemAssembly = Assembly.LoadFrom(dllName); + pluginAssembly.Add(itemAssembly); - if (typeInterface != null) + foreach (Type pluginType in itemAssembly.GetTypes()) + { + if (pluginType.IsPublic) { - m_log.Info("[CLIENTSTACK]: Added IClientNetworkServer Interface"); - plugin = pluginType; - return; + Type typeInterface = pluginType.GetInterface("IClientNetworkServer", true); + + if (typeInterface != null) + { + m_log.Info("[CLIENTSTACK]: Added IClientNetworkServer Interface"); + plugin.Add(pluginType); + break; + } } } } - } catch (ReflectionTypeLoadException e) - { - foreach (Exception e2 in e.LoaderExceptions) + catch (ReflectionTypeLoadException e) { - m_log.Error(e2.ToString()); + foreach (Exception e2 in e.LoaderExceptions) + { + m_log.Error(e2.ToString()); + } + throw e; } - throw e; } } @@ -84,11 +99,11 @@ namespace OpenSim.Region.ClientStack /// /// /// - public IClientNetworkServer CreateServer( + public List CreateServers( IPAddress _listenIP, ref uint port, int proxyPortOffset, bool allow_alternate_port, AgentCircuitManager authenticateClass) { - return CreateServer( + return CreateServers( _listenIP, ref port, proxyPortOffset, allow_alternate_port, null, authenticateClass); } @@ -105,20 +120,24 @@ namespace OpenSim.Region.ClientStack /// /// /// - public IClientNetworkServer CreateServer( + public List CreateServers( IPAddress _listenIP, ref uint port, int proxyPortOffset, bool allow_alternate_port, IConfigSource configSource, AgentCircuitManager authenticateClass) { + List servers = new List(); if (plugin != null) { - IClientNetworkServer server = - (IClientNetworkServer)Activator.CreateInstance(pluginAssembly.GetType(plugin.ToString())); - - server.Initialise( - _listenIP, ref port, proxyPortOffset, allow_alternate_port, - configSource, authenticateClass); - - return server; + for (int i = 0; i < plugin.Count; i++) + { + IClientNetworkServer server = + (IClientNetworkServer) Activator.CreateInstance(pluginAssembly[i].GetType(plugin[i].ToString())); + + server.Initialise( + _listenIP, ref port, proxyPortOffset, allow_alternate_port, + configSource, authenticateClass); + servers.Add(server); + } + return servers; } m_log.Error("[CLIENTSTACK]: Couldn't initialize a new server"); diff --git a/OpenSim/Region/ClientStack/TCPJSONStream/ClientNetworkContext.cs b/OpenSim/Region/ClientStack/TCPJSONStream/ClientNetworkContext.cs index 591f817b8f..b077b6aaea 100644 --- a/OpenSim/Region/ClientStack/TCPJSONStream/ClientNetworkContext.cs +++ b/OpenSim/Region/ClientStack/TCPJSONStream/ClientNetworkContext.cs @@ -55,6 +55,9 @@ namespace OpenSim.Region.ClientStack.TCPJSONStream } } + catch (Exception) + { + } } /// /// send a whole buffer diff --git a/OpenSim/Region/ClientStack/TCPJSONStream/TCPJsonWebSocketServer.cs b/OpenSim/Region/ClientStack/TCPJSONStream/TCPJsonWebSocketServer.cs index 0713bf4403..c0f679296a 100644 --- a/OpenSim/Region/ClientStack/TCPJSONStream/TCPJsonWebSocketServer.cs +++ b/OpenSim/Region/ClientStack/TCPJSONStream/TCPJsonWebSocketServer.cs @@ -99,7 +99,7 @@ namespace OpenSim.Region.ClientStack.TCPJSONStream } ClientNetworkContext context = new ClientNetworkContext((IPEndPoint) socket.RemoteEndPoint, _port, new NetworkStream(socket), 16384, socket); - HttpRequestParser parser; + //HttpRequestParser parser; context.BeginRead(); } diff --git a/bin/OpenSimDefaults.ini b/bin/OpenSimDefaults.ini index c60579b834..cc0809427e 100644 --- a/bin/OpenSimDefaults.ini +++ b/bin/OpenSimDefaults.ini @@ -58,7 +58,7 @@ ; ## ; Set this to the DLL containing the client stack to use. - clientstack_plugin="OpenSim.Region.ClientStack.LindenUDP.dll" + clientstack_plugin="OpenSim.Region.ClientStack.LindenUDP.dll,OpenSim.Region.ClientStack.TCPJSONStream.dll" ; ## ; ## REGIONS From 29d521e2733bf8dc11cfdbdad104f9f141f7c895 Mon Sep 17 00:00:00 2001 From: teravus Date: Sun, 3 Feb 2013 07:56:31 -0500 Subject: [PATCH 3/8] Changing OpenSimDefaults back to default --- bin/OpenSimDefaults.ini | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bin/OpenSimDefaults.ini b/bin/OpenSimDefaults.ini index cc0809427e..c60579b834 100644 --- a/bin/OpenSimDefaults.ini +++ b/bin/OpenSimDefaults.ini @@ -58,7 +58,7 @@ ; ## ; Set this to the DLL containing the client stack to use. - clientstack_plugin="OpenSim.Region.ClientStack.LindenUDP.dll,OpenSim.Region.ClientStack.TCPJSONStream.dll" + clientstack_plugin="OpenSim.Region.ClientStack.LindenUDP.dll" ; ## ; ## REGIONS From 1dc09d8e8f4a6caa321d0227722af97ee4aeed6a Mon Sep 17 00:00:00 2001 From: teravus Date: Tue, 5 Feb 2013 18:02:25 -0500 Subject: [PATCH 4/8] We're not really done here.. but we're getting there. Socket Read is working.. Still have to do Header.ToBytes and compose a websocket frame with a payload. --- .../Servers/HttpServer/BaseHttpServer.cs | 38 +- .../Framework/Servers/Tests/OSHttpTests.cs | 5 + .../TCPJSONStream/OpenSimWebSocketBase.cs | 6 +- bin/HttpServer_OpenSim.dll | Bin 115712 -> 116224 bytes bin/HttpServer_OpenSim.pdb | Bin 413184 -> 302592 bytes bin/HttpServer_OpenSim.xml | 8996 ++++++++--------- 6 files changed, 4543 insertions(+), 4502 deletions(-) diff --git a/OpenSim/Framework/Servers/HttpServer/BaseHttpServer.cs b/OpenSim/Framework/Servers/HttpServer/BaseHttpServer.cs index b24336deee..dcfe99a070 100644 --- a/OpenSim/Framework/Servers/HttpServer/BaseHttpServer.cs +++ b/OpenSim/Framework/Servers/HttpServer/BaseHttpServer.cs @@ -54,6 +54,8 @@ namespace OpenSim.Framework.Servers.HttpServer private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private HttpServerLogWriter httpserverlog = new HttpServerLogWriter(); + public delegate void WebSocketRequestDelegate(string servicepath, WebSocketHTTPServerHandler handler); + /// /// Gets or sets the debug level. /// @@ -87,6 +89,9 @@ namespace OpenSim.Framework.Servers.HttpServer protected Dictionary m_pollHandlers = new Dictionary(); + protected Dictionary m_WebSocketHandlers = + new Dictionary(); + protected uint m_port; protected uint m_sslport; protected bool m_ssl; @@ -170,6 +175,22 @@ namespace OpenSim.Framework.Servers.HttpServer } } + public void AddWebSocketHandler(string servicepath, WebSocketRequestDelegate handler) + { + lock (m_WebSocketHandlers) + { + if (!m_WebSocketHandlers.ContainsKey(servicepath)) + m_WebSocketHandlers.Add(servicepath, handler); + } + } + + public void RemoveWebSocketHandler(string servicepath) + { + lock (m_WebSocketHandlers) + if (m_WebSocketHandlers.ContainsKey(servicepath)) + m_WebSocketHandlers.Remove(servicepath); + } + public List GetStreamHandlerKeys() { lock (m_streamHandlers) @@ -409,9 +430,24 @@ namespace OpenSim.Framework.Servers.HttpServer public void OnHandleRequestIOThread(IHttpClientContext context, IHttpRequest request) { + OSHttpRequest req = new OSHttpRequest(context, request); + WebSocketRequestDelegate dWebSocketRequestDelegate = null; + lock (m_WebSocketHandlers) + { + if (m_WebSocketHandlers.ContainsKey(req.RawUrl)) + dWebSocketRequestDelegate = m_WebSocketHandlers[req.RawUrl]; + } + if (dWebSocketRequestDelegate != null) + { + dWebSocketRequestDelegate(req.Url.AbsolutePath, new WebSocketHTTPServerHandler(req, context, 16384)); + return; + } + OSHttpResponse resp = new OSHttpResponse(new HttpResponse(context, request),context); + HandleRequest(req, resp); + // !!!HACK ALERT!!! // There seems to be a bug in the underlying http code that makes subsequent requests @@ -500,7 +536,7 @@ namespace OpenSim.Framework.Servers.HttpServer LogIncomingToStreamHandler(request, requestHandler); response.ContentType = requestHandler.ContentType; // Lets do this defaulting before in case handler has varying content type. - + if (requestHandler is IStreamedRequestHandler) { IStreamedRequestHandler streamedRequestHandler = requestHandler as IStreamedRequestHandler; diff --git a/OpenSim/Framework/Servers/Tests/OSHttpTests.cs b/OpenSim/Framework/Servers/Tests/OSHttpTests.cs index 3412e0f2f7..5b912b4ea2 100644 --- a/OpenSim/Framework/Servers/Tests/OSHttpTests.cs +++ b/OpenSim/Framework/Servers/Tests/OSHttpTests.cs @@ -70,6 +70,11 @@ namespace OpenSim.Framework.Servers.Tests public void Close() { } public bool EndWhenDone { get { return false;} set { return;}} + public HTTPNetworkContext GiveMeTheNetworkStreamIKnowWhatImDoing() + { + return new HTTPNetworkContext(); + } + public event EventHandler Disconnected = delegate { }; /// /// A request have been received in the context. diff --git a/OpenSim/Region/ClientStack/TCPJSONStream/OpenSimWebSocketBase.cs b/OpenSim/Region/ClientStack/TCPJSONStream/OpenSimWebSocketBase.cs index 379438d384..6d025434a7 100644 --- a/OpenSim/Region/ClientStack/TCPJSONStream/OpenSimWebSocketBase.cs +++ b/OpenSim/Region/ClientStack/TCPJSONStream/OpenSimWebSocketBase.cs @@ -47,7 +47,7 @@ namespace OpenSim.Region.ClientStack.TCPJSONStream public void NetworkStop() { - m_tcpServer.Stop(); + // m_tcpServer.Stop(); } public bool HandlesRegion(Location x) @@ -62,12 +62,12 @@ namespace OpenSim.Region.ClientStack.TCPJSONStream public void Start() { - m_tcpServer.Start(); + //m_tcpServer.Start(); } public void Stop() { - m_tcpServer.Stop(); + // m_tcpServer.Stop(); } } } \ No newline at end of file diff --git a/bin/HttpServer_OpenSim.dll b/bin/HttpServer_OpenSim.dll index d910bb9d27ff9026d5b061592f6c1b05f4cd0801..9cd1e0857b42badaad81e63a72ebb2c26effeb8a 100755 GIT binary patch literal 116224 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 z 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 diff --git a/bin/HttpServer_OpenSim.pdb b/bin/HttpServer_OpenSim.pdb index b9161e1802f78b40d40abb781c7a8650f0c0dc5d..d20a0c5074c5c0c5d35a409c4667a0e275481e51 100644 GIT binary patch literal 302592 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 YWONkT9d?(%?h@Et0=rA#Kf46}8)UK$YXATM literal 413184 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

#d_AE` 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 - -

- A session store is used to store and load sessions on a media. - The default implementation () saves/retrieves sessions from memory. - - - - - Creates a new http session with a generated id. - - A object - - - - Creates a new http session with a specific id - - Id used to identify the new cookie.. - A object. - - Id should be generated by the store implementation if it's null or . - - - - - Load an existing session. - - Session id (usually retrieved from a client side cookie). - A session if found; otherwise null. - - - - Save an updated session to the store. - - Session id (usually retrieved from a client side cookie). - If Id property have not been specified. - - - - We use the flyweight pattern which reuses small objects - instead of creating new each time. - - Unused session that should be reused next time Create is called. - - - - Remove expired sessions - - - - - Remove a session - - id of the session. - - - - Load a session from the store - - - null if session is not found. - - - - Number of minutes before a session expires. - - Default time is 20 minutes. - - - - Contains server side HTTP request information. - - - - - Called during parsing of a . - - Name of the header, should not be URL encoded - Value of the header, should not be URL encoded - If a header is incorrect. - - - - Add bytes to the body - - buffer to read bytes from - where to start read - number of bytes to read - Number of bytes actually read (same as length unless we got all body bytes). - If body is not writable - bytes is null. - offset is out of range. - - - - Clear everything in the request - - - - - Decode body into a form. - - A list with form decoders. - If body contents is not valid for the chosen decoder. - If body is still being transferred. - - - - Sets the cookies. - - The cookies. - - - - Create a response object. - - Context for the connected client. - A new . - - - - Gets kind of types accepted by the client. - - - - - Gets or sets body stream. - - - - - Gets whether the body is complete. - - - - - Gets or sets kind of connection used for the session. - - - - - Gets or sets number of bytes in the body. - - - - - Gets cookies that was sent with the request. - - - - - Gets form parameters. - - - - - Gets headers sent by the client. - - - - - Gets or sets version of HTTP protocol that's used. - - - Probably or . - - - - - - Gets whether the request was made by Ajax (Asynchronous JavaScript) - - - - - Gets or sets requested method. - - - Will always be in upper case. - - - - - - Gets parameter from or . - - - - - Gets variables sent in the query string - - - - - Gets or sets requested URI. - - - - - Gets URI absolute path divided into parts. - - - // URI is: http://gauffin.com/code/tiny/ - Console.WriteLine(request.UriParts[0]); // result: code - Console.WriteLine(request.UriParts[1]); // result: tiny - - - If you're using controllers than the first part is controller name, - the second part is method name and the third part is Id property. - - - - - - Gets or sets path and query. - - - - Are only used during request parsing. Cannot be set after "Host" header have been - added. - - - - - Class that receives Requests from a . - - - - - Client have been disconnected. - - Client that was disconnected. - Reason - - - - - Invoked when a client context have received a new HTTP request - - Client that received the request. - Request that was received. - - - - - Delegate used by to populate select options. - - current object (for instance a User). - Text that should be displayed in the value part of a <optiongt;-tag. - Text shown in the select list. - - // Class that is going to be used in a SELECT-tag. - public class User - { - private readonly string _realName; - private readonly int _id; - public User(int id, string realName) - { - _id = id; - _realName = realName; - } - public string RealName - { - get { return _realName; } - } - - public int Id - { - get { return _id; } - } - } - - // Using an inline delegate to generate the select list - public void UserInlineDelegate() - { - List<User> items = new List<User>(); - items.Add(new User(1, "adam")); - items.Add(new User(2, "bertial")); - items.Add(new User(3, "david")); - string htmlSelect = Select("users", "users", items, delegate(object o, out object id, out object value) - { - User user = (User)o; - id = user.Id; - value = user.RealName; - }, 2, true); - } - - // Using an method as delegate to generate the select list. - public void UseExternalDelegate() - { - List<User> items = new List<User>(); - items.Add(new User(1, "adam")); - items.Add(new User(2, "bertial")); - items.Add(new User(3, "david")); - string htmlSelect = Select("users", "users", items, UserOptions, 1, true); - } - - // delegate returning id and title - public static void UserOptions(object o, out object id, out object title) - { - User user = (User)o; - id = user.Id; - value = user.RealName; - } /// - - - - The server understood the request, but is refusing to fulfill it. - Authorization will not help and the request SHOULD NOT be repeated. - If the request method was not HEAD and the server wishes to make public why the request has not been fulfilled, - it SHOULD describe the reason for the refusal in the entity. If the server does not wish to make this information - available to the client, the status code 404 (Not Found) can be used instead. - - Text taken from: http://www.submissionchamber.com/help-guides/error-codes.php - - - - - All HTTP based exceptions will derive this class. - - - - - Create a new HttpException - - http status code (sent in the response) - error description - - - - Create a new HttpException - - http status code (sent in the response) - error description - inner exception - - - - status code to use in the response. - - - - - Initializes a new instance of the class. - - error message - - - - A session stored in memory. - - - - - Interface for sessions - - - - - Remove everything from the session - - - - - Remove everything from the session - - True if the session is cleared due to expiration - - - - Session id - - - - - Should - - Name of the session variable - null if it's not set - If the object cant be serialized. - - - - When the session was last accessed. - This property is touched by the http server each time the - session is requested. - - - - - Number of session variables. - - - - - Event triggered upon clearing the session - - - - - - - A unique id used by the sessions store to identify the session - - - - Id - - - - - - Remove everything from the session - - - - - Clears the specified expire. - - True if the session is cleared due to expiration - - - - Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources. - - 2 - - - - Session id - - - - - Should - - Name of the session variable - null if it's not set - - - - when the session was last accessed. - - - Used to determine when the session should be removed. - - - - - Number of values in the session - - - - - Flag to indicate that the session have been changed - and should be saved into the session store. - - - - - Event triggered upon clearing the session - - - - - cookie being sent back to the browser. - - - - - - cookie sent by the client/browser - - - - - - Constructor. - - cookie identifier - cookie content - id or content is null - id is empty - - - - Gets the cookie HTML representation. - - cookie string - - - - Gets the cookie identifier. - - - - - Cookie value. Set to null to remove cookie. - - - - - Constructor. - - cookie identifier - cookie content - cookie expiration date. Use DateTime.MinValue for session cookie. - id or content is null - id is empty - - - - Create a new cookie - - name identifying the cookie - cookie value - when the cookie expires. Setting DateTime.MinValue will delete the cookie when the session is closed. - Path to where the cookie is valid - Domain that the cookie is valid for. - - - - Create a new cookie - - Name and value will be used - when the cookie expires. - - - - Gets the cookie HTML representation. - - cookie string - - - - When the cookie expires. - DateTime.MinValue means that the cookie expires when the session do so. - - - - - Cookie is only valid under this path. - - - - - Contains a connection to a browser/client. - - - - - Disconnect from client - - error to report in the event. - - - - Send a response. - - Either or - HTTP status code - reason for the status code. - HTML body contents, can be null or empty. - A content type to return the body as, i.e. 'text/html' or 'text/plain', defaults to 'text/html' if null or empty - If is invalid. - - - - Send a response. - - Either or - HTTP status code - reason for the status code. - - - - Send a response. - - - - - - send a whole buffer - - buffer to send - - - - - Send data using the stream - - Contains data to send - Start position in buffer - number of bytes to send - - - - - - Closes the streams and disposes of the unmanaged resources - - - - - Using SSL or other encryption method. - - - - - Using SSL or other encryption method. - - - - - The context have been disconnected. - - - Event can be used to clean up a context, or to reuse it. - - - - - A request have been received in the context. - - - - - A have been disconnected. - - - - - Initializes a new instance of the class. - - Reason to disconnection. - - - - Gets reason to why client disconnected. - - - - - - - - - - Initializes a new instance of the class. - - The request. - - - - Gets received request. - - - - - Contains a listener that doesn't do anything with the connections. - - - - - Listen for regular HTTP connections - - IP Address to accept connections on - TCP Port to listen on, default HTTP port is 80. - Factory used to create es. - address is null. - Port must be a positive number. - - - - Initializes a new instance of the class. - - IP Address to accept connections on - TCP Port to listen on, default HTTPS port is 443 - Factory used to create es. - Certificate to use - - - - Initializes a new instance of the class. - - IP Address to accept connections on - TCP Port to listen on, default HTTPS port is 443 - Factory used to create es. - Certificate to use - which HTTPS protocol to use, default is TLS. - - - Exception. - - - - Will try to accept connections one more time. - - If any exceptions is thrown. - - - - Can be used to create filtering of new connections. - - Accepted socket - true if connection can be accepted; otherwise false. - - - - Start listen for new connections - - Number of connections that can stand in a queue to be accepted. - Listener have already been started. - - - - Stop the listener - - - - - - Gives you a change to receive log entries for all internals of the HTTP library. - - - You may not switch log writer after starting the listener. - - - - - True if we should turn on trace logs. - - - - - Catch exceptions not handled by the listener. - - - Exceptions will be thrown during debug mode if this event is not used, - exceptions will be printed to console and suppressed during release mode. - - - - - A request have been received from a . - - - - - - - - http://www.faqs.org/rfcs/rfc1867.html - - - - - Interface for form content decoders. - - - - - - - Stream containing the content - Content type (with any additional info like boundry). Content type is always supplied in lower case - Stream enconding - A http form, or null if content could not be parsed. - If contents in the stream is not valid input data. - - - - Checks if the decoder can handle the mime type - - Content type (with any additional info like boundry). Content type is always supplied in lower case. - True if the decoder can parse the specified content type - - - - multipart/form-data - - - - - form-data - - - - - - - Stream containing the content - Content type (with any additional info like boundry). Content type is always supplied in lower case - Stream enconding - A http form, or null if content could not be parsed. - If contents in the stream is not valid input data. - If any parameter is null - - - - Checks if the decoder can handle the mime type - - Content type (with any additional info like boundry). Content type is always supplied in lower case. - True if the decoder can parse the specified content type - - - - The requested resource was not found in the web server. - - - - - Create a new exception - - message describing the error - inner exception - - - - Create a new exception - - message describing the error - - - - Delegate used to let authentication modules authenticate the user name and password. - - Realm that the user want to authenticate in - User name specified by client - Can either be user password or implementation specific token. - object that will be stored in a session variable called if authentication was successful. - throw forbidden exception if too many attempts have been made. - - - Use to specify that the token is a HA1 token. (MD5 generated - string from realm, user name and password); Md5String(userName + ":" + realm + ":" + password); - - - - - - Let's you decide on a system level if authentication is required. - - HTTP request from client - true if user should be authenticated. - throw if no more attempts are allowed. - If no more attempts are allowed - - - - Authentication modules are used to implement different - kind of HTTP authentication. - - - - - Tag used for authentication. - - - - - Initializes a new instance of the class. - - Delegate used to provide information used during authentication. - Delegate used to determine if authentication is required (may be null). - - - - Initializes a new instance of the class. - - Delegate used to provide information used during authentication. - - - - Create a response that can be sent in the WWW-Authenticate header. - - Realm that the user should authenticate in - Array with optional options. - A correct authentication request. - If realm is empty or null. - - - - An authentication response have been received from the web browser. - Check if it's correct - - Contents from the Authorization header - Realm that should be authenticated - GET/POST/PUT/DELETE etc. - options to specific implementations - Authentication object that is stored for the request. A user class or something like that. - if is invalid - If any of the parameters is empty or null. - - - - Used to invoke the authentication delegate that is used to lookup the user name/realm. - - Realm (domain) that user want to authenticate in - User name - Password used for validation. Some implementations got password in clear text, they are then sent to client. - object that will be stored in the request to help you identify the user if authentication was successful. - true if authentication was successful - - - - Determines if authentication is required. - - HTTP request from browser - true if user should be authenticated. - throw from your delegate if no more attempts are allowed. - If no more attempts are allowed - - - - name used in HTTP request. - - - - - Contains some kind of input from the browser/client. - can be QueryString, form data or any other request body content. - - - - - Base class for request data containers - - - - - Adds a parameter mapped to the presented name - - The name to map the parameter to - The parameter value - - - - Returns true if the container contains the requested parameter - - Parameter id - True if parameter exists - - - - Returns a request parameter - - The name associated with the parameter - - - - Representation of a non-initialized class instance - - - Variable telling the class that it is non-initialized - - - - Initializes a new instance of the class. - - form name. - - - - Initializes a new instance of the class. - - form name. - if set to true all changes will be ignored. - this constructor should only be used by Empty - - - Creates a deep copy of the HttpInput class - The object to copy - The function makes a deep copy of quite a lot which can be slow - - - - Add a new element. Form array elements are parsed - and added in a correct hierarchy. - - Name is converted to lower case. - - name is null. - Cannot add stuff to . - - - - Returns true if the class contains a with the corresponding name. - - The field/query string name - True if the value exists - - - - Parses an item and returns it. - This function is primarily used to parse array items as in user[name]. - - - - - - - Outputs the instance representing all its values joined together - - - - Returns all items as an unescaped query string. - - - - - Extracts one parameter from an array - - Containing the string array - All but the first value - - string test1 = ExtractOne("system[user][extension][id]"); - string test2 = ExtractOne(test1); - string test3 = ExtractOne(test2); - // test1 = user[extension][id] - // test2 = extension[id] - // test3 = id - - - - Resets all data contained by class - - - - Returns an enumerator that iterates through the collection. - - - - A that can be used to iterate through the collection. - - 1 - - - - Returns an enumerator that iterates through a collection. - - - - An object that can be used to iterate through the collection. - - 2 - - - - Form name as lower case - - - - - Get a form item. - - - Returns if item was not found. - - - - Small design by contract implementation. - - - - - Check whether a parameter is empty. - - Parameter value - Parameter name, or error description. - value is empty. - - - - Checks whether a parameter is null. - - Parameter value - Parameter name, or error description. - value is null. - - - - Checks whether a parameter is null. - - - Parameter value - Parameter name, or error description. - value is null. - - - - Contains all HTTP Methods (according to the HTTP 1.1 specification) - - See: http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html - - - - - - The DELETE method requests that the origin server delete the resource identified by the Request-URI. - - - - This method MAY be overridden by human intervention (or other means) on the origin server. - The client cannot be guaranteed that the operation has been carried out, even if the status code - returned from the origin server indicates that the action has been completed successfully. - - - However, the server SHOULD NOT indicate success unless, at the time the response is given, - it intends to delete the resource or move it to an inaccessible location. - - - A successful response SHOULD be 200 (OK) if the response includes an entity describing the status, - 202 (Accepted) if the action has not yet been enacted, - or 204 (No Content) if the action has been enacted but the response does not include an entity. - - - If the request passes through a cache and the Request-URI identifies one or more currently cached entities, - those entries SHOULD be treated as stale. Responses to this method are not cacheable. - - - - - - The GET method means retrieve whatever information (in the form of an entity) is identified by the Request-URI. - - - - If the Request-URI refers to a data-producing process, it is the produced data which shall be returned as the - entity in the response and not the source text of the process, unless that text happens to be the output of the process. - - - The semantics of the GET method change to a "conditional GET" if the request message includes an - If-Modified-Since, If-Unmodified-Since, If-Match, If-None-Match, or If-Range header field. - A conditional GET method requests that the entity be transferred only under the circumstances described - by the conditional header field(s). The conditional GET method is intended to reduce unnecessary network - usage by allowing cached entities to be refreshed without requiring multiple requests or transferring - data already held by the client. - - - - - - The HEAD method is identical to GET except that the server MUST NOT return a message-body in the response. - - - The meta information contained in the HTTP headers in response to a HEAD request SHOULD be identical to the - information sent in response to a GET request. This method can be used for obtaining meta information about - the entity implied by the request without transferring the entity-body itself. - - This method is often used for testing hypertext links for validity, accessibility, and recent modification. - - - - - The OPTIONS method represents a request for information about the communication options available on the request/response chain identified by the Request-URI. - - - This method allows the client to determine the options and/or requirements associated with a resource, or the capabilities of a server, without implying a resource action or initiating a resource retrieval. - - - - - The POST method is used to request that the origin server accept the entity enclosed - in the request as a new subordinate of the resource identified by the Request-URI in the Request-Line. - - - POST is designed to allow a uniform method to cover the following functions: - - - Annotation of existing resources; - - Posting a message to a bulletin board, newsgroup, mailing list, or similar group of articles; - - Providing a block of data, such as the result of submitting a form, to a data-handling process; - - Extending a database through an append operation. - - - - If a resource has been created on the origin server, the response SHOULD be 201 (Created) and - contain an entity which describes the status of the request and refers to the new resource, and a - Location header (see section 14.30). - - - The action performed by the POST method might not result in a resource that can be identified by a URI. - In this case, either 200 (OK) or 204 (No Content) is the appropriate response status, depending on - whether or not the response includes an entity that describes the result. - - Responses to this method are not cacheable, unless the response includes appropriate Cache-Control - or Expires header fields. However, the 303 (See Other) response can be used to direct the user agent - to retrieve a cacheable resource. - - - - - - The PUT method requests that the enclosed entity be stored under the supplied Request-URI. - - - - - If the Request-URI refers to an already existing resource, the enclosed entity SHOULD be considered as a - modified version of the one residing on the origin server. - - If the Request-URI does not point to an existing resource, and that URI is capable of being defined as a new - resource by the requesting user agent, the origin server can create the resource with that URI. - - If a new resource is created, the origin server MUST inform the user agent via the 201 (Created) response. - - If an existing resource is modified, either the 200 (OK) or 204 (No Content) response codes SHOULD be sent to - indicate successful completion of the request. - - If the resource could not be created or modified with the Request-URI, an appropriate error response SHOULD be - given that reflects the nature of the problem. - - - - The recipient of the entity MUST NOT ignore any Content-* (e.g. Content-Range) headers that it does not - understand or implement and MUST return a 501 (Not Implemented) response in such cases. - - - - - - The TRACE method is used to invoke a remote, application-layer loop- back of the request message. - - - - - Contains all HTTP Methods (according to the HTTP 1.1 specification) - - See: http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html - - - - - - The DELETE method requests that the origin server delete the resource identified by the Request-URI. - - - - This method MAY be overridden by human intervention (or other means) on the origin server. - The client cannot be guaranteed that the operation has been carried out, even if the status code - returned from the origin server indicates that the action has been completed successfully. - - - However, the server SHOULD NOT indicate success unless, at the time the response is given, - it intends to delete the resource or move it to an inaccessible location. - - - A successful response SHOULD be 200 (OK) if the response includes an entity describing the status, - 202 (Accepted) if the action has not yet been enacted, - or 204 (No Content) if the action has been enacted but the response does not include an entity. - - - If the request passes through a cache and the Request-URI identifies one or more currently cached entities, - those entries SHOULD be treated as stale. Responses to this method are not cacheable. - - - - - - The GET method means retrieve whatever information (in the form of an entity) is identified by the Request-URI. - - - - If the Request-URI refers to a data-producing process, it is the produced data which shall be returned as the - entity in the response and not the source text of the process, unless that text happens to be the output of the process. - - - The semantics of the GET method change to a "conditional GET" if the request message includes an - If-Modified-Since, If-Unmodified-Since, If-Match, If-None-Match, or If-Range header field. - A conditional GET method requests that the entity be transferred only under the circumstances described - by the conditional header field(s). The conditional GET method is intended to reduce unnecessary network - usage by allowing cached entities to be refreshed without requiring multiple requests or transferring - data already held by the client. - - - - - - The HEAD method is identical to GET except that the server MUST NOT return a message-body in the response. - - - The meta information contained in the HTTP headers in response to a HEAD request SHOULD be identical to the - information sent in response to a GET request. This method can be used for obtaining meta information about - the entity implied by the request without transferring the entity-body itself. - - This method is often used for testing hypertext links for validity, accessibility, and recent modification. - - - - - The OPTIONS method represents a request for information about the communication options available on the request/response chain identified by the Request-URI. - - - This method allows the client to determine the options and/or requirements associated with a resource, or the capabilities of a server, without implying a resource action or initiating a resource retrieval. - - - - - The POST method is used to request that the origin server accept the entity enclosed - in the request as a new subordinate of the resource identified by the Request-URI in the Request-Line. - - - POST is designed to allow a uniform method to cover the following functions: - - - Annotation of existing resources; - - Posting a message to a bulletin board, newsgroup, mailing list, or similar group of articles; - - Providing a block of data, such as the result of submitting a form, to a data-handling process; - - Extending a database through an append operation. - - - - If a resource has been created on the origin server, the response SHOULD be 201 (Created) and - contain an entity which describes the status of the request and refers to the new resource, and a - Location header (see section 14.30). - - - The action performed by the POST method might not result in a resource that can be identified by a URI. - In this case, either 200 (OK) or 204 (No Content) is the appropriate response status, depending on - whether or not the response includes an entity that describes the result. - - Responses to this method are not cacheable, unless the response includes appropriate Cache-Control - or Expires header fields. However, the 303 (See Other) response can be used to direct the user agent - to retrieve a cacheable resource. - - - - - - The PUT method requests that the enclosed entity be stored under the supplied Request-URI. - - - - - If the Request-URI refers to an already existing resource, the enclosed entity SHOULD be considered as a - modified version of the one residing on the origin server. - - If the Request-URI does not point to an existing resource, and that URI is capable of being defined as a new - resource by the requesting user agent, the origin server can create the resource with that URI. - - If a new resource is created, the origin server MUST inform the user agent via the 201 (Created) response. - - If an existing resource is modified, either the 200 (OK) or 204 (No Content) response codes SHOULD be sent to - indicate successful completion of the request. - - If the resource could not be created or modified with the Request-URI, an appropriate error response SHOULD be - given that reflects the nature of the problem. - - - - The recipient of the entity MUST NOT ignore any Content-* (e.g. Content-Range) headers that it does not - understand or implement and MUST return a 501 (Not Implemented) response in such cases. - - - - - - The TRACE method is used to invoke a remote, application-layer loop- back of the request message. - - - - - Priority for log entries - - - - - - Very detailed logs to be able to follow the flow of the program. - - - - - Logs to help debug errors in the application - - - - - Information to be able to keep track of state changes etc. - - - - - Something did not go as we expected, but it's no problem. - - - - - Something that should not fail failed, but we can still keep - on going. - - - - - Something failed, and we cannot handle it properly. - - - - - Interface used to write to log files. - - - - - Write an entry to the log file. - - object that is writing to the log - importance of the log message - the message - - - - This class writes to the console. It colors the output depending on the logprio and includes a 3-level stacktrace (in debug mode) - - - - - - The actual instance of this class. - - - - - Logwriters the specified source. - - object that wrote the logentry. - Importance of the log message - The message. - - - - Get color for the specified logprio - - prio for the log entry - A for the prio - - - - Default log writer, writes everything to null (nowhere). - - - - - - The logging instance. - - - - - Writes everything to null - - object that wrote the log entry. - Importance of the log message - The message. - - - - Inversion of control interface. - - - - - Add a component instance - - Interface type - Instance to add - - - - Get a component. - - Interface type - Component if registered, otherwise null. - - Component will get created if needed. - - - - - Checks if the specified component interface have been added. - - - true if found; otherwise false. - - - - Add a component. - - Type being requested. - Type being created. - - - - Returns item either from a form or a query string (checks them in that order) - - - - Representation of a non-initialized HttpParam - - - Initialises the class to hold a value either from a post request or a querystring request - - - - The add method is not availible for HttpParam - since HttpParam checks both Request.Form and Request.QueryString - - name identifying the value - value to add - - - - - Checks whether the form or querystring has the specified value - - Name, case sensitive - true if found; otherwise false. - - - - Returns an enumerator that iterates through the collection. - - - - A that can be used to iterate through the collection. - - 1 - - - - Returns an enumerator that iterates through a collection. - - - - An object that can be used to iterate through the collection. - - 2 - - - - Fetch an item from the form or querystring (in that order). - - - Item if found; otherwise HttpInputItem.EmptyLanguageNode - - - Container for posted form data - - - Instance to help mark a non-initialized form - - - Initializes a form container with the specified name - - - - Makes a deep copy of the input - - The input to copy - - - - Adds a file to the collection of posted files - - The file to add - If the file is already added - If file is null - If the instance is HttpForm.EmptyForm which cannot be modified - - - - Checks if the form contains a specified file - - Field name of the file parameter - True if the file exists - If the instance is HttpForm.EmptyForm which cannot be modified - - - - Retrieves a file held by by the form - - The identifier of the file - The requested file or null if the file was not found - If name is null or empty - If the instance is HttpForm.EmptyForm which cannot be modified - - - Disposes all held HttpFile's and resets values - - - - Retrieves the number of files added to the - - 0 if no files are added - - - - The object form class takes an object and creates form items for it. - - - - - Initializes a new instance of the class. - - - form name *and* id. - action to do when form is posted. - - - - - Initializes a new instance of the class. - - form name *and* id. - action to do when form is posted. - object to get values from - - - - Initializes a new instance of the class. - - form action. - object to get values from. - - - - write out the FORM-tag. - - generated html code - - - - Writeout the form tag - - form should be posted through ajax. - generated html code - - - - Generates a text box. - - - - generated html code - - - - password box - - - - generated html code - - - - Hiddens the specified property name. - - Name of the property. - The options. - generated html code - - - - Labels the specified property name. - - property in object. - caption - generated html code - - - - Generate a checkbox - - property in object - checkbox value - additional html attributes. - generated html code - - - - Write a html select tag - - object property. - id column - The title column. - The options. - - - - - Selects the specified property name. - - Name of the property. - The items. - The id column. - The title column. - The options. - - - - - Write a submit tag. - - button caption - html submit tag - - - - html end form tag - - html - - - - This provider is used to let us implement any type of form decoding we want without - having to rewrite anything else in the server. - - - - - - - Should contain boundary and type, as in: multipart/form-data; boundary=---------------------------230051238959 - Stream containing form data. - Encoding used when decoding the stream - if no parser was found. - If stream is null or not readable. - If stream contents cannot be decoded properly. - - - - Add a decoder. - - - - - - - Number of added decoders. - - - - - Use with care. - - - - - Decoder used for unknown content types. - - - - - We dont want to let the server to die due to exceptions thrown in worker threads. - therefore we use this delegate to give you a change to handle uncaught exceptions. - - Class that the exception was thrown in. - Exception - - Server will throw a InternalServerException in release version if you dont - handle this delegate. - - - - - Contains a connection to a browser/client. - - - Remember to after you have hooked the event. - - TODO: Maybe this class should be broken up into HttpClientChannel and HttpClientContext? - - - - Initializes a new instance of the class. - - true if the connection is secured (SSL/TLS) - client that connected. - Stream used for communication - Used to create a . - Size of buffer to use when reading data. Must be at least 4096 bytes. - If fails - Stream must be writable and readable. - - - - Process incoming body bytes. - - - Bytes - - - - - - - - - - - Start reading content. - - - Make sure to call base.Start() if you override this method. - - - - - Clean up context. - - - Make sure to call base.Cleanup() if you override the method. - - - - - Disconnect from client - - error to report in the event. - - - - Send a response. - - Either or - HTTP status code - reason for the status code. - HTML body contents, can be null or empty. - A content type to return the body as, i.e. 'text/html' or 'text/plain', defaults to 'text/html' if null or empty - If is invalid. - - - - Send a response. - - Either or - HTTP status code - reason for the status code. - - - - Send a response. - - - - - - send a whole buffer - - buffer to send - - - - - Send data using the stream - - Contains data to send - Start position in buffer - number of bytes to send - - - - - - This context have been cleaned, which means that it can be reused. - - - - - Context have been started (a new client have connected) - - - - - Overload to specify own type. - - - Must be specified before the context is being used. - - - - - Using SSL or other encryption method. - - - - - Using SSL or other encryption method. - - - - - Specify which logger to use. - - - - - Gets or sets the network stream. - - - - - Gets or sets IP address that the client connected from. - - - - - Gets or sets port that the client connected from. - - - - - The context have been disconnected. - - - Event can be used to clean up a context, or to reuse it. - - - - - A request have been received in the context. - - - - - Helpers to make XML handling easier - - - - - Serializes object to XML. - - object to serialize. - XML - - Removes name spaces and adds indentation - - - - - Create an object from a XML string - - Type of object - XML string - object - - - - Can handle application/x-www-form-urlencoded - - - - - - Stream containing the content - Content type (with any additional info like boundry). Content type is always supplied in lower case - Stream encoding - - A HTTP form, or null if content could not be parsed. - - If contents in the stream is not valid input data. - - - - Checks if the decoder can handle the mime type - - Content type (with any additional info like boundry). Content type is always supplied in lower case. - True if the decoder can parse the specified content type - - - - Invoked when a client have been accepted by the - - - Can be used to revoke incoming connections - - - - - Initializes a new instance of the class. - - The socket. - - - - Client may not be handled. - - - - - Accepted socket. - - - - - Client should be revoked. - - - - - Arguments sent when a is cleared - - - - - Instantiates the arguments for the event - - True if the session is cleared due to expiration - - - - Returns true if the session is cleared due to expiration - - - - - Delegate for when a IHttpSession is cleared - - this is being cleared. - Arguments for the clearing - - - - Event arguments used when a new header have been parsed. - - - - - Initializes a new instance of the class. - - Name of header. - Header value. - - - - Initializes a new instance of the class. - - - - - Gets or sets header name. - - - - - Gets or sets header value. - - - - Class to handle loading of resource files - - - - Initializes a new instance of the class. - - - - - Initializes a new instance of the class. - - logger. - - - - Loads resources from a namespace in the given assembly to an URI - - The URI to map the resources to - The assembly in which the resources reside - The namespace from which to load the resources - - - resourceLoader.LoadResources("/user/", typeof(User).Assembly, "MyLib.Models.User.Views"); - - Will make the resource MyLib.Models.User.Views.list.Haml accessible via /user/list.haml or /user/list/ - - The amount of loaded files, giving you the possibility of making sure the resources needed gets loaded - If a resource has already been mapped to an uri - - - - Retrieves a stream for the specified resource path if loaded otherwise null - - Path to the resource to retrieve a stream for - A stream or null if the resource couldn't be found - - - - Fetch all files from the resource that matches the specified arguments. - - The path to the resource to extract - - a list of files if found; or an empty array if no files are found. - - Search path must end with an asterisk for finding arbitrary files - - - - Fetch all files from the resource that matches the specified arguments. - - Where the file should reside. - Files to check - - a list of files if found; or an empty array if no files are found. - - - - - Returns whether or not the loader has an instance of the file requested - - The name of the template/file - True if the loader can provide the file - - - - redirects from one URL to another. - - - - - Rules are used to perform operations before a request is being handled. - Rules can be used to create routing etc. - - - - - Process the incoming request. - - incoming HTTP request - outgoing HTTP response - true if response should be sent to the browser directly (no other rules or modules will be processed). - - returning true means that no modules will get the request. Returning true is typically being done - for redirects. - - If request or response is null. - - - - Initializes a new instance of the class. - - Absolute path (no server name) - Absolute path (no server name) - - server.Add(new RedirectRule("/", "/user/index")); - - - - - Initializes a new instance of the class. - - Absolute path (no server name) - Absolute path (no server name) - true if request should be redirected, false if the request URI should be replaced. - - server.Add(new RedirectRule("/", "/user/index")); - - - - - Process the incoming request. - - incoming HTTP request - outgoing HTTP response - true if response should be sent to the browser directly (no other rules or modules will be processed). - - returning true means that no modules will get the request. Returning true is typically being done - for redirects. - - - - - Gets string to match request URI with. - - Is compared to request.Uri.AbsolutePath - - - - Gets where to redirect. - - - - - Gets whether server should redirect client. - - - false means that the rule will replace - the current request URI with the new one from this class. - true means that a redirect response is sent to the client. - - - - - Used to queue incoming requests. - - - - - Initializes a new instance of the class. - - Called when a request should be processed. - - - - Used to process queued requests. - - - - - Gets or sets maximum number of allowed simultaneous requests. - - - - - Gets or sets maximum number of requests queuing to be handled. - - - - - Specifies how many requests the HTTP server is currently processing. - - - - - Used two queue incoming requests to avoid - thread starvation. - - - - - Method used to process a queued request - - Context that the request was received from. - Request to process. - - - - Parses a HTTP request directly from a stream - - - - - Event driven parser used to parse incoming HTTP requests. - - - The parser supports partial messages and keeps the states between - each parsed buffer. It's therefore important that the parser gets - ed if a client disconnects. - - - - - Parse partial or complete message. - - buffer containing incoming bytes - where in buffer that parsing should start - number of bytes to parse - Unparsed bytes left in buffer. - BadRequestException. - - - - Clear parser state. - - - - - Current state in parser. - - - - - A request have been successfully parsed. - - - - - More body bytes have been received. - - - - - Request line have been received. - - - - - A header have been received. - - - - - Gets or sets the log writer. - - - - - Create a new request parser - - delegate receiving log entries. - - - - Add a number of bytes to the body - - buffer containing more body bytes. - starting offset in buffer - number of bytes, from offset, to read. - offset to continue from. - - - - Remove all state information for the request. - - - - - Parse request line - - - If line is incorrect - Expects the following format: "Method SP Request-URI SP HTTP-Version CRLF" - - - - We've parsed a new header. - - Name in lower case - Value, unmodified. - If content length cannot be parsed. - - - - Parse a message - - bytes to parse. - where in buffer that parsing should start - number of bytes to parse, starting on . - offset (where to start parsing next). - BadRequestException. - - - - Gets or sets the log writer. - - - - - Current state in parser. - - - - - A request have been successfully parsed. - - - - - More body bytes have been received. - - - - - Request line have been received. - - - - - A header have been received. - - - - - Response that is sent back to the web browser / client. - - A response can be sent if different ways. The easiest one is - to just fill the Body stream with content, everything else - will then be taken care of by the framework. The default content-type - is text/html, you should change it if you send anything else. - - The second and slighty more complex way is to send the response - as parts. Start with sending the header using the SendHeaders method and - then you can send the body using SendBody method, but do not forget - to set ContentType and ContentLength before doing so. - - - public void MyHandler(IHttpRequest request, IHttpResponse response) - { - - } - - - - - Add another header to the document. - - Name of the header, case sensitive, use lower cases. - Header values can span over multiple lines as long as each line starts with a white space. New line chars should be \r\n - If headers already been sent. - If value conditions have not been met. - Adding any header will override the default ones and those specified by properties. - - - - Send headers and body to the browser. - - If content have already been sent. - - - - Make sure that you have specified ContentLength and sent the headers first. - - - If headers have not been sent. - - offest of first byte to send - number of bytes to send. - - - This method can be used if you want to send body contents without caching them first. This - is recommended for larger files to keep the memory usage low. - - - - Make sure that you have specified ContentLength and sent the headers first. - - - If headers have not been sent. - - - - This method can be used if you want to send body contents without caching them first. This - is recommended for larger files to keep the memory usage low. - - - - Send headers to the client. - - If headers already been sent. - - - - - - - Redirect client to somewhere else using the 302 status code. - - Destination of the redirect - If headers already been sent. - You can not do anything more with the request when a redirect have been done. This should be your last - action. - - - - redirect to somewhere - - where the redirect should go - - No body are allowed when doing redirects. - - - - - The body stream is used to cache the body contents - before sending everything to the client. It's the simplest - way to serve documents. - - - - - Defines the version of the HTTP Response for applications where it's required - for this to be forced. - - - - - The chunked encoding modifies the body of a message in order to - transfer it as a series of chunks, each with its own size indicator, - followed by an OPTIONAL trailer containing entity-header fields. This - allows dynamically produced content to be transferred along with the - information necessary for the recipient to verify that it has - received the full message. - - - - - Kind of connection - - - - - Encoding to use when sending stuff to the client. - - Default is UTF8 - - - - Number of seconds to keep connection alive - - Only used if Connection property is set to ConnectionType.KeepAlive - - - - Status code that is sent to the client. - - Default is HttpStatusCode.Ok - - - - Information about why a specific status code was used. - - - - - Size of the body. MUST be specified before sending the header, - unless property Chunked is set to true. - - - - - Kind of content in the body - - Default is text/html - - - - Headers have been sent to the client- - - You can not send any additional headers if they have already been sent. - - - - The whole response have been sent. - - - - - Cookies that should be created/changed. - - - - - Type of HTTP connection - - - - - Connection is closed after each request-response - - - - - Connection is kept alive for X seconds (unless another request have been made) - - - - - Response that is sent back to the web browser / client. - - - - A response can be sent if different ways. The easiest one is - to just fill the Body stream with content, everything else - will then be taken care of by the framework. The default content-type - is text/html, you should change it if you send anything else. - - The second and slightly more complex way is to send the response - as parts. Start with sending the header using the SendHeaders method and - then you can send the body using SendBody method, but do not forget - to set and before doing so. - - - - - // Example using response body. - class MyModule : HttpModule - { - public override bool Process(IHttpRequest request, IHttpResponse response, IHttpSession session) - { - StreamWriter writer = new StreamWriter(response.Body); - writer.WriteLine("Hello dear World!"); - writer.Flush(); - - // return true to tell webserver that we've handled the url - return true; - } - } - - - todo: add two examples, using SendHeaders/SendBody and just the Body stream. - - - - Initializes a new instance of the class. - - Client that send the . - Contains information of what the client want to receive. - cannot be empty. - - - - Initializes a new instance of the class. - - Client that send the . - Version of HTTP protocol that the client uses. - Type of HTTP connection used. - - - - Add another header to the document. - - Name of the header, case sensitive, use lower cases. - Header values can span over multiple lines as long as each line starts with a white space. New line chars should be \r\n - If headers already been sent. - If value conditions have not been met. - Adding any header will override the default ones and those specified by properties. - - - - Send headers and body to the browser. - - If content have already been sent. - - - - Make sure that you have specified and sent the headers first. - - - If headers have not been sent. - - offset of first byte to send - number of bytes to send. - - - This method can be used if you want to send body contents without caching them first. This - is recommended for larger files to keep the memory usage low. - - - - Make sure that you have specified and sent the headers first. - - - If headers have not been sent. - - - - This method can be used if you want to send body contents without caching them first. This - is recommended for larger files to keep the memory usage low. - - - - Send headers to the client. - - If headers already been sent. - - - - - - - Redirect client to somewhere else using the 302 status code. - - Destination of the redirect - If headers already been sent. - You can not do anything more with the request when a redirect have been done. This should be your last - action. - - - - redirect to somewhere - - where the redirect should go - - No body are allowed when doing redirects. - - - - - The body stream is used to cache the body contents - before sending everything to the client. It's the simplest - way to serve documents. - - - - - The chunked encoding modifies the body of a message in order to - transfer it as a series of chunks, each with its own size indicator, - followed by an OPTIONAL trailer containing entity-header fields. This - allows dynamically produced content to be transferred along with the - information necessary for the recipient to verify that it has - received the full message. - - - - - Defines the version of the HTTP Response for applications where it's required - for this to be forced. - - - - - Kind of connection - - - - - Encoding to use when sending stuff to the client. - - Default is UTF8 - - - - Number of seconds to keep connection alive - - Only used if Connection property is set to . - - - - Status code that is sent to the client. - - Default is - - - - Information about why a specific status code was used. - - - - - Size of the body. MUST be specified before sending the header, - unless property Chunked is set to true. - - - - - Kind of content in the body - - Default type is "text/html" - - - - Headers have been sent to the client- - - You can not send any additional headers if they have already been sent. - - - - The whole response have been sent. - - - - - Cookies that should be created/changed. - - - - - represents a HTTP input item. Each item can have multiple sub items, a sub item - is made in a HTML form by using square brackets - - - // becomes: - Console.WriteLine("Value: {0}", form["user"]["FirstName"].Value); - - - All names in a form SHOULD be in lowercase. - - - - Representation of a non-initialized . - - - - Initializes an input item setting its name/identifier and value - - Parameter name/id - Parameter value - - - Creates a deep copy of the item specified - The item to copy - The function makes a deep copy of quite a lot which can be slow - - - - Add another value to this item - - Value to add. - Cannot add stuff to . - - - - checks if a sub-item exists (and has a value). - - name in lower case - true if the sub-item exists and has a value; otherwise false. - - - Returns a formatted representation of the instance with the values of all contained parameters - - - - Outputs the string in a formatted manner - - A prefix to append, used internally - produce a query string - - - - Add a sub item. - - Can contain array formatting, the item is then parsed and added in multiple levels - Value to add. - Argument is null. - Cannot add stuff to . - - - - Returns an enumerator that iterates through the collection. - - - - A that can be used to iterate through the collection. - - 1 - - - - Returns an enumerator that iterates through a collection. - - - - An object that can be used to iterate through the collection. - - 2 - - - - Outputs the string in a formatted manner - - A prefix to append, used internally - - - - - Number of values - - - - - Get a sub item - - name in lower case. - if no item was found. - - - - Name of item (in lower case). - - - - - Returns the first value, or null if no value exist. - - - - - Returns the last value, or null if no value exist. - - - - - Returns the list with values. - - - - - - - name in lower case - - - - - Helpers making it easier to work with forms. - - - - - - Used to let the website use different JavaScript libraries. - Default is - - - - - Create a <form> tag. - - name of form - action to invoke on submit - form should be posted as Ajax - HTML code - - - // without options - WebHelper.FormStart("frmLogin", "/user/login", Request.IsAjax); - - // with options - WebHelper.FormStart("frmLogin", "/user/login", Request.IsAjax, "style", "display:inline", "class", "greenForm"); - - - HTML attributes or JavaScript options. - Method will ALWAYS be POST. - options must consist of name, value, name, value - - - - Creates a select list with the values in a collection. - - Name of the SELECT-tag - collection used to generate options. - delegate used to return id and title from objects. - value that should be marked as selected. - First row should contain an empty value. - string containing a SELECT-tag. - - - - - Creates a select list with the values in a collection. - - Name of the SELECT-tag - Id of the SELECT-tag - collection used to generate options. - delegate used to return id and title from objects. - value that should be marked as selected. - First row should contain an empty value. - string containing a SELECT-tag. - - - - // Class that is going to be used in a SELECT-tag. - public class User - { - private readonly string _realName; - private readonly int _id; - public User(int id, string realName) - { - _id = id; - _realName = realName; - } - public string RealName - { - get { return _realName; } - } - - public int Id - { - get { return _id; } - } - } - - // Using an inline delegate to generate the select list - public void UserInlineDelegate() - { - List<User> items = new List<User>(); - items.Add(new User(1, "adam")); - items.Add(new User(2, "bertial")); - items.Add(new User(3, "david")); - string htmlSelect = Select("users", "users", items, delegate(object o, out object id, out object value) - { - User user = (User)o; - id = user.Id; - value = user.RealName; - }, 2, true); - } - - // Using an method as delegate to generate the select list. - public void UseExternalDelegate() - { - List<User> items = new List<User>(); - items.Add(new User(1, "adam")); - items.Add(new User(2, "bertial")); - items.Add(new User(3, "david")); - string htmlSelect = Select("users", "users", items, UserOptions, 1, true); - } - - // delegate returning id and title - public static void UserOptions(object o, out object id, out object title) - { - User user = (User)o; - id = user.Id; - value = user.RealName; - } - - - name, id, collection or getIdTitle is null. - - - - Creates a select list with the values in a collection. - - Name of the SELECT-tag - Id of the SELECT-tag - collection used to generate options. - delegate used to return id and title from objects. - value that should be marked as selected. - First row should contain an empty value. - name, value collection of extra HTML attributes. - string containing a SELECT-tag. - - name, id, collection or getIdTitle is null. - Invalid HTML attribute list. - - - - Generate a list of HTML options - - collection used to generate options. - delegate used to return id and title from objects. - value that should be marked as selected. - First row should contain an empty value. - - collection or getIdTitle is null. - - - sb is null. - - - - Creates a check box. - - element name - element value - determines if the check box is selected or not. This is done differently depending on the - type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if - the box is checked or not. - a list with additional attributes (name, value, name, value). - a generated radio button - - - - Creates a check box. - - element name - element id - element value - determines if the check box is selected or not. This is done differently depending on the - type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if - the box is checked or not. - a list with additional attributes (name, value, name, value). - a generated radio button - - value in your business object. (check box will be selected if it matches the element value) - - - - - Creates a check box. - - element name - element id - determines if the check box is selected or not. This is done differently depending on the - type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if - the box is checked or not. - a list with additional attributes (name, value, name, value). - a generated radio button - will set value to "1". - - - - Creates a RadioButton. - - element name - element value - determines if the radio button is selected or not. This is done differently depending on the - type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if - the box is checked or not. - a list with additional attributes (name, value, name, value). - a generated radio button - - - - Creates a RadioButton. - - element name - element id - element value - determines if the radio button is selected or not. This is done differently depending on the - type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if - the box is checked or not. - a list with additional attributes (name, value, name, value). - a generated radio button - - - - form close tag - - - - - - Add a component instance - - Interface type - Instance to add - - - - Get a component. - - Interface type - Component if registered, otherwise null. - - Component will get created if needed. - - - - If instance cannot be created. - - - - Checks if the specified component interface have been added. - - - true if found; otherwise false. - - - - Add a component. - - Type being requested. - Type being created. - Type have already been mapped. - - - - Arguments used when more body bytes have come. - - - - - Initializes a new instance of the class. - - buffer that contains the received bytes. - offset in buffer where to start processing. - number of bytes from that should be parsed. - - - - Initializes a new instance of the class. - - - - - Gets or sets buffer that contains the received bytes. - - - - - Gets or sets number of bytes from that should be parsed. - - - - - Gets or sets offset in buffer where to start processing. - - - - - Used to create and reuse contexts. - - - - - Used to create es. - - - - - Creates a that handles a connected client. - - Client socket (accepted by the ). - A creates . - - - - Create a secure . - - Client socket (accepted by the ). - HTTPS certificate to use. - Kind of HTTPS protocol. Usually TLS or SSL. - A created . - - - - A request have been received from one of the contexts. - - - - - Initializes a new instance of the class. - - The writer. - Amount of bytes to read from the incoming socket stream. - Used to create a request parser. - - - - Create a new context. - - true if socket is running HTTPS. - Client that connected - Network/SSL stream. - A context. - - - - Create a new context. - - true if HTTPS is used. - Remote client - Network stream, uses . - A new context (always). - - - - Create a secure . - - Client socket (accepted by the ). - HTTPS certificate to use. - Kind of HTTPS protocol. Usually TLS or SSL. - - A created . - - - - - Creates a that handles a connected client. - - Client socket (accepted by the ). - - A creates . - - - - - True if detailed trace logs should be written. - - - - - A request have been received from one of the contexts. - - - - - Custom network stream to mark sockets as reusable when disposing the stream. - - - - - Creates a new instance of the class for the specified . - - - The that the will use to send and receive data. - - - The parameter is null. - - - The parameter is not connected. - -or- - The property of the parameter is not . - -or- - The parameter is in a nonblocking state. - - - - - Initializes a new instance of the class for the specified with the specified ownership. - - - The that the will use to send and receive data. - - - Set to true to indicate that the will take ownership of the ; otherwise, false. - - - The parameter is null. - - - The parameter is not connected. - -or- - the value of the property of the parameter is not . - -or- - the parameter is in a nonblocking state. - - - - - Creates a new instance of the class for the specified with the specified access rights. - - - The that the will use to send and receive data. - - - A bitwise combination of the values that specify the type of access given to the over the provided . - - - The parameter is null. - - - The parameter is not connected. - -or- - the property of the parameter is not . - -or- - the parameter is in a nonblocking state. - - - - - Creates a new instance of the class for the specified with the specified access rights and the specified ownership. - - - The that the will use to send and receive data. - - - A bitwise combination of the values that specifies the type of access given to the over the provided . - - - Set to true to indicate that the will take ownership of the ; otherwise, false. - - - The parameter is null. - - - The parameter is not connected. - -or- - The property of the parameter is not . - -or- - The parameter is in a nonblocking state. - - - - - Closes the current stream and releases any resources (such as sockets and file handles) associated with the current stream. - - - - - Releases the unmanaged resources used by the and optionally releases the managed resources. - - true to release both managed and unmanaged resources; false to release only unmanaged resources. - - - - Serves files that are stored in embedded resources. - - - - - A HttpModule can be used to serve Uri's. The module itself - decides if it should serve a Uri or not. In this way, you can - get a very flexible http application since you can let multiple modules - serve almost similar urls. - - - Throw if you are using a and want to prompt for user name/password. - - - - - Method that process the url - - Information sent by the browser about the request - Information that is being sent back to the client. - Session used to - true if this module handled the request. - - - - Set the log writer to use. - - logwriter to use. - - - - Log something. - - importance of log message - message - - - - If true specifies that the module doesn't consume the processing of a request so that subsequent modules - can continue processing afterwards. Default is false. - - - - - Initializes a new instance of the class. - Runs to make sure the basic mime types are available, they can be cleared later - through the use of if desired. - - - - - Initializes a new instance of the class. - Runs to make sure the basic mime types are available, they can be cleared later - through the use of if desired. - - The log writer to use when logging events - - - - Mimtypes that this class can handle per default - - - - - Loads resources from a namespace in the given assembly to an uri - - The uri to map the resources to - The assembly in which the resources reside - The namespace from which to load the resources - - resourceLoader.LoadResources("/user/", typeof(User).Assembly, "MyLib.Models.User.Views"); - - will make ie the resource MyLib.Models.User.Views.stylesheet.css accessible via /user/stylesheet.css - - The amount of loaded files, giving you the possibility of making sure the resources needed gets loaded - - - - Returns true if the module can handle the request - - - - - Method that process the url - - Information sent by the browser about the request - Information that is being sent back to the client. - Session used to - true if this module handled the request. - - - - List with all mime-type that are allowed. - - All other mime types will result in a Forbidden http status code. - - - - The purpose of this module is to serve files. - - - - - Initializes a new instance of the class. - - Uri to serve, for instance "/files/" - Path on hard drive where we should start looking for files - If true a Last-Modifed header will be sent upon requests urging web browser to cache files - - - - Initializes a new instance of the class. - - Uri to serve, for instance "/files/" - Path on hard drive where we should start looking for files - - - - Mimtypes that this class can handle per default - - - - - Determines if the request should be handled by this module. - Invoked by the - - - true if this module should handle it. - - - Illegal path - - - - check if source contains any of the chars. - - - - - - - - Method that process the Uri. - - Information sent by the browser about the request - Information that is being sent back to the client. - Session used to - Failed to find file extension - File type is forbidden. - - - - return a file extension from an absolute Uri path (or plain filename) - - - - - - - List with all mime-type that are allowed. - - All other mime types will result in a Forbidden http status code. - - - - characters that may not exist in a path. - - - fileMod.ForbiddenChars = new string[]{ "\\", "..", ":" }; - - - - - The server encountered an unexpected condition which prevented it from fulfilling the request. - - - - - Initializes a new instance of the class. - - - - - Initializes a new instance of the class. - - error message. - - - - Initializes a new instance of the class. - - error message. - inner exception. - - - - Class to make dynamic binding of redirects. Instead of having to specify a number of similar redirect rules - a regular expression can be used to identify redirect URLs and their targets. - - - [a-z0-9]+)", "/users/${target}?find=true", RegexOptions.IgnoreCase) - ]]> - - - - - Initializes a new instance of the class. - - Expression to match URL - Expression to generate URL - - [a-zA-Z0-9]+)", "/user/${first}")); - Result of ie. /employee1 will then be /user/employee1 - ]]> - - - - - Initializes a new instance of the class. - - Expression to match URL - Expression to generate URL - Regular expression options to use, can be null - - [a-zA-Z0-9]+)", "/user/{first}", RegexOptions.IgnoreCase)); - Result of ie. /employee1 will then be /user/employee1 - ]]> - - - - - Initializes a new instance of the class. - - Expression to match URL - Expression to generate URL - Regular expression options to apply - true if request should be redirected, false if the request URI should be replaced. - - [a-zA-Z0-9]+)", "/user/${first}", RegexOptions.None)); - Result of ie. /employee1 will then be /user/employee1 - ]]> - - Argument is null. - - - - - Process the incoming request. - - incoming HTTP request - outgoing HTTP response - true if response should be sent to the browser directly (no other rules or modules will be processed). - - returning true means that no modules will get the request. Returning true is typically being done - for redirects. - - If request or response is null - - - - Used when the request line have been successfully parsed. - - - - - Initializes a new instance of the class. - - The HTTP method. - The URI path. - The HTTP version. - - - - Initializes a new instance of the class. - - - - - Gets or sets http method. - - - Should be one of the methods declared in . - - - - - Gets or sets the version of the HTTP protocol that the client want to use. - - - - - Gets or sets requested URI path. - - Delegate used to find a realm/domain. @@ -3982,360 +322,1511 @@ exceptions will be printed to console and suppressed during release mode. - + - The request requires user authentication. The response MUST include a - WWW-Authenticate header field (section 14.47) containing a challenge - applicable to the requested resource. - - The client MAY repeat the request with a suitable Authorization header - field (section 14.8). If the request already included Authorization - credentials, then the 401 response indicates that authorization has been - refused for those credentials. If the 401 response contains the same challenge - as the prior response, and the user agent has already attempted authentication - at least once, then the user SHOULD be presented the entity that was given in the response, - since that entity might include relevant diagnostic information. - - HTTP access authentication is explained in rfc2617: - http://www.ietf.org/rfc/rfc2617.txt - - (description is taken from - http://www.submissionchamber.com/help-guides/error-codes.php#sec10.4.2) + Serves files that are stored in embedded resources. - + - Create a new unauhtorized exception. - - - - - - Create a new unauhtorized exception. - - reason to why the request was unauthorized. - inner exception - - - - Create a new unauhtorized exception. - - reason to why the request was unauthorized. - - - - Lists content type mime types. - - - - - text/plain - - - - - text/haml - - - - - content type for javascript documents = application/javascript + A HttpModule can be used to serve Uri's. The module itself + decides if it should serve a Uri or not. In this way, you can + get a very flexible http application since you can let multiple modules + serve almost similar urls. - - RFC 4329 states that text/javascript have been superseeded by - application/javascript. You might still want to check browser versions - since older ones do not support application/javascript. - - Browser support: http://krijnhoetmer.nl/stuff/javascript/mime-types/ + Throw if you are using a and want to prompt for user name/password. - + - text/xml + Method that process the url + + Information sent by the browser about the request + Information that is being sent back to the client. + Session used to + true if this module handled the request. + + + + Set the log writer to use. + + logwriter to use. + + + + Log something. + + importance of log message + message + + + + If true specifies that the module doesn't consume the processing of a request so that subsequent modules + can continue processing afterwards. Default is false. - + - A list of content types + Initializes a new instance of the class. + Runs to make sure the basic mime types are available, they can be cleared later + through the use of if desired. - + + Initializes a new instance of the class. + Runs to make sure the basic mime types are available, they can be cleared later + through the use of if desired. + + The log writer to use when logging events + + + + Mimtypes that this class can handle per default + + + + + Loads resources from a namespace in the given assembly to an uri + + The uri to map the resources to + The assembly in which the resources reside + The namespace from which to load the resources + + resourceLoader.LoadResources("/user/", typeof(User).Assembly, "MyLib.Models.User.Views"); - - Semicolon separated content types. + will make ie the resource MyLib.Models.User.Views.stylesheet.css accessible via /user/stylesheet.css + + The amount of loaded files, giving you the possibility of making sure the resources needed gets loaded - + + + Returns true if the module can handle the request + + + + + Method that process the url + + Information sent by the browser about the request + Information that is being sent back to the client. + Session used to + true if this module handled the request. + + + + List with all mime-type that are allowed. + + All other mime types will result in a Forbidden http status code. + + + + Contains some kind of input from the browser/client. + can be QueryString, form data or any other request body content. + + + + + Base class for request data containers + + + + + Adds a parameter mapped to the presented name + + The name to map the parameter to + The parameter value + + + + Returns true if the container contains the requested parameter + + Parameter id + True if parameter exists + + + + Returns a request parameter + + The name associated with the parameter + + + + Representation of a non-initialized class instance + + + Variable telling the class that it is non-initialized + + + + Initializes a new instance of the class. + + form name. + + + + Initializes a new instance of the class. + + form name. + if set to true all changes will be ignored. + this constructor should only be used by Empty + + + Creates a deep copy of the HttpInput class + The object to copy + The function makes a deep copy of quite a lot which can be slow + + + + Add a new element. Form array elements are parsed + and added in a correct hierarchy. + + Name is converted to lower case. + + name is null. + Cannot add stuff to . + + + + Returns true if the class contains a with the corresponding name. + + The field/query string name + True if the value exists + + + + Parses an item and returns it. + This function is primarily used to parse array items as in user[name]. + + + + + + + Outputs the instance representing all its values joined together + + + + Returns all items as an unescaped query string. + + + + + Extracts one parameter from an array + + Containing the string array + All but the first value + + string test1 = ExtractOne("system[user][extension][id]"); + string test2 = ExtractOne(test1); + string test3 = ExtractOne(test2); + // test1 = user[extension][id] + // test2 = extension[id] + // test3 = id + + + + Resets all data contained by class + + + + Returns an enumerator that iterates through the collection. + + + + A that can be used to iterate through the collection. + + 1 + + Returns an enumerator that iterates through a collection. + - An object that can be used to iterate through the collection. + An object that can be used to iterate through the collection. + 2 - + - Searches for the specified type - - Can also be a part of a type (searching for "xml" would return true for "application/xml"). - true if type was found. - - - - Get this first content type. + Form name as lower case - + - Fetch a content type + Get a form item. - Part of type ("xml" would return "application/xml") - - All content types are in lower case. + + Returns if item was not found. - + - Session store using memory for each session. + The server understood the request, but is refusing to fulfill it. + Authorization will not help and the request SHOULD NOT be repeated. + If the request method was not HEAD and the server wishes to make public why the request has not been fulfilled, + it SHOULD describe the reason for the refusal in the entity. If the server does not wish to make this information + available to the client, the status code 404 (Not Found) can be used instead. + + Text taken from: http://www.submissionchamber.com/help-guides/error-codes.php - + - Initializes the class setting the expirationtimer to clean the session every minute + All HTTP based exceptions will derive this class. - + - Delegate for the cleanup timer + Create a new HttpException + + http status code (sent in the response) + error description + + + + Create a new HttpException + + http status code (sent in the response) + error description + inner exception + + + + status code to use in the response. - + - Creates a new http session + Initializes a new instance of the class. - + error message - + - Creates a new http session with a specific id + This class is created as a wrapper, since there are two different cookie types in .Net (Cookie and HttpCookie). + The framework might switch class in the future and we dont want to have to replace all instances + + + + + Let's copy all the cookies. + + value from cookie header. + + + + Adds a cookie in the collection. + + cookie to add + cookie is null + + + + Gets a collection enumerator on the cookie list. + + collection enumerator + + + + Remove all cookies. + + + + + Returns an enumerator that iterates through the collection. + + + + A that can be used to iterate through the collection. + + 1 + + + + Remove a cookie from the collection. + + Name of cookie. + + + + Gets the count of cookies in the collection. + + + + + Gets the cookie of a given identifier (null if not existing). + + + + + Contains a connection to a browser/client. + + + + + Disconnect from client + + error to report in the event. + + + + Send a response. + + Either or + HTTP status code + reason for the status code. + HTML body contents, can be null or empty. + A content type to return the body as, i.e. 'text/html' or 'text/plain', defaults to 'text/html' if null or empty + If is invalid. + + + + Send a response. + + Either or + HTTP status code + reason for the status code. + + + + Send a response. + + + + + + send a whole buffer + + buffer to send + + + + + Send data using the stream + + Contains data to send + Start position in buffer + number of bytes to send + + + + + + Closes the streams and disposes of the unmanaged resources + + + + + Using SSL or other encryption method. + + + + + Using SSL or other encryption method. + + + + + The context have been disconnected. - Id used to identify the new cookie.. - A object. - Id should be generated by the store implementation if it's null or . + Event can be used to clean up a context, or to reuse it. - + - Load an existing session. - - - - - - - Save an updated session to the store. - - - - - - We use the flyweight pattern which reuses small objects - instead of creating new each time. - - EmptyLanguageNode (unused) session that should be reused next time Create is called. - - - - Remove expired sessions + A request have been received in the context. - + - Remove a session - - id of the session. - - - - Load a session from the store - - - null if session is not found. - - - - Number of minutes before a session expires. - Default is 20 minutes. + A have been disconnected. - + - Webhelper provides helpers for common tasks in HTML. + Initializes a new instance of the class. + + Reason to disconnection. + + + + Gets reason to why client disconnected. - + - Used to let the website use different javascript libraries. - Default is + - + - Creates a link that invokes through ajax. + Initializes a new instance of the class. + + The request. + + + + Gets received request. + + + + + Returns item either from a form or a query string (checks them in that order) + + + + Representation of a non-initialized HttpParam + + + Initialises the class to hold a value either from a post request or a querystring request + + + + The add method is not availible for HttpParam + since HttpParam checks both Request.Form and Request.QueryString + + name identifying the value + value to add + + + + + Checks whether the form or querystring has the specified value + + Name, case sensitive + true if found; otherwise false. + + + + Returns an enumerator that iterates through the collection. + + + + A that can be used to iterate through the collection. + + 1 + + + + Returns an enumerator that iterates through a collection. + + + + An object that can be used to iterate through the collection. + + 2 + + + + Fetch an item from the form or querystring (in that order). + + + Item if found; otherwise HttpInputItem.EmptyLanguageNode + + + + Container class for posted files + + + + + Creates a container for a posted file + + The identifier of the post field + The file path + The content type of the file + The name of the file uploaded + If any parameter is null or empty + + + + Creates a container for a posted file + + If any parameter is null or empty + + + Destructor disposing the file + + + + Deletes the temporary file + + True if manual dispose + + + + Disposing interface, cleans up managed resources (the temporary file) and suppresses finalization + + + + + The name/id of the file + + + + + The full file path + + + + + The name of the uploaded file + + + + + The type of file + + + + + This decoder converts XML documents to form items. + Each element becomes a subitem in the form, and each attribute becomes an item. - url to fetch - link title - - optional options in format "key, value, key, value". - Javascript options starts with ':'. - - a link tag - WebHelper.AjaxRequest("/users/add/", "Add user", "method:", "post", "onclick", "validate('this');"); + // xml: somethingdata + // result: + // form["hello"].Value = "something" + // form["hello"]["id"].Value = 1 + // form["hello"]["world]["id"].Value = 1 + // form["hello"]["world"].Value = "data" + + The original xml document is stored in form["__xml__"].Value. + - + - Builds a link that updates an element with the fetched ajax content. + Interface for form content decoders. - Url to fetch content from - link title - html element to update with the results of the ajax request. - optional options in format "key, value, key, value" - A link tag. - + - A link that pop ups a Dialog (overlay div) + - url to contents of dialog - link title - name/value of html attributes. - A "a"-tag that popups a dialog when clicked - - WebHelper.DialogLink("/user/show/1", "show user", "onmouseover", "alert('booh!');"); - + Stream containing the content + Content type (with any additional info like boundry). Content type is always supplied in lower case + Stream enconding + A http form, or null if content could not be parsed. + If contents in the stream is not valid input data. - + - Create/Open a dialog box using ajax + Checks if the decoder can handle the mime type - - - - + Content type (with any additional info like boundry). Content type is always supplied in lower case. + True if the decoder can parse the specified content type - + - Close a javascript dialog window/div. + - javascript for closing a dialog. - + Stream containing the content + Content type (with any additional info like boundry). Content type is always supplied in lower case + Stream encoding + Note: contentType and encoding are not used? + A http form, or null if content could not be parsed. + - + - Create a <form> tag. + Recursive function that will go through an xml element and store it's content + to the form item. - name of form - action to invoke on submit - form should be posted as ajax - html code - - WebHelper.FormStart("frmLogin", "/user/login", Request.IsAjax); - + (parent) Item in form that content should be added to. + Node that should be parsed. - + - Create a link tag. + Checks if the decoder can handle the mime type - url to go to - link title (text that is displayed) - html attributes, name, value, name, value - html code - - WebHelper.Link("/user/show/1", "Show user", "id", "showUser", "onclick", "return confirm('Are you shure?');"); - + Content type (with any additional info like boundry). Content type is always supplied in lower case. + True if the decoder can parse the specified content type - + - Build a link + The object form class takes an object and creates form items for it. - url to go to. - title of link (displayed text) - extra html attributes. - a complete link - + - Build a link + Initializes a new instance of the class. - url to go to. - title of link (displayed text) - extra html attributes. - a complete link - more options + + form name *and* id. + action to do when form is posted. + - + - Obsolete + Initializes a new instance of the class. - Obsolete - Obsolete - Obsolete - Obsolete - Obsolete - Obsolete + form name *and* id. + action to do when form is posted. + object to get values from - + - Obsolete + Initializes a new instance of the class. - Obsolete - Obsolete - Obsolete - Obsolete - Obsolete - Obsolete - Obsolete + form action. + object to get values from. - + - Render errors into a UL with class "errors" + write out the FORM-tag. - class used by UL-tag. - items to list - an unordered html list. + generated html code - + - Render errors into a UL with class "errors" + Writeout the form tag - class used by UL-tag. - items to list - an unordered html list. + form should be posted through ajax. + generated html code - + - Render errors into a UL with class "errors" + Generates a text box. - - - - - - Generates a list with html attributes. - - StringBuilder that the options should be added to. - attributes set by user. - attributes set by any of the helper classes. - - - - Generates a list with html attributes. - - StringBuilder that the options should be added to. + + generated html code + + + + password box + + + + generated html code + + + + Hiddens the specified property name. + + Name of the property. + The options. + generated html code + + + + Labels the specified property name. + + property in object. + caption + generated html code + + + + Generate a checkbox + + property in object + checkbox value + additional html attributes. + generated html code + + + + Write a html select tag + + object property. + id column + The title column. + The options. + + + + + Selects the specified property name. + + Name of the property. + The items. + The id column. + The title column. + The options. + + + + + Write a submit tag. + + button caption + html submit tag + + + + html end form tag + + html + + + + + + + http://www.faqs.org/rfcs/rfc1867.html + + + + + multipart/form-data + + + + + form-data + + + + + + + Stream containing the content + Content type (with any additional info like boundry). Content type is always supplied in lower case + Stream enconding + A http form, or null if content could not be parsed. + If contents in the stream is not valid input data. + If any parameter is null + + + + Checks if the decoder can handle the mime type + + Content type (with any additional info like boundry). Content type is always supplied in lower case. + True if the decoder can parse the specified content type + + + + The request could not be understood by the server due to malformed syntax. + The client SHOULD NOT repeat the request without modifications. + + Text taken from: http://www.submissionchamber.com/help-guides/error-codes.php + + + + + Create a new bad request exception. + + reason to why the request was bad. + + + + Create a new bad request exception. + + reason to why the request was bad. + inner exception + + + + Cookies that should be set. + + + + + Adds a cookie in the collection. + + cookie to add + cookie is null + + + + Copy a request cookie + + + When the cookie should expire + + + + Gets a collection enumerator on the cookie list. + + collection enumerator + + + + Remove all cookies + + + + + Returns an enumerator that iterates through the collection. + + + + A that can be used to iterate through the collection. + + 1 + + + + Gets the count of cookies in the collection. + + + + + Gets the cookie of a given identifier (null if not existing). + + + + + cookie being sent back to the browser. + + + + + + cookie sent by the client/browser + + + + + + Constructor. + + cookie identifier + cookie content + id or content is null + id is empty + + + + Gets the cookie HTML representation. + + cookie string + + + + Gets the cookie identifier. + + + + + Cookie value. Set to null to remove cookie. + + + + + Constructor. + + cookie identifier + cookie content + cookie expiration date. Use DateTime.MinValue for session cookie. + id or content is null + id is empty + + + + Create a new cookie + + name identifying the cookie + cookie value + when the cookie expires. Setting DateTime.MinValue will delete the cookie when the session is closed. + Path to where the cookie is valid + Domain that the cookie is valid for. + + + + Create a new cookie + + Name and value will be used + when the cookie expires. + + + + Gets the cookie HTML representation. + + cookie string + + + + When the cookie expires. + DateTime.MinValue means that the cookie expires when the session do so. + + + + + Cookie is only valid under this path. + + + + + Inversion of control interface. + + + + + Add a component instance + + Interface type + Instance to add + + + + Get a component. + + Interface type + Component if registered, otherwise null. + + Component will get created if needed. + + + + + Checks if the specified component interface have been added. + + + true if found; otherwise false. + + + + Add a component. + + Type being requested. + Type being created. + + + + Contains a listener that doesn't do anything with the connections. + + + + + Listen for regular HTTP connections + + IP Address to accept connections on + TCP Port to listen on, default HTTP port is 80. + Factory used to create es. + address is null. + Port must be a positive number. + + + + Initializes a new instance of the class. + + IP Address to accept connections on + TCP Port to listen on, default HTTPS port is 443 + Factory used to create es. + Certificate to use + + + + Initializes a new instance of the class. + + IP Address to accept connections on + TCP Port to listen on, default HTTPS port is 443 + Factory used to create es. + Certificate to use + which HTTPS protocol to use, default is TLS. + + + Exception. + + + + Will try to accept connections one more time. + + If any exceptions is thrown. + + + + Can be used to create filtering of new connections. + + Accepted socket + true if connection can be accepted; otherwise false. + + + + Start listen for new connections + + Number of connections that can stand in a queue to be accepted. + Listener have already been started. + + + + Stop the listener + + + + + + Gives you a change to receive log entries for all internals of the HTTP library. + + + You may not switch log writer after starting the listener. + + + + + True if we should turn on trace logs. + + + + + Catch exceptions not handled by the listener. + + + Exceptions will be thrown during debug mode if this event is not used, + exceptions will be printed to console and suppressed during release mode. + + + + + A request have been received from a . + + + + + New implementation of the HTTP listener. + + + Use the Create methods to create a default listener. + + + + + Initializes a new instance of the class. + + IP Address to accept connections on + TCP Port to listen on, default HTTP port is 80. + Factory used to create es. + address is null. + Port must be a positive number. + + + + Initializes a new instance of the class. + + The address. + The port. + The factory. + The certificate. + + + + Initializes a new instance of the class. + + The address. + The port. + The factory. + The certificate. + The protocol. + + + + Creates a new instance with default factories. + + Address that the listener should accept connections on. + Port that listener should accept connections on. + Created HTTP listener. + + + + Creates a new instance with default factories. + + Address that the listener should accept connections on. + Port that listener should accept connections on. + Certificate to use + Created HTTP listener. + + + + Creates a new instance with default factories. + + Address that the listener should accept connections on. + Port that listener should accept connections on. + Certificate to use + which HTTPS protocol to use, default is TLS. + Created HTTP listener. + + + + Can be used to create filtering of new connections. + + Accepted socket + + true if connection can be accepted; otherwise false. + + + + + A client have been accepted, but not handled, by the listener. + + + + + redirects from one URL to another. + + + + + Rules are used to perform operations before a request is being handled. + Rules can be used to create routing etc. + + + + + Process the incoming request. + + incoming HTTP request + outgoing HTTP response + true if response should be sent to the browser directly (no other rules or modules will be processed). + + returning true means that no modules will get the request. Returning true is typically being done + for redirects. + + If request or response is null. + + + + Initializes a new instance of the class. + + Absolute path (no server name) + Absolute path (no server name) + + server.Add(new RedirectRule("/", "/user/index")); + + + + + Initializes a new instance of the class. + + Absolute path (no server name) + Absolute path (no server name) + true if request should be redirected, false if the request URI should be replaced. + + server.Add(new RedirectRule("/", "/user/index")); + + + + + Process the incoming request. + + incoming HTTP request + outgoing HTTP response + true if response should be sent to the browser directly (no other rules or modules will be processed). + + returning true means that no modules will get the request. Returning true is typically being done + for redirects. + + + + + Gets string to match request URI with. + + Is compared to request.Uri.AbsolutePath + + + + Gets where to redirect. + + + + + Gets whether server should redirect client. + + + false means that the rule will replace + the current request URI with the new one from this class. + true means that a redirect response is sent to the client. + + + + + Parses a HTTP request directly from a stream + + + + + Event driven parser used to parse incoming HTTP requests. + + + The parser supports partial messages and keeps the states between + each parsed buffer. It's therefore important that the parser gets + ed if a client disconnects. + + + + + Parse partial or complete message. + + buffer containing incoming bytes + where in buffer that parsing should start + number of bytes to parse + Unparsed bytes left in buffer. + BadRequestException. + + + + Clear parser state. + + + + + Current state in parser. + + + + + A request have been successfully parsed. + + + + + More body bytes have been received. + + + + + Request line have been received. + + + + + A header have been received. + + + + + Gets or sets the log writer. + + + + + Create a new request parser + + delegate receiving log entries. + + + + Add a number of bytes to the body + + buffer containing more body bytes. + starting offset in buffer + number of bytes, from offset, to read. + offset to continue from. + + + + Remove all state information for the request. + + + + + Parse request line + + + If line is incorrect + Expects the following format: "Method SP Request-URI SP HTTP-Version CRLF" + + + + We've parsed a new header. + + Name in lower case + Value, unmodified. + If content length cannot be parsed. + + + + Parse a message + + bytes to parse. + where in buffer that parsing should start + number of bytes to parse, starting on . + offset (where to start parsing next). + BadRequestException. + + + + Gets or sets the log writer. + + + + + Current state in parser. + + + + + A request have been successfully parsed. + + + + + More body bytes have been received. + + + + + Request line have been received. + + + + + A header have been received. + + + + + Contains server side HTTP request information. + + + + + Called during parsing of a . + + Name of the header, should not be URL encoded + Value of the header, should not be URL encoded + If a header is incorrect. + + + + Add bytes to the body + + buffer to read bytes from + where to start read + number of bytes to read + Number of bytes actually read (same as length unless we got all body bytes). + If body is not writable + bytes is null. + offset is out of range. + + + + Clear everything in the request + + + + + Decode body into a form. + + A list with form decoders. + If body contents is not valid for the chosen decoder. + If body is still being transferred. + + + + Sets the cookies. + + The cookies. + + + + Create a response object. + + Context for the connected client. + A new . + + + + Gets kind of types accepted by the client. + + + + + Gets or sets body stream. + + + + + Gets whether the body is complete. + + + + + Gets or sets kind of connection used for the session. + + + + + Gets or sets number of bytes in the body. + + + + + Gets cookies that was sent with the request. + + + + + Gets form parameters. + + + + + Gets headers sent by the client. + + + + + Gets or sets version of HTTP protocol that's used. + + + Probably or . + + + + + + Gets whether the request was made by Ajax (Asynchronous JavaScript) + + + + + Gets or sets requested method. + + + Will always be in upper case. + + + + + + Gets parameter from or . + + + + + Gets variables sent in the query string + + + + + Gets or sets requested URI. + + + + + Gets URI absolute path divided into parts. + + + // URI is: http://gauffin.com/code/tiny/ + Console.WriteLine(request.UriParts[0]); // result: code + Console.WriteLine(request.UriParts[1]); // result: tiny + + + If you're using controllers than the first part is controller name, + the second part is method name and the third part is Id property. + + + + + + Gets or sets path and query. + + + + Are only used during request parsing. Cannot be set after "Host" header have been + added. + + + + + PrototypeJS implementation of the javascript functions. + @@ -4427,99 +1918,7 @@ - - - - - - - - - - Represents a field in a multipart form - - - - The request could not be understood by the server due to malformed syntax. - The client SHOULD NOT repeat the request without modifications. - - Text taken from: http://www.submissionchamber.com/help-guides/error-codes.php - - - - - Create a new bad request exception. - - reason to why the request was bad. - - - - Create a new bad request exception. - - reason to why the request was bad. - inner exception - - - - Container class for posted files - - - - - Creates a container for a posted file - - The identifier of the post field - The file path - The content type of the file - The name of the file uploaded - If any parameter is null or empty - - - - Creates a container for a posted file - - If any parameter is null or empty - - - Destructor disposing the file - - - - Deletes the temporary file - - True if manual dispose - - - - Disposing interface, cleans up managed resources (the temporary file) and suppresses finalization - - - - - The name/id of the file - - - - - The full file path - - - - - The name of the uploaded file - - - - - The type of file - - - - - Will contain helper functions for javascript. - - - + Requests a url through ajax @@ -4537,12 +1936,21 @@ - + + + Determins if a list of strings contains a specific value + + options to check in + value to find + true if value was found + case insensitive + + Ajax requests that updates an element with the fetched content - url to fetch. Url is NOT enclosed in quotes by the implementation. You need to do that yourself. + URL to fetch. URL is NOT enclosed in quotes by the implementation. You need to do that yourself. element to update options in format "key, value, key, value". All keys should end with colon. A link tag. @@ -4552,20 +1960,2596 @@ - + - Opens contents in a dialog window. + A link that pop ups a Dialog (overlay div) - url to contents of dialog + URL to contents of dialog link title - name, value, name, value, all parameter names should end with colon. + name, value, name, value + + A "a"-tag that popups a dialog when clicked + + Requires Control.Modal found here: http://livepipe.net/projects/control_modal/ + And the following JavaScript (load it in application.js): + + Event.observe(window, 'load', + function() { + document.getElementsByClassName('modal').each(function(link){ new Control.Modal(link); }); + } + ); + + + + WebHelper.DialogLink("/user/show/1", "show user", "onmouseover", "alert('booh!');"); + - + + + create a modal dialog (usually using DIVs) + + url to fetch + dialog title + javascript/html attributes. javascript options ends with colon ':'. + + + Close a javascript dialog window/div. javascript for closing a dialog. - + + + + + javascript action that should be added to the "onsubmit" event in the form tag. + + remember to encapsulate strings in '' + + All javascript option names should end with colon. + + + JSHelper.AjaxRequest("/user/show/1", "onsuccess:", "$('userInfo').update(result);"); + + + + + + Helpers making it easier to work with forms. + + + + + + Used to let the website use different JavaScript libraries. + Default is + + + + + Create a <form> tag. + + name of form + action to invoke on submit + form should be posted as Ajax + HTML code + + + // without options + WebHelper.FormStart("frmLogin", "/user/login", Request.IsAjax); + + // with options + WebHelper.FormStart("frmLogin", "/user/login", Request.IsAjax, "style", "display:inline", "class", "greenForm"); + + + HTML attributes or JavaScript options. + Method will ALWAYS be POST. + options must consist of name, value, name, value + + + + Creates a select list with the values in a collection. + + Name of the SELECT-tag + collection used to generate options. + delegate used to return id and title from objects. + value that should be marked as selected. + First row should contain an empty value. + string containing a SELECT-tag. + + + + + Creates a select list with the values in a collection. + + Name of the SELECT-tag + Id of the SELECT-tag + collection used to generate options. + delegate used to return id and title from objects. + value that should be marked as selected. + First row should contain an empty value. + string containing a SELECT-tag. + + + + // Class that is going to be used in a SELECT-tag. + public class User + { + private readonly string _realName; + private readonly int _id; + public User(int id, string realName) + { + _id = id; + _realName = realName; + } + public string RealName + { + get { return _realName; } + } + + public int Id + { + get { return _id; } + } + } + + // Using an inline delegate to generate the select list + public void UserInlineDelegate() + { + List<User> items = new List<User>(); + items.Add(new User(1, "adam")); + items.Add(new User(2, "bertial")); + items.Add(new User(3, "david")); + string htmlSelect = Select("users", "users", items, delegate(object o, out object id, out object value) + { + User user = (User)o; + id = user.Id; + value = user.RealName; + }, 2, true); + } + + // Using an method as delegate to generate the select list. + public void UseExternalDelegate() + { + List<User> items = new List<User>(); + items.Add(new User(1, "adam")); + items.Add(new User(2, "bertial")); + items.Add(new User(3, "david")); + string htmlSelect = Select("users", "users", items, UserOptions, 1, true); + } + + // delegate returning id and title + public static void UserOptions(object o, out object id, out object title) + { + User user = (User)o; + id = user.Id; + value = user.RealName; + } + + + name, id, collection or getIdTitle is null. + + + + Creates a select list with the values in a collection. + + Name of the SELECT-tag + Id of the SELECT-tag + collection used to generate options. + delegate used to return id and title from objects. + value that should be marked as selected. + First row should contain an empty value. + name, value collection of extra HTML attributes. + string containing a SELECT-tag. + + name, id, collection or getIdTitle is null. + Invalid HTML attribute list. + + + + Generate a list of HTML options + + collection used to generate options. + delegate used to return id and title from objects. + value that should be marked as selected. + First row should contain an empty value. + + collection or getIdTitle is null. + + + sb is null. + + + + Creates a check box. + + element name + element value + determines if the check box is selected or not. This is done differently depending on the + type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if + the box is checked or not. + a list with additional attributes (name, value, name, value). + a generated radio button + + + + Creates a check box. + + element name + element id + element value + determines if the check box is selected or not. This is done differently depending on the + type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if + the box is checked or not. + a list with additional attributes (name, value, name, value). + a generated radio button + + value in your business object. (check box will be selected if it matches the element value) + + + + + Creates a check box. + + element name + element id + determines if the check box is selected or not. This is done differently depending on the + type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if + the box is checked or not. + a list with additional attributes (name, value, name, value). + a generated radio button + will set value to "1". + + + + Creates a RadioButton. + + element name + element value + determines if the radio button is selected or not. This is done differently depending on the + type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if + the box is checked or not. + a list with additional attributes (name, value, name, value). + a generated radio button + + + + Creates a RadioButton. + + element name + element id + element value + determines if the radio button is selected or not. This is done differently depending on the + type of variable. A boolean simply triggers checked or not, all other types are compared with "value" to determine if + the box is checked or not. + a list with additional attributes (name, value, name, value). + a generated radio button + + + + form close tag + + + + + + We dont want to let the server to die due to exceptions thrown in worker threads. + therefore we use this delegate to give you a change to handle uncaught exceptions. + + Class that the exception was thrown in. + Exception + + Server will throw a InternalServerException in release version if you dont + handle this delegate. + + + + + Implements HTTP Digest authentication. It's more secure than Basic auth since password is + encrypted with a "key" from the server. + + + Keep in mind that the password is encrypted with MD5. Use a combination of SSL and digest auth to be secure. + + + + + Authentication modules are used to implement different + kind of HTTP authentication. + + + + + Tag used for authentication. + + + + + Initializes a new instance of the class. + + Delegate used to provide information used during authentication. + Delegate used to determine if authentication is required (may be null). + + + + Initializes a new instance of the class. + + Delegate used to provide information used during authentication. + + + + Create a response that can be sent in the WWW-Authenticate header. + + Realm that the user should authenticate in + Array with optional options. + A correct authentication request. + If realm is empty or null. + + + + An authentication response have been received from the web browser. + Check if it's correct + + Contents from the Authorization header + Realm that should be authenticated + GET/POST/PUT/DELETE etc. + options to specific implementations + Authentication object that is stored for the request. A user class or something like that. + if is invalid + If any of the parameters is empty or null. + + + + Used to invoke the authentication delegate that is used to lookup the user name/realm. + + Realm (domain) that user want to authenticate in + User name + Password used for validation. Some implementations got password in clear text, they are then sent to client. + object that will be stored in the request to help you identify the user if authentication was successful. + true if authentication was successful + + + + Determines if authentication is required. + + HTTP request from browser + true if user should be authenticated. + throw from your delegate if no more attempts are allowed. + If no more attempts are allowed + + + + name used in HTTP request. + + + + + Initializes a new instance of the class. + + Delegate used to provide information used during authentication. + Delegate used to determine if authentication is required (may be null). + + + + Initializes a new instance of the class. + + Delegate used to provide information used during authentication. + + + + Used by test classes to be able to use hardcoded values + + + + + An authentication response have been received from the web browser. + Check if it's correct + + Contents from the Authorization header + Realm that should be authenticated + GET/POST/PUT/DELETE etc. + First option: true if username/password is correct but not cnonce + + Authentication object that is stored for the request. A user class or something like that. + + if authenticationHeader is invalid + If any of the paramters is empty or null. + + + + Encrypts parameters into a Digest string + + Realm that the user want to log into. + User logging in + Users password. + HTTP method. + Uri/domain that generated the login prompt. + Quality of Protection. + "Number used ONCE" + Hexadecimal request counter. + "Client Number used ONCE" + Digest encrypted string + + + + + + Md5 hex encoded "userName:realm:password", without the quotes. + Md5 hex encoded "method:uri", without the quotes + Quality of Protection + "Number used ONCE" + Hexadecimal request counter. + Client number used once + + + + + Create a response that can be sent in the WWW-Authenticate header. + + Realm that the user should authenticate in + First options specifies if true if username/password is correct but not cnonce. + A correct auth request. + If realm is empty or null. + + + + Decodes authorization header value + + header value + Encoding that the buffer is in + All headers and their values if successful; otherwise null + + NameValueCollection header = DigestAuthentication.Decode("response=\"6629fae49393a05397450978507c4ef1\",\r\nc=00001", Encoding.ASCII); + + Can handle lots of whitespaces and new lines without failing. + + + + Gets the current nonce. + + + + + + Gets the Md5 hash bin hex2. + + To be hashed. + + + + + determines if the nonce is valid or has expired. + + nonce value (check wikipedia for info) + true if the nonce has not expired. + + + + name used in http request. + + + + + Gets or sets whether the token supplied in is a + HA1 generated string. + + + + + Generic helper functions for HTTP + + + + + Version string for HTTP v1.0 + + + + + Version string for HTTP v1.1 + + + + + An empty URI + + + + + Parses a query string. + + Query string (URI encoded) + A object if successful; otherwise + queryString is null. + If string cannot be parsed. + + + + Delegate used to let authentication modules authenticate the user name and password. + + Realm that the user want to authenticate in + User name specified by client + Can either be user password or implementation specific token. + object that will be stored in a session variable called if authentication was successful. + throw forbidden exception if too many attempts have been made. + + + Use to specify that the token is a HA1 token. (MD5 generated + string from realm, user name and password); Md5String(userName + ":" + realm + ":" + password); + + + + + + Let's you decide on a system level if authentication is required. + + HTTP request from client + true if user should be authenticated. + throw if no more attempts are allowed. + If no more attempts are allowed + + + + Arguments used when more body bytes have come. + + + + + Initializes a new instance of the class. + + buffer that contains the received bytes. + offset in buffer where to start processing. + number of bytes from that should be parsed. + + + + Initializes a new instance of the class. + + + + + Gets or sets buffer that contains the received bytes. + + + + + Gets or sets number of bytes from that should be parsed. + + + + + Gets or sets offset in buffer where to start processing. + + + + + Contains all HTTP Methods (according to the HTTP 1.1 specification) + + See: http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html + + + + + + The DELETE method requests that the origin server delete the resource identified by the Request-URI. + + + + This method MAY be overridden by human intervention (or other means) on the origin server. + The client cannot be guaranteed that the operation has been carried out, even if the status code + returned from the origin server indicates that the action has been completed successfully. + + + However, the server SHOULD NOT indicate success unless, at the time the response is given, + it intends to delete the resource or move it to an inaccessible location. + + + A successful response SHOULD be 200 (OK) if the response includes an entity describing the status, + 202 (Accepted) if the action has not yet been enacted, + or 204 (No Content) if the action has been enacted but the response does not include an entity. + + + If the request passes through a cache and the Request-URI identifies one or more currently cached entities, + those entries SHOULD be treated as stale. Responses to this method are not cacheable. + + + + + + The GET method means retrieve whatever information (in the form of an entity) is identified by the Request-URI. + + + + If the Request-URI refers to a data-producing process, it is the produced data which shall be returned as the + entity in the response and not the source text of the process, unless that text happens to be the output of the process. + + + The semantics of the GET method change to a "conditional GET" if the request message includes an + If-Modified-Since, If-Unmodified-Since, If-Match, If-None-Match, or If-Range header field. + A conditional GET method requests that the entity be transferred only under the circumstances described + by the conditional header field(s). The conditional GET method is intended to reduce unnecessary network + usage by allowing cached entities to be refreshed without requiring multiple requests or transferring + data already held by the client. + + + + + + The HEAD method is identical to GET except that the server MUST NOT return a message-body in the response. + + + The meta information contained in the HTTP headers in response to a HEAD request SHOULD be identical to the + information sent in response to a GET request. This method can be used for obtaining meta information about + the entity implied by the request without transferring the entity-body itself. + + This method is often used for testing hypertext links for validity, accessibility, and recent modification. + + + + + The OPTIONS method represents a request for information about the communication options available on the request/response chain identified by the Request-URI. + + + This method allows the client to determine the options and/or requirements associated with a resource, or the capabilities of a server, without implying a resource action or initiating a resource retrieval. + + + + + The POST method is used to request that the origin server accept the entity enclosed + in the request as a new subordinate of the resource identified by the Request-URI in the Request-Line. + + + POST is designed to allow a uniform method to cover the following functions: + + + Annotation of existing resources; + + Posting a message to a bulletin board, newsgroup, mailing list, or similar group of articles; + + Providing a block of data, such as the result of submitting a form, to a data-handling process; + + Extending a database through an append operation. + + + + If a resource has been created on the origin server, the response SHOULD be 201 (Created) and + contain an entity which describes the status of the request and refers to the new resource, and a + Location header (see section 14.30). + + + The action performed by the POST method might not result in a resource that can be identified by a URI. + In this case, either 200 (OK) or 204 (No Content) is the appropriate response status, depending on + whether or not the response includes an entity that describes the result. + + Responses to this method are not cacheable, unless the response includes appropriate Cache-Control + or Expires header fields. However, the 303 (See Other) response can be used to direct the user agent + to retrieve a cacheable resource. + + + + + + The PUT method requests that the enclosed entity be stored under the supplied Request-URI. + + + + + If the Request-URI refers to an already existing resource, the enclosed entity SHOULD be considered as a + modified version of the one residing on the origin server. + + If the Request-URI does not point to an existing resource, and that URI is capable of being defined as a new + resource by the requesting user agent, the origin server can create the resource with that URI. + + If a new resource is created, the origin server MUST inform the user agent via the 201 (Created) response. + + If an existing resource is modified, either the 200 (OK) or 204 (No Content) response codes SHOULD be sent to + indicate successful completion of the request. + + If the resource could not be created or modified with the Request-URI, an appropriate error response SHOULD be + given that reflects the nature of the problem. + + + + The recipient of the entity MUST NOT ignore any Content-* (e.g. Content-Range) headers that it does not + understand or implement and MUST return a 501 (Not Implemented) response in such cases. + + + + + + The TRACE method is used to invoke a remote, application-layer loop- back of the request message. + + + + + Contains all HTTP Methods (according to the HTTP 1.1 specification) + + See: http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html + + + + + + The DELETE method requests that the origin server delete the resource identified by the Request-URI. + + + + This method MAY be overridden by human intervention (or other means) on the origin server. + The client cannot be guaranteed that the operation has been carried out, even if the status code + returned from the origin server indicates that the action has been completed successfully. + + + However, the server SHOULD NOT indicate success unless, at the time the response is given, + it intends to delete the resource or move it to an inaccessible location. + + + A successful response SHOULD be 200 (OK) if the response includes an entity describing the status, + 202 (Accepted) if the action has not yet been enacted, + or 204 (No Content) if the action has been enacted but the response does not include an entity. + + + If the request passes through a cache and the Request-URI identifies one or more currently cached entities, + those entries SHOULD be treated as stale. Responses to this method are not cacheable. + + + + + + The GET method means retrieve whatever information (in the form of an entity) is identified by the Request-URI. + + + + If the Request-URI refers to a data-producing process, it is the produced data which shall be returned as the + entity in the response and not the source text of the process, unless that text happens to be the output of the process. + + + The semantics of the GET method change to a "conditional GET" if the request message includes an + If-Modified-Since, If-Unmodified-Since, If-Match, If-None-Match, or If-Range header field. + A conditional GET method requests that the entity be transferred only under the circumstances described + by the conditional header field(s). The conditional GET method is intended to reduce unnecessary network + usage by allowing cached entities to be refreshed without requiring multiple requests or transferring + data already held by the client. + + + + + + The HEAD method is identical to GET except that the server MUST NOT return a message-body in the response. + + + The meta information contained in the HTTP headers in response to a HEAD request SHOULD be identical to the + information sent in response to a GET request. This method can be used for obtaining meta information about + the entity implied by the request without transferring the entity-body itself. + + This method is often used for testing hypertext links for validity, accessibility, and recent modification. + + + + + The OPTIONS method represents a request for information about the communication options available on the request/response chain identified by the Request-URI. + + + This method allows the client to determine the options and/or requirements associated with a resource, or the capabilities of a server, without implying a resource action or initiating a resource retrieval. + + + + + The POST method is used to request that the origin server accept the entity enclosed + in the request as a new subordinate of the resource identified by the Request-URI in the Request-Line. + + + POST is designed to allow a uniform method to cover the following functions: + + + Annotation of existing resources; + + Posting a message to a bulletin board, newsgroup, mailing list, or similar group of articles; + + Providing a block of data, such as the result of submitting a form, to a data-handling process; + + Extending a database through an append operation. + + + + If a resource has been created on the origin server, the response SHOULD be 201 (Created) and + contain an entity which describes the status of the request and refers to the new resource, and a + Location header (see section 14.30). + + + The action performed by the POST method might not result in a resource that can be identified by a URI. + In this case, either 200 (OK) or 204 (No Content) is the appropriate response status, depending on + whether or not the response includes an entity that describes the result. + + Responses to this method are not cacheable, unless the response includes appropriate Cache-Control + or Expires header fields. However, the 303 (See Other) response can be used to direct the user agent + to retrieve a cacheable resource. + + + + + + The PUT method requests that the enclosed entity be stored under the supplied Request-URI. + + + + + If the Request-URI refers to an already existing resource, the enclosed entity SHOULD be considered as a + modified version of the one residing on the origin server. + + If the Request-URI does not point to an existing resource, and that URI is capable of being defined as a new + resource by the requesting user agent, the origin server can create the resource with that URI. + + If a new resource is created, the origin server MUST inform the user agent via the 201 (Created) response. + + If an existing resource is modified, either the 200 (OK) or 204 (No Content) response codes SHOULD be sent to + indicate successful completion of the request. + + If the resource could not be created or modified with the Request-URI, an appropriate error response SHOULD be + given that reflects the nature of the problem. + + + + The recipient of the entity MUST NOT ignore any Content-* (e.g. Content-Range) headers that it does not + understand or implement and MUST return a 501 (Not Implemented) response in such cases. + + + + + + The TRACE method is used to invoke a remote, application-layer loop- back of the request message. + + + + + Used to create and reuse contexts. + + + + + Used to create es. + + + + + Creates a that handles a connected client. + + Client socket (accepted by the ). + A creates . + + + + Create a secure . + + Client socket (accepted by the ). + HTTPS certificate to use. + Kind of HTTPS protocol. Usually TLS or SSL. + A created . + + + + A request have been received from one of the contexts. + + + + + Initializes a new instance of the class. + + The writer. + Amount of bytes to read from the incoming socket stream. + Used to create a request parser. + + + + Create a new context. + + true if socket is running HTTPS. + Client that connected + Network/SSL stream. + A context. + + + + Create a new context. + + true if HTTPS is used. + Remote client + Network stream, uses . + A new context (always). + + + + Create a secure . + + Client socket (accepted by the ). + HTTPS certificate to use. + Kind of HTTPS protocol. Usually TLS or SSL. + + A created . + + + + + Creates a that handles a connected client. + + Client socket (accepted by the ). + + A creates . + + + + + True if detailed trace logs should be written. + + + + + A request have been received from one of the contexts. + + + + + Custom network stream to mark sockets as reusable when disposing the stream. + + + + + Creates a new instance of the class for the specified . + + + The that the will use to send and receive data. + + + The parameter is null. + + + The parameter is not connected. + -or- + The property of the parameter is not . + -or- + The parameter is in a nonblocking state. + + + + + Initializes a new instance of the class for the specified with the specified ownership. + + + The that the will use to send and receive data. + + + Set to true to indicate that the will take ownership of the ; otherwise, false. + + + The parameter is null. + + + The parameter is not connected. + -or- + the value of the property of the parameter is not . + -or- + the parameter is in a nonblocking state. + + + + + Creates a new instance of the class for the specified with the specified access rights. + + + The that the will use to send and receive data. + + + A bitwise combination of the values that specify the type of access given to the over the provided . + + + The parameter is null. + + + The parameter is not connected. + -or- + the property of the parameter is not . + -or- + the parameter is in a nonblocking state. + + + + + Creates a new instance of the class for the specified with the specified access rights and the specified ownership. + + + The that the will use to send and receive data. + + + A bitwise combination of the values that specifies the type of access given to the over the provided . + + + Set to true to indicate that the will take ownership of the ; otherwise, false. + + + The parameter is null. + + + The parameter is not connected. + -or- + The property of the parameter is not . + -or- + The parameter is in a nonblocking state. + + + + + Closes the current stream and releases any resources (such as sockets and file handles) associated with the current stream. + + + + + Releases the unmanaged resources used by the and optionally releases the managed resources. + + true to release both managed and unmanaged resources; false to release only unmanaged resources. + + + + Invoked when a client have been accepted by the + + + Can be used to revoke incoming connections + + + + + Initializes a new instance of the class. + + The socket. + + + + Client may not be handled. + + + + + Accepted socket. + + + + + Client should be revoked. + + + + + A session stored in memory. + + + + + Interface for sessions + + + + + Remove everything from the session + + + + + Remove everything from the session + + True if the session is cleared due to expiration + + + + Session id + + + + + Should + + Name of the session variable + null if it's not set + If the object cant be serialized. + + + + When the session was last accessed. + This property is touched by the http server each time the + session is requested. + + + + + Number of session variables. + + + + + Event triggered upon clearing the session + + + + + + + A unique id used by the sessions store to identify the session + + + + Id + + + + + + Remove everything from the session + + + + + Clears the specified expire. + + True if the session is cleared due to expiration + + + + Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources. + + 2 + + + + Session id + + + + + Should + + Name of the session variable + null if it's not set + + + + when the session was last accessed. + + + Used to determine when the session should be removed. + + + + + Number of values in the session + + + + + Flag to indicate that the session have been changed + and should be saved into the session store. + + + + + Event triggered upon clearing the session + + + + + A reverse proxy are used to act as a bridge between local (protected/hidden) websites + and public clients. + + A typical usage is to allow web servers on non standard ports to still be available + to the public clients, or allow web servers on private ips to be available. + + + + + + + Base url requested from browser + Base url on private web server + + // this will return contents from http://192.168.1.128/view/jonas when client requests http://www.gauffin.com/user/view/jonas + _server.Add(new ReverseProxyModule("http://www.gauffin.com/user/", "http://192.168.1.128/"); + + + + + Method that determines if an url should be handled or not by the module + + Url requested by the client. + true if module should handle the url. + + + + Method that process the url + + Information sent by the browser about the request + Information that is being sent back to the client. + Session used to + + + + Can handle application/x-www-form-urlencoded + + + + + + Stream containing the content + Content type (with any additional info like boundry). Content type is always supplied in lower case + Stream encoding + + A HTTP form, or null if content could not be parsed. + + If contents in the stream is not valid input data. + + + + Checks if the decoder can handle the mime type + + Content type (with any additional info like boundry). Content type is always supplied in lower case. + True if the decoder can parse the specified content type + + + + This provider is used to let us implement any type of form decoding we want without + having to rewrite anything else in the server. + + + + + + + Should contain boundary and type, as in: multipart/form-data; boundary=---------------------------230051238959 + Stream containing form data. + Encoding used when decoding the stream + if no parser was found. + If stream is null or not readable. + If stream contents cannot be decoded properly. + + + + Add a decoder. + + + + + + + Number of added decoders. + + + + + Use with care. + + + + + Decoder used for unknown content types. + + + + + The server encountered an unexpected condition which prevented it from fulfilling the request. + + + + + Initializes a new instance of the class. + + + + + Initializes a new instance of the class. + + error message. + + + + Initializes a new instance of the class. + + error message. + inner exception. + + + + Response that is sent back to the web browser / client. + + A response can be sent if different ways. The easiest one is + to just fill the Body stream with content, everything else + will then be taken care of by the framework. The default content-type + is text/html, you should change it if you send anything else. + + The second and slighty more complex way is to send the response + as parts. Start with sending the header using the SendHeaders method and + then you can send the body using SendBody method, but do not forget + to set ContentType and ContentLength before doing so. + + + public void MyHandler(IHttpRequest request, IHttpResponse response) + { + + } + + + + + Add another header to the document. + + Name of the header, case sensitive, use lower cases. + Header values can span over multiple lines as long as each line starts with a white space. New line chars should be \r\n + If headers already been sent. + If value conditions have not been met. + Adding any header will override the default ones and those specified by properties. + + + + Send headers and body to the browser. + + If content have already been sent. + + + + Make sure that you have specified ContentLength and sent the headers first. + + + If headers have not been sent. + + offest of first byte to send + number of bytes to send. + + + This method can be used if you want to send body contents without caching them first. This + is recommended for larger files to keep the memory usage low. + + + + Make sure that you have specified ContentLength and sent the headers first. + + + If headers have not been sent. + + + + This method can be used if you want to send body contents without caching them first. This + is recommended for larger files to keep the memory usage low. + + + + Send headers to the client. + + If headers already been sent. + + + + + + + Redirect client to somewhere else using the 302 status code. + + Destination of the redirect + If headers already been sent. + You can not do anything more with the request when a redirect have been done. This should be your last + action. + + + + redirect to somewhere + + where the redirect should go + + No body are allowed when doing redirects. + + + + + The body stream is used to cache the body contents + before sending everything to the client. It's the simplest + way to serve documents. + + + + + Defines the version of the HTTP Response for applications where it's required + for this to be forced. + + + + + The chunked encoding modifies the body of a message in order to + transfer it as a series of chunks, each with its own size indicator, + followed by an OPTIONAL trailer containing entity-header fields. This + allows dynamically produced content to be transferred along with the + information necessary for the recipient to verify that it has + received the full message. + + + + + Kind of connection + + + + + Encoding to use when sending stuff to the client. + + Default is UTF8 + + + + Number of seconds to keep connection alive + + Only used if Connection property is set to ConnectionType.KeepAlive + + + + Status code that is sent to the client. + + Default is HttpStatusCode.Ok + + + + Information about why a specific status code was used. + + + + + Size of the body. MUST be specified before sending the header, + unless property Chunked is set to true. + + + + + Kind of content in the body + + Default is text/html + + + + Headers have been sent to the client- + + You can not send any additional headers if they have already been sent. + + + + The whole response have been sent. + + + + + Cookies that should be created/changed. + + + + + Type of HTTP connection + + + + + Connection is closed after each request-response + + + + + Connection is kept alive for X seconds (unless another request have been made) + + + + + The website module let's you handle multiple websites in the same server. + It uses the "Host" header to check which site you want. + + It's recommended that you do not + add any other modules to HttpServer if you are using the website module. Instead, + add all wanted modules to each website. + + + + + + domain name that should be handled. + + + + + Method that process the url + + Information sent by the browser about the request + Information that is being sent back to the client. + Session used to + + + + Name of site. + + + + + Used to inform http server that + + + + + Eventarguments used when an exception is thrown by a module + + the exception + + + + Exception thrown in a module + + + + + represents a HTTP input item. Each item can have multiple sub items, a sub item + is made in a HTML form by using square brackets + + + // becomes: + Console.WriteLine("Value: {0}", form["user"]["FirstName"].Value); + + + All names in a form SHOULD be in lowercase. + + + + Representation of a non-initialized . + + + + Initializes an input item setting its name/identifier and value + + Parameter name/id + Parameter value + + + Creates a deep copy of the item specified + The item to copy + The function makes a deep copy of quite a lot which can be slow + + + + Add another value to this item + + Value to add. + Cannot add stuff to . + + + + checks if a sub-item exists (and has a value). + + name in lower case + true if the sub-item exists and has a value; otherwise false. + + + Returns a formatted representation of the instance with the values of all contained parameters + + + + Outputs the string in a formatted manner + + A prefix to append, used internally + produce a query string + + + + Add a sub item. + + Can contain array formatting, the item is then parsed and added in multiple levels + Value to add. + Argument is null. + Cannot add stuff to . + + + + Returns an enumerator that iterates through the collection. + + + + A that can be used to iterate through the collection. + + 1 + + + + Returns an enumerator that iterates through a collection. + + + + An object that can be used to iterate through the collection. + + 2 + + + + Outputs the string in a formatted manner + + A prefix to append, used internally + + + + + Number of values + + + + + Get a sub item + + name in lower case. + if no item was found. + + + + Name of item (in lower case). + + + + + Returns the first value, or null if no value exist. + + + + + Returns the last value, or null if no value exist. + + + + + Returns the list with values. + + + + + + + name in lower case + + + + Class to handle loading of resource files + + + + Initializes a new instance of the class. + + + + + Initializes a new instance of the class. + + logger. + + + + Loads resources from a namespace in the given assembly to an URI + + The URI to map the resources to + The assembly in which the resources reside + The namespace from which to load the resources + + + resourceLoader.LoadResources("/user/", typeof(User).Assembly, "MyLib.Models.User.Views"); + + Will make the resource MyLib.Models.User.Views.list.Haml accessible via /user/list.haml or /user/list/ + + The amount of loaded files, giving you the possibility of making sure the resources needed gets loaded + If a resource has already been mapped to an uri + + + + Retrieves a stream for the specified resource path if loaded otherwise null + + Path to the resource to retrieve a stream for + A stream or null if the resource couldn't be found + + + + Fetch all files from the resource that matches the specified arguments. + + The path to the resource to extract + + a list of files if found; or an empty array if no files are found. + + Search path must end with an asterisk for finding arbitrary files + + + + Fetch all files from the resource that matches the specified arguments. + + Where the file should reside. + Files to check + + a list of files if found; or an empty array if no files are found. + + + + + Returns whether or not the loader has an instance of the file requested + + The name of the template/file + True if the loader can provide the file + + + + Used when the request line have been successfully parsed. + + + + + Initializes a new instance of the class. + + The HTTP method. + The URI path. + The HTTP version. + + + + Initializes a new instance of the class. + + + + + Gets or sets http method. + + + Should be one of the methods declared in . + + + + + Gets or sets the version of the HTTP protocol that the client want to use. + + + + + Gets or sets requested URI path. + + + + + Class that receives Requests from a . + + + + + Client have been disconnected. + + Client that was disconnected. + Reason + + + + + Invoked when a client context have received a new HTTP request + + Client that received the request. + Request that was received. + + + + Container for posted form data + + + Instance to help mark a non-initialized form + + + Initializes a form container with the specified name + + + + Makes a deep copy of the input + + The input to copy + + + + Adds a file to the collection of posted files + + The file to add + If the file is already added + If file is null + If the instance is HttpForm.EmptyForm which cannot be modified + + + + Checks if the form contains a specified file + + Field name of the file parameter + True if the file exists + If the instance is HttpForm.EmptyForm which cannot be modified + + + + Retrieves a file held by by the form + + The identifier of the file + The requested file or null if the file was not found + If name is null or empty + If the instance is HttpForm.EmptyForm which cannot be modified + + + Disposes all held HttpFile's and resets values + + + + Retrieves the number of files added to the + + 0 if no files are added + + + + Contains a connection to a browser/client. + + + Remember to after you have hooked the event. + + TODO: Maybe this class should be broken up into HttpClientChannel and HttpClientContext? + + + + Initializes a new instance of the class. + + true if the connection is secured (SSL/TLS) + client that connected. + Stream used for communication + Used to create a . + Size of buffer to use when reading data. Must be at least 4096 bytes. + If fails + Stream must be writable and readable. + + + + Process incoming body bytes. + + + Bytes + + + + + + + + + + + Start reading content. + + + Make sure to call base.Start() if you override this method. + + + + + Clean up context. + + + Make sure to call base.Cleanup() if you override the method. + + + + + Disconnect from client + + error to report in the event. + + + + Send a response. + + Either or + HTTP status code + reason for the status code. + HTML body contents, can be null or empty. + A content type to return the body as, i.e. 'text/html' or 'text/plain', defaults to 'text/html' if null or empty + If is invalid. + + + + Send a response. + + Either or + HTTP status code + reason for the status code. + + + + Send a response. + + + + + + send a whole buffer + + buffer to send + + + + + Send data using the stream + + Contains data to send + Start position in buffer + number of bytes to send + + + + + + This context have been cleaned, which means that it can be reused. + + + + + Context have been started (a new client have connected) + + + + + Overload to specify own type. + + + Must be specified before the context is being used. + + + + + Using SSL or other encryption method. + + + + + Using SSL or other encryption method. + + + + + Specify which logger to use. + + + + + Gets or sets the network stream. + + + + + Gets or sets IP address that the client connected from. + + + + + Gets or sets port that the client connected from. + + + + + The context have been disconnected. + + + Event can be used to clean up a context, or to reuse it. + + + + + A request have been received in the context. + + + + + Helpers to make XML handling easier + + + + + Serializes object to XML. + + object to serialize. + XML + + Removes name spaces and adds indentation + + + + + Create an object from a XML string + + Type of object + XML string + object + + + + + + + + + + + Represents a field in a multipart form + + + + Small design by contract implementation. + + + + + Check whether a parameter is empty. + + Parameter value + Parameter name, or error description. + value is empty. + + + + Checks whether a parameter is null. + + Parameter value + Parameter name, or error description. + value is null. + + + + Checks whether a parameter is null. + + + Parameter value + Parameter name, or error description. + value is null. + + + + Priority for log entries + + + + + + Very detailed logs to be able to follow the flow of the program. + + + + + Logs to help debug errors in the application + + + + + Information to be able to keep track of state changes etc. + + + + + Something did not go as we expected, but it's no problem. + + + + + Something that should not fail failed, but we can still keep + on going. + + + + + Something failed, and we cannot handle it properly. + + + + + Interface used to write to log files. + + + + + Write an entry to the log file. + + object that is writing to the log + importance of the log message + the message + + + + This class writes to the console. It colors the output depending on the logprio and includes a 3-level stacktrace (in debug mode) + + + + + + The actual instance of this class. + + + + + Logwriters the specified source. + + object that wrote the logentry. + Importance of the log message + The message. + + + + Get color for the specified logprio + + prio for the log entry + A for the prio + + + + Default log writer, writes everything to null (nowhere). + + + + + + The logging instance. + + + + + Writes everything to null + + object that wrote the log entry. + Importance of the log message + The message. + + + + Response that is sent back to the web browser / client. + + + + A response can be sent if different ways. The easiest one is + to just fill the Body stream with content, everything else + will then be taken care of by the framework. The default content-type + is text/html, you should change it if you send anything else. + + The second and slightly more complex way is to send the response + as parts. Start with sending the header using the SendHeaders method and + then you can send the body using SendBody method, but do not forget + to set and before doing so. + + + + + // Example using response body. + class MyModule : HttpModule + { + public override bool Process(IHttpRequest request, IHttpResponse response, IHttpSession session) + { + StreamWriter writer = new StreamWriter(response.Body); + writer.WriteLine("Hello dear World!"); + writer.Flush(); + + // return true to tell webserver that we've handled the url + return true; + } + } + + + todo: add two examples, using SendHeaders/SendBody and just the Body stream. + + + + Initializes a new instance of the class. + + Client that send the . + Contains information of what the client want to receive. + cannot be empty. + + + + Initializes a new instance of the class. + + Client that send the . + Version of HTTP protocol that the client uses. + Type of HTTP connection used. + + + + Add another header to the document. + + Name of the header, case sensitive, use lower cases. + Header values can span over multiple lines as long as each line starts with a white space. New line chars should be \r\n + If headers already been sent. + If value conditions have not been met. + Adding any header will override the default ones and those specified by properties. + + + + Send headers and body to the browser. + + If content have already been sent. + + + + Make sure that you have specified and sent the headers first. + + + If headers have not been sent. + + offset of first byte to send + number of bytes to send. + + + This method can be used if you want to send body contents without caching them first. This + is recommended for larger files to keep the memory usage low. + + + + Make sure that you have specified and sent the headers first. + + + If headers have not been sent. + + + + This method can be used if you want to send body contents without caching them first. This + is recommended for larger files to keep the memory usage low. + + + + Send headers to the client. + + If headers already been sent. + + + + + + + Redirect client to somewhere else using the 302 status code. + + Destination of the redirect + If headers already been sent. + You can not do anything more with the request when a redirect have been done. This should be your last + action. + + + + redirect to somewhere + + where the redirect should go + + No body are allowed when doing redirects. + + + + + The body stream is used to cache the body contents + before sending everything to the client. It's the simplest + way to serve documents. + + + + + The chunked encoding modifies the body of a message in order to + transfer it as a series of chunks, each with its own size indicator, + followed by an OPTIONAL trailer containing entity-header fields. This + allows dynamically produced content to be transferred along with the + information necessary for the recipient to verify that it has + received the full message. + + + + + Defines the version of the HTTP Response for applications where it's required + for this to be forced. + + + + + Kind of connection + + + + + Encoding to use when sending stuff to the client. + + Default is UTF8 + + + + Number of seconds to keep connection alive + + Only used if Connection property is set to . + + + + Status code that is sent to the client. + + Default is + + + + Information about why a specific status code was used. + + + + + Size of the body. MUST be specified before sending the header, + unless property Chunked is set to true. + + + + + Kind of content in the body + + Default type is "text/html" + + + + Headers have been sent to the client- + + You can not send any additional headers if they have already been sent. + + + + The whole response have been sent. + + + + + Cookies that should be created/changed. + + + + + The requested resource was not found in the web server. + + + + + Create a new exception + + message describing the error + inner exception + + + + Create a new exception + + message describing the error + + + + Session store using memory for each session. + + + + + A session store is used to store and load sessions on a media. + The default implementation () saves/retrieves sessions from memory. + + + + + Creates a new http session with a generated id. + + A object + + + + Creates a new http session with a specific id + + Id used to identify the new cookie.. + A object. + + Id should be generated by the store implementation if it's null or . + + + + + Load an existing session. + + Session id (usually retrieved from a client side cookie). + A session if found; otherwise null. + + + + Save an updated session to the store. + + Session id (usually retrieved from a client side cookie). + If Id property have not been specified. + + + + We use the flyweight pattern which reuses small objects + instead of creating new each time. + + Unused session that should be reused next time Create is called. + + + + Remove expired sessions + + + + + Remove a session + + id of the session. + + + + Load a session from the store + + + null if session is not found. + + + + Number of minutes before a session expires. + + Default time is 20 minutes. + + + + Initializes the class setting the expirationtimer to clean the session every minute + + + + + Delegate for the cleanup timer + + + + + Creates a new http session + + + + + + Creates a new http session with a specific id + + Id used to identify the new cookie.. + A object. + + Id should be generated by the store implementation if it's null or . + + + + + Load an existing session. + + + + + + + Save an updated session to the store. + + + + + + We use the flyweight pattern which reuses small objects + instead of creating new each time. + + EmptyLanguageNode (unused) session that should be reused next time Create is called. + + + + Remove expired sessions + + + + + Remove a session + + id of the session. + + + + Load a session from the store + + + null if session is not found. + + + + Number of minutes before a session expires. + Default is 20 minutes. + + + + + Arguments sent when a is cleared + + + + + Instantiates the arguments for the event + + True if the session is cleared due to expiration + + + + Returns true if the session is cleared due to expiration + + + + + Delegate for when a IHttpSession is cleared + + this is being cleared. + Arguments for the clearing + + + + Used to queue incoming requests. + + + + + Initializes a new instance of the class. + + Called when a request should be processed. + + + + Used to process queued requests. + + + + + Gets or sets maximum number of allowed simultaneous requests. + + + + + Gets or sets maximum number of requests queuing to be handled. + + + + + Specifies how many requests the HTTP server is currently processing. + + + + + Used two queue incoming requests to avoid + thread starvation. + + + + + Method used to process a queued request + + Context that the request was received from. + Request to process. + + + + Event arguments used when a new header have been parsed. + + + + + Initializes a new instance of the class. + + Name of header. + Header value. + + + + Initializes a new instance of the class. + + + + + Gets or sets header name. + + + + + Gets or sets header value. + @@ -4743,34 +4727,108 @@ Gets cookies that was sent with the request. - + - The website module let's you handle multiple websites in the same server. - It uses the "Host" header to check which site you want. + Add a component instance - It's recommended that you do not - add any other modules to HttpServer if you are using the website module. Instead, - add all wanted modules to each website. + Interface type + Instance to add - + - + Get a component. - domain name that should be handled. - + Interface type + Component if registered, otherwise null. + + Component will get created if needed. + - - - Method that process the url - - Information sent by the browser about the request - Information that is being sent back to the client. - Session used to + + If instance cannot be created. - + - Name of site. + Checks if the specified component interface have been added. + + true if found; otherwise false. + + + + Add a component. + + Type being requested. + Type being created. + Type have already been mapped. + + + + Class to make dynamic binding of redirects. Instead of having to specify a number of similar redirect rules + a regular expression can be used to identify redirect URLs and their targets. + + + [a-z0-9]+)", "/users/${target}?find=true", RegexOptions.IgnoreCase) + ]]> + + + + + Initializes a new instance of the class. + + Expression to match URL + Expression to generate URL + + [a-zA-Z0-9]+)", "/user/${first}")); + Result of ie. /employee1 will then be /user/employee1 + ]]> + + + + + Initializes a new instance of the class. + + Expression to match URL + Expression to generate URL + Regular expression options to use, can be null + + [a-zA-Z0-9]+)", "/user/{first}", RegexOptions.IgnoreCase)); + Result of ie. /employee1 will then be /user/employee1 + ]]> + + + + + Initializes a new instance of the class. + + Expression to match URL + Expression to generate URL + Regular expression options to apply + true if request should be redirected, false if the request URI should be replaced. + + [a-zA-Z0-9]+)", "/user/${first}", RegexOptions.None)); + Result of ie. /employee1 will then be /user/employee1 + ]]> + + Argument is null. + + + + + Process the incoming request. + + incoming HTTP request + outgoing HTTP response + true if response should be sent to the browser directly (no other rules or modules will be processed). + + returning true means that no modules will get the request. Returning true is typically being done + for redirects. + + If request or response is null @@ -4812,30 +4870,6 @@ Retrieves the full path name to the resource file - - - Creates request parsers when needed. - - - - - Creates request parsers when needed. - - - - - Create a new request parser. - - Used when logging should be enabled. - A new request parser. - - - - Create a new request parser. - - Used when logging should be enabled. - A new request parser. - The "basic" authentication scheme is based on the model that the @@ -4921,319 +4955,12 @@ Adding bytes to body - + - This decoder converts XML documents to form items. - Each element becomes a subitem in the form, and each attribute becomes an item. - - - // xml: somethingdata - // result: - // form["hello"].Value = "something" - // form["hello"]["id"].Value = 1 - // form["hello"]["world]["id"].Value = 1 - // form["hello"]["world"].Value = "data" - - - The original xml document is stored in form["__xml__"].Value. - - - - - - - Stream containing the content - Content type (with any additional info like boundry). Content type is always supplied in lower case - Stream encoding - Note: contentType and encoding are not used? - A http form, or null if content could not be parsed. - - - - - Recursive function that will go through an xml element and store it's content - to the form item. - - (parent) Item in form that content should be added to. - Node that should be parsed. - - - - Checks if the decoder can handle the mime type - - Content type (with any additional info like boundry). Content type is always supplied in lower case. - True if the decoder can parse the specified content type - - - - Cookies that should be set. + Will contain helper functions for javascript. - - - Adds a cookie in the collection. - - cookie to add - cookie is null - - - - Copy a request cookie - - - When the cookie should expire - - - - Gets a collection enumerator on the cookie list. - - collection enumerator - - - - Remove all cookies - - - - - Returns an enumerator that iterates through the collection. - - - - A that can be used to iterate through the collection. - - 1 - - - - Gets the count of cookies in the collection. - - - - - Gets the cookie of a given identifier (null if not existing). - - - - - This class is created as a wrapper, since there are two different cookie types in .Net (Cookie and HttpCookie). - The framework might switch class in the future and we dont want to have to replace all instances - - - - - Let's copy all the cookies. - - value from cookie header. - - - - Adds a cookie in the collection. - - cookie to add - cookie is null - - - - Gets a collection enumerator on the cookie list. - - collection enumerator - - - - Remove all cookies. - - - - - Returns an enumerator that iterates through the collection. - - - - A that can be used to iterate through the collection. - - 1 - - - - Remove a cookie from the collection. - - Name of cookie. - - - - Gets the count of cookies in the collection. - - - - - Gets the cookie of a given identifier (null if not existing). - - - - - New implementation of the HTTP listener. - - - Use the Create methods to create a default listener. - - - - - Initializes a new instance of the class. - - IP Address to accept connections on - TCP Port to listen on, default HTTP port is 80. - Factory used to create es. - address is null. - Port must be a positive number. - - - - Initializes a new instance of the class. - - The address. - The port. - The factory. - The certificate. - - - - Initializes a new instance of the class. - - The address. - The port. - The factory. - The certificate. - The protocol. - - - - Creates a new instance with default factories. - - Address that the listener should accept connections on. - Port that listener should accept connections on. - Created HTTP listener. - - - - Creates a new instance with default factories. - - Address that the listener should accept connections on. - Port that listener should accept connections on. - Certificate to use - Created HTTP listener. - - - - Creates a new instance with default factories. - - Address that the listener should accept connections on. - Port that listener should accept connections on. - Certificate to use - which HTTPS protocol to use, default is TLS. - Created HTTP listener. - - - - Can be used to create filtering of new connections. - - Accepted socket - - true if connection can be accepted; otherwise false. - - - - - A client have been accepted, but not handled, by the listener. - - - - - Generic helper functions for HTTP - - - - - Version string for HTTP v1.0 - - - - - Version string for HTTP v1.1 - - - - - An empty URI - - - - - Parses a query string. - - Query string (URI encoded) - A object if successful; otherwise - queryString is null. - If string cannot be parsed. - - - - A reverse proxy are used to act as a bridge between local (protected/hidden) websites - and public clients. - - A typical usage is to allow web servers on non standard ports to still be available - to the public clients, or allow web servers on private ips to be available. - - - - - - - Base url requested from browser - Base url on private web server - - // this will return contents from http://192.168.1.128/view/jonas when client requests http://www.gauffin.com/user/view/jonas - _server.Add(new ReverseProxyModule("http://www.gauffin.com/user/", "http://192.168.1.128/"); - - - - - Method that determines if an url should be handled or not by the module - - Url requested by the client. - true if module should handle the url. - - - - Method that process the url - - Information sent by the browser about the request - Information that is being sent back to the client. - Session used to - - - - Used to inform http server that - - - - - Eventarguments used when an exception is thrown by a module - - the exception - - - - Exception thrown in a module - - - - - PrototypeJS implementation of the javascript functions. - - - + Requests a url through ajax @@ -5251,21 +4978,12 @@ - - - Determins if a list of strings contains a specific value - - options to check in - value to find - true if value was found - case insensitive - - + Ajax requests that updates an element with the fetched content - URL to fetch. URL is NOT enclosed in quotes by the implementation. You need to do that yourself. + url to fetch. Url is NOT enclosed in quotes by the implementation. You need to do that yourself. element to update options in format "key, value, key, value". All keys should end with colon. A link tag. @@ -5275,179 +4993,461 @@ - + - A link that pop ups a Dialog (overlay div) + Opens contents in a dialog window. - URL to contents of dialog + url to contents of dialog link title - name, value, name, value - - A "a"-tag that popups a dialog when clicked - - Requires Control.Modal found here: http://livepipe.net/projects/control_modal/ - And the following JavaScript (load it in application.js): - - Event.observe(window, 'load', - function() { - document.getElementsByClassName('modal').each(function(link){ new Control.Modal(link); }); - } - ); - - - - WebHelper.DialogLink("/user/show/1", "show user", "onmouseover", "alert('booh!');"); - + name, value, name, value, all parameter names should end with colon. - - - create a modal dialog (usually using DIVs) - - url to fetch - dialog title - javascript/html attributes. javascript options ends with colon ':'. - - - + Close a javascript dialog window/div. javascript for closing a dialog. - + - + - javascript action that should be added to the "onsubmit" event in the form tag. + Lists content type mime types. - remember to encapsulate strings in '' - - All javascript option names should end with colon. - - - JSHelper.AjaxRequest("/user/show/1", "onsuccess:", "$('userInfo').update(result);"); - - - + - Implements HTTP Digest authentication. It's more secure than Basic auth since password is - encrypted with a "key" from the server. + text/plain + + + + + text/haml + + + + + content type for javascript documents = application/javascript - Keep in mind that the password is encrypted with MD5. Use a combination of SSL and digest auth to be secure. + + RFC 4329 states that text/javascript have been superseeded by + application/javascript. You might still want to check browser versions + since older ones do not support application/javascript. + + Browser support: http://krijnhoetmer.nl/stuff/javascript/mime-types/ - + - Initializes a new instance of the class. - - Delegate used to provide information used during authentication. - Delegate used to determine if authentication is required (may be null). - - - - Initializes a new instance of the class. - - Delegate used to provide information used during authentication. - - - - Used by test classes to be able to use hardcoded values + text/xml - + - An authentication response have been received from the web browser. - Check if it's correct + A list of content types - Contents from the Authorization header - Realm that should be authenticated - GET/POST/PUT/DELETE etc. - First option: true if username/password is correct but not cnonce - - Authentication object that is stored for the request. A user class or something like that. - - if authenticationHeader is invalid - If any of the paramters is empty or null. - - - Encrypts parameters into a Digest string - - Realm that the user want to log into. - User logging in - Users password. - HTTP method. - Uri/domain that generated the login prompt. - Quality of Protection. - "Number used ONCE" - Hexadecimal request counter. - "Client Number used ONCE" - Digest encrypted string - - + - Md5 hex encoded "userName:realm:password", without the quotes. - Md5 hex encoded "method:uri", without the quotes - Quality of Protection - "Number used ONCE" - Hexadecimal request counter. - Client number used once + Semicolon separated content types. + + + + Returns an enumerator that iterates through a collection. + + + An object that can be used to iterate through the collection. + + + + + Searches for the specified type + + Can also be a part of a type (searching for "xml" would return true for "application/xml"). + true if type was found. + + + + Get this first content type. + + + + + Fetch a content type + + Part of type ("xml" would return "application/xml") + + All content types are in lower case. + + + + Creates request parsers when needed. + + + + + Creates request parsers when needed. + + + + + Create a new request parser. + + Used when logging should be enabled. + A new request parser. + + + + Create a new request parser. + + Used when logging should be enabled. + A new request parser. + + + + The request requires user authentication. The response MUST include a + WWW-Authenticate header field (section 14.47) containing a challenge + applicable to the requested resource. + + The client MAY repeat the request with a suitable Authorization header + field (section 14.8). If the request already included Authorization + credentials, then the 401 response indicates that authorization has been + refused for those credentials. If the 401 response contains the same challenge + as the prior response, and the user agent has already attempted authentication + at least once, then the user SHOULD be presented the entity that was given in the response, + since that entity might include relevant diagnostic information. + + HTTP access authentication is explained in rfc2617: + http://www.ietf.org/rfc/rfc2617.txt + + (description is taken from + http://www.submissionchamber.com/help-guides/error-codes.php#sec10.4.2) + + + + + Create a new unauhtorized exception. + + + + + + Create a new unauhtorized exception. + + reason to why the request was unauthorized. + inner exception + + + + Create a new unauhtorized exception. + + reason to why the request was unauthorized. + + + + The purpose of this module is to serve files. + + + + + Initializes a new instance of the class. + + Uri to serve, for instance "/files/" + Path on hard drive where we should start looking for files + If true a Last-Modifed header will be sent upon requests urging web browser to cache files + + + + Initializes a new instance of the class. + + Uri to serve, for instance "/files/" + Path on hard drive where we should start looking for files + + + + Mimtypes that this class can handle per default + + + + + Determines if the request should be handled by this module. + Invoked by the + + + true if this module should handle it. + + + Illegal path + + + + check if source contains any of the chars. + + + - + - Create a response that can be sent in the WWW-Authenticate header. + Method that process the Uri. - Realm that the user should authenticate in - First options specifies if true if username/password is correct but not cnonce. - A correct auth request. - If realm is empty or null. + Information sent by the browser about the request + Information that is being sent back to the client. + Session used to + Failed to find file extension + File type is forbidden. - + - Decodes authorization header value + return a file extension from an absolute Uri path (or plain filename) + + + + + + + List with all mime-type that are allowed. + + All other mime types will result in a Forbidden http status code. + + + + characters that may not exist in a path. - header value - Encoding that the buffer is in - All headers and their values if successful; otherwise null - NameValueCollection header = DigestAuthentication.Decode("response=\"6629fae49393a05397450978507c4ef1\",\r\nc=00001", Encoding.ASCII); + fileMod.ForbiddenChars = new string[]{ "\\", "..", ":" }; - Can handle lots of whitespaces and new lines without failing. - + - Gets the current nonce. + Webhelper provides helpers for common tasks in HTML. + + + + Used to let the website use different javascript libraries. + Default is + + + + + Creates a link that invokes through ajax. + + url to fetch + link title + + optional options in format "key, value, key, value". + Javascript options starts with ':'. + + a link tag + + WebHelper.AjaxRequest("/users/add/", "Add user", "method:", "post", "onclick", "validate('this');"); + + + + + Builds a link that updates an element with the fetched ajax content. + + Url to fetch content from + link title + html element to update with the results of the ajax request. + optional options in format "key, value, key, value" + A link tag. + + + + A link that pop ups a Dialog (overlay div) + + url to contents of dialog + link title + name/value of html attributes. + A "a"-tag that popups a dialog when clicked + + WebHelper.DialogLink("/user/show/1", "show user", "onmouseover", "alert('booh!');"); + + + + + Create/Open a dialog box using ajax + + + + - + - Gets the Md5 hash bin hex2. + Close a javascript dialog window/div. - To be hashed. + javascript for closing a dialog. + + + + + Create a <form> tag. + + name of form + action to invoke on submit + form should be posted as ajax + html code + + WebHelper.FormStart("frmLogin", "/user/login", Request.IsAjax); + + + + + Create a link tag. + + url to go to + link title (text that is displayed) + html attributes, name, value, name, value + html code + + WebHelper.Link("/user/show/1", "Show user", "id", "showUser", "onclick", "return confirm('Are you shure?');"); + + + + + Build a link + + url to go to. + title of link (displayed text) + extra html attributes. + a complete link + + + + Build a link + + url to go to. + title of link (displayed text) + extra html attributes. + a complete link + more options + + + + Obsolete + + Obsolete + Obsolete + Obsolete + Obsolete + Obsolete + Obsolete + + + + Obsolete + + Obsolete + Obsolete + Obsolete + Obsolete + Obsolete + Obsolete + Obsolete + + + + Render errors into a UL with class "errors" + + class used by UL-tag. + items to list + an unordered html list. + + + + Render errors into a UL with class "errors" + + class used by UL-tag. + items to list + an unordered html list. + + + + Render errors into a UL with class "errors" + + - + - determines if the nonce is valid or has expired. + Generates a list with html attributes. - nonce value (check wikipedia for info) - true if the nonce has not expired. + StringBuilder that the options should be added to. + attributes set by user. + attributes set by any of the helper classes. - + - name used in http request. + Generates a list with html attributes. + StringBuilder that the options should be added to. + - + - Gets or sets whether the token supplied in is a - HA1 generated string. + Delegate used by to populate select options. + current object (for instance a User). + Text that should be displayed in the value part of a <optiongt;-tag. + Text shown in the select list. + + // Class that is going to be used in a SELECT-tag. + public class User + { + private readonly string _realName; + private readonly int _id; + public User(int id, string realName) + { + _id = id; + _realName = realName; + } + public string RealName + { + get { return _realName; } + } + + public int Id + { + get { return _id; } + } + } + + // Using an inline delegate to generate the select list + public void UserInlineDelegate() + { + List<User> items = new List<User>(); + items.Add(new User(1, "adam")); + items.Add(new User(2, "bertial")); + items.Add(new User(3, "david")); + string htmlSelect = Select("users", "users", items, delegate(object o, out object id, out object value) + { + User user = (User)o; + id = user.Id; + value = user.RealName; + }, 2, true); + } + + // Using an method as delegate to generate the select list. + public void UseExternalDelegate() + { + List<User> items = new List<User>(); + items.Add(new User(1, "adam")); + items.Add(new User(2, "bertial")); + items.Add(new User(3, "david")); + string htmlSelect = Select("users", "users", items, UserOptions, 1, true); + } + + // delegate returning id and title + public static void UserOptions(object o, out object id, out object title) + { + User user = (User)o; + id = user.Id; + value = user.RealName; + } /// From 4867a7cbbf7302845fff031db5eae6fbf93bf26b Mon Sep 17 00:00:00 2001 From: teravus Date: Thu, 7 Feb 2013 10:26:48 -0500 Subject: [PATCH 5/8] This is the final commit that enables the Websocket handler --- .../Servers/HttpServer/BaseHttpServer.cs | 12 +- .../HttpServer/WebsocketServerHandler.cs | 1085 +++++++++++++++++ bin/HttpServer_OpenSim.dll | Bin 116224 -> 116224 bytes bin/HttpServer_OpenSim.pdb | Bin 302592 -> 343552 bytes 4 files changed, 1095 insertions(+), 2 deletions(-) create mode 100644 OpenSim/Framework/Servers/HttpServer/WebsocketServerHandler.cs diff --git a/OpenSim/Framework/Servers/HttpServer/BaseHttpServer.cs b/OpenSim/Framework/Servers/HttpServer/BaseHttpServer.cs index dcfe99a070..70c531c105 100644 --- a/OpenSim/Framework/Servers/HttpServer/BaseHttpServer.cs +++ b/OpenSim/Framework/Servers/HttpServer/BaseHttpServer.cs @@ -54,7 +54,15 @@ namespace OpenSim.Framework.Servers.HttpServer private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private HttpServerLogWriter httpserverlog = new HttpServerLogWriter(); - public delegate void WebSocketRequestDelegate(string servicepath, WebSocketHTTPServerHandler handler); + + /// + /// This is a pending websocket request before it got an sucessful upgrade response. + /// The consumer must call handler.HandshakeAndUpgrade() to signal to the handler to + /// start the connection and optionally provide an origin authentication method. + /// + /// + /// + public delegate void WebSocketRequestDelegate(string servicepath, WebSocketHttpServerHandler handler); /// /// Gets or sets the debug level. @@ -440,7 +448,7 @@ namespace OpenSim.Framework.Servers.HttpServer } if (dWebSocketRequestDelegate != null) { - dWebSocketRequestDelegate(req.Url.AbsolutePath, new WebSocketHTTPServerHandler(req, context, 16384)); + dWebSocketRequestDelegate(req.Url.AbsolutePath, new WebSocketHttpServerHandler(req, context, 8192)); return; } diff --git a/OpenSim/Framework/Servers/HttpServer/WebsocketServerHandler.cs b/OpenSim/Framework/Servers/HttpServer/WebsocketServerHandler.cs new file mode 100644 index 0000000000..cfb1605001 --- /dev/null +++ b/OpenSim/Framework/Servers/HttpServer/WebsocketServerHandler.cs @@ -0,0 +1,1085 @@ +/* + * 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.IO; +using System.Security.Cryptography; +using System.Text; +using HttpServer; + +namespace OpenSim.Framework.Servers.HttpServer +{ + // Sealed class. If you're going to unseal it, implement IDisposable. + /// + /// This class implements websockets. It grabs the network context from C#Webserver and utilizes it directly as a tcp streaming service + /// + public sealed class WebSocketHttpServerHandler : BaseRequestHandler + { + + private class WebSocketState + { + public List ReceivedBytes; + public int ExpectedBytes; + public WebsocketFrameHeader Header; + public bool FrameComplete; + public WebSocketFrame ContinuationFrame; + } + + /// + /// Binary Data will trigger this event + /// + public event DataDelegate OnData; + + /// + /// Textual Data will trigger this event + /// + public event TextDelegate OnText; + + /// + /// A ping request form the other side will trigger this event. + /// This class responds to the ping automatically. You shouldn't send a pong. + /// it's informational. + /// + public event PingDelegate OnPing; + + /// + /// This is a response to a ping you sent. + /// + public event PongDelegate OnPong; + + /// + /// This is a regular HTTP Request... This may be removed in the future. + /// + public event RegularHttpRequestDelegate OnRegularHttpRequest; + + /// + /// When the upgrade from a HTTP request to a Websocket is completed, this will be fired + /// + public event UpgradeCompletedDelegate OnUpgradeCompleted; + + /// + /// If the upgrade failed, this will be fired + /// + public event UpgradeFailedDelegate OnUpgradeFailed; + + /// + /// When the websocket is closed, this will be fired. + /// + public event CloseDelegate OnClose; + + /// + /// Set this delegate to allow your module to validate the origin of the + /// Websocket request. Primary line of defense against cross site scripting + /// + public ValidateHandshake HandshakeValidateMethodOverride = null; + + private OSHttpRequest _request; + private HTTPNetworkContext _networkContext; + private IHttpClientContext _clientContext; + + private int _pingtime = 0; + private byte[] _buffer; + private int _bufferPosition; + private int _bufferLength; + private bool _closing; + private bool _upgraded; + + private const string HandshakeAcceptText = + "HTTP/1.1 101 Switching Protocols\r\n" + + "upgrade: websocket\r\n" + + "Connection: Upgrade\r\n" + + "sec-websocket-accept: {0}\r\n\r\n";// + + //"{1}"; + + private const string HandshakeDeclineText = + "HTTP/1.1 {0} {1}\r\n" + + "Connection: close\r\n\r\n"; + + /// + /// Mysterious constant defined in RFC6455 to append to the client provided security key + /// + private const string WebsocketHandshakeAcceptHashConstant = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11"; + + public WebSocketHttpServerHandler(OSHttpRequest preq, IHttpClientContext pContext, int bufferlen) + : base(preq.HttpMethod, preq.Url.OriginalString) + { + _request = preq; + _networkContext = pContext.GiveMeTheNetworkStreamIKnowWhatImDoing(); + _clientContext = pContext; + _bufferLength = bufferlen; + _buffer = new byte[_bufferLength]; + } + + // Sealed class implments destructor and an internal dispose method. complies with C# unmanaged resource best practices. + ~WebSocketHttpServerHandler() + { + Dispose(); + + } + + /// + /// Sets the length of the stream buffer + /// + /// Byte length. + public void SetChunksize(int pChunk) + { + if (!_upgraded) + { + _buffer = new byte[pChunk]; + } + else + { + throw new InvalidOperationException("You must set the chunksize before the connection is upgraded"); + } + } + + /// + /// This is the famous nagle. + /// + public bool NoDelay_TCP_Nagle + { + get + { + if (_networkContext != null && _networkContext.Socket != null) + { + return _networkContext.Socket.NoDelay; + } + else + { + throw new InvalidOperationException("The socket has been shutdown"); + } + } + set + { + if (_networkContext != null && _networkContext.Socket != null) + _networkContext.Socket.NoDelay = value; + else + { + throw new InvalidOperationException("The socket has been shutdown"); + } + } + } + + /// + /// This triggers the websocket to start the upgrade process... + /// This is a Generalized Networking 'common sense' helper method. Some people expect to call Start() instead + /// of the more context appropriate HandshakeAndUpgrade() + /// + public void Start() + { + HandshakeAndUpgrade(); + } + + /// + /// This triggers the websocket start the upgrade process + /// + public void HandshakeAndUpgrade() + { + string webOrigin = string.Empty; + string websocketKey = string.Empty; + string acceptKey = string.Empty; + string accepthost = string.Empty; + if (!string.IsNullOrEmpty(_request.Headers["origin"])) + webOrigin = _request.Headers["origin"]; + + if (!string.IsNullOrEmpty(_request.Headers["sec-websocket-key"])) + websocketKey = _request.Headers["sec-websocket-key"]; + + if (!string.IsNullOrEmpty(_request.Headers["host"])) + accepthost = _request.Headers["host"]; + + if (string.IsNullOrEmpty(_request.Headers["upgrade"])) + { + FailUpgrade(OSHttpStatusCode.ClientErrorBadRequest, "no upgrade request submitted"); + } + + string connectionheader = _request.Headers["upgrade"]; + if (connectionheader.ToLower() != "websocket") + { + FailUpgrade(OSHttpStatusCode.ClientErrorBadRequest, "no connection upgrade request submitted"); + } + + // If the object consumer provided a method to validate the origin, we should call it and give the client a success or fail. + // If not.. we should accept any. The assumption here is that there would be no Websocket handlers registered in baseHTTPServer unless + // Something asked for it... + if (HandshakeValidateMethodOverride != null) + { + if (HandshakeValidateMethodOverride(webOrigin, websocketKey, accepthost)) + { + acceptKey = GenerateAcceptKey(websocketKey); + string rawaccept = string.Format(HandshakeAcceptText, acceptKey); + SendUpgradeSuccess(rawaccept); + + } + else + { + FailUpgrade(OSHttpStatusCode.ClientErrorForbidden, "Origin Validation Failed"); + } + } + else + { + acceptKey = GenerateAcceptKey(websocketKey); + string rawaccept = string.Format(HandshakeAcceptText, acceptKey); + SendUpgradeSuccess(rawaccept); + } + } + + /// + /// Generates a handshake response key string based on the client's + /// provided key to prove to the client that we're allowing the Websocket + /// upgrade of our own free will and we were not coerced into doing it. + /// + /// Client provided security key + /// + private static string GenerateAcceptKey(string key) + { + if (string.IsNullOrEmpty(key)) + return string.Empty; + + string acceptkey = key + WebsocketHandshakeAcceptHashConstant; + + SHA1 hashobj = SHA1.Create(); + string ret = Convert.ToBase64String(hashobj.ComputeHash(Encoding.UTF8.GetBytes(acceptkey))); + hashobj.Clear(); + + return ret; + } + + /// + /// Informs the otherside that we accepted their upgrade request + /// + /// The HTTP 1.1 101 response that says Yay \o/ + private void SendUpgradeSuccess(string pHandshakeResponse) + { + // Create a new websocket state so we can keep track of data in between network reads. + WebSocketState socketState = new WebSocketState() { ReceivedBytes = new List(), Header = WebsocketFrameHeader.HeaderDefault(), FrameComplete = true}; + + byte[] bhandshakeResponse = Encoding.UTF8.GetBytes(pHandshakeResponse); + try + { + + // Begin reading the TCP stream before writing the Upgrade success message to the other side of the stream. + _networkContext.Stream.BeginRead(_buffer, 0, _bufferLength, OnReceive, socketState); + + // Write the upgrade handshake success message + _networkContext.Stream.Write(bhandshakeResponse, 0, bhandshakeResponse.Length); + _networkContext.Stream.Flush(); + _upgraded = true; + UpgradeCompletedDelegate d = OnUpgradeCompleted; + if (d != null) + d(this, new UpgradeCompletedEventArgs()); + } + catch (IOException fail) + { + Close(string.Empty); + } + catch (ObjectDisposedException fail) + { + Close(string.Empty); + } + + } + + /// + /// The server has decided not to allow the upgrade to a websocket for some reason. The Http 1.1 response that says Nay >:( + /// + /// HTTP Status reflecting the reason why + /// Textual reason for the upgrade fail + private void FailUpgrade(OSHttpStatusCode pCode, string pMessage ) + { + string handshakeResponse = string.Format(HandshakeDeclineText, (int)pCode, pMessage.Replace("\n", string.Empty).Replace("\r", string.Empty)); + byte[] bhandshakeResponse = Encoding.UTF8.GetBytes(handshakeResponse); + _networkContext.Stream.Write(bhandshakeResponse, 0, bhandshakeResponse.Length); + _networkContext.Stream.Flush(); + _networkContext.Stream.Dispose(); + + UpgradeFailedDelegate d = OnUpgradeFailed; + if (d != null) + d(this,new UpgradeFailedEventArgs()); + } + + + /// + /// This is our ugly Async OnReceive event handler. + /// This chunks the input stream based on the length of the provided buffer and processes out + /// as many frames as it can. It then moves the unprocessed data to the beginning of the buffer. + /// + /// Our Async State from beginread + private void OnReceive(IAsyncResult ar) + { + WebSocketState _socketState = ar.AsyncState as WebSocketState; + try + { + int bytesRead = _networkContext.Stream.EndRead(ar); + if (bytesRead == 0) + { + // Do Disconnect + _networkContext.Stream.Dispose(); + _networkContext = null; + return; + } + _bufferPosition += bytesRead; + + if (_bufferPosition > _bufferLength) + { + // Message too big for chunksize.. not sure how this happened... + //Close(string.Empty); + } + + int offset = 0; + bool headerread = true; + int headerforwardposition = 0; + while (headerread && offset < bytesRead) + { + if (_socketState.FrameComplete) + { + WebsocketFrameHeader pheader = WebsocketFrameHeader.ZeroHeader; + + headerread = WebSocketReader.TryReadHeader(_buffer, offset, _bufferPosition - offset, out pheader, + out headerforwardposition); + offset += headerforwardposition; + + if (headerread) + { + _socketState.FrameComplete = false; + + if (pheader.PayloadLen > 0) + { + if ((int) pheader.PayloadLen > _bufferPosition - offset) + { + byte[] writebytes = new byte[_bufferPosition - offset]; + + Buffer.BlockCopy(_buffer, offset, writebytes, 0, (int) _bufferPosition - offset); + _socketState.ExpectedBytes = (int) pheader.PayloadLen; + _socketState.ReceivedBytes.AddRange(writebytes); + _socketState.Header = pheader; // We need to add the header so that we can unmask it + offset += (int) _bufferPosition - offset; + } + else + { + byte[] writebytes = new byte[pheader.PayloadLen]; + Buffer.BlockCopy(_buffer, offset, writebytes, 0, (int) pheader.PayloadLen); + WebSocketReader.Mask(pheader.Mask, writebytes); + pheader.IsMasked = false; + _socketState.FrameComplete = true; + _socketState.ReceivedBytes.AddRange(writebytes); + _socketState.Header = pheader; + offset += (int) pheader.PayloadLen; + } + } + else + { + pheader.Mask = 0; + _socketState.FrameComplete = true; + _socketState.Header = pheader; + } + + + + if (_socketState.FrameComplete) + { + ProcessFrame(_socketState); + _socketState.Header.SetDefault(); + _socketState.ReceivedBytes.Clear(); + _socketState.ExpectedBytes = 0; + + } + + } + } + else + { + WebsocketFrameHeader frameHeader = _socketState.Header; + int bytesleft = _socketState.ExpectedBytes - _socketState.ReceivedBytes.Count; + + if (bytesleft > _bufferPosition) + { + byte[] writebytes = new byte[_bufferPosition]; + + Buffer.BlockCopy(_buffer, offset, writebytes, 0, (int) _bufferPosition); + _socketState.ReceivedBytes.AddRange(writebytes); + _socketState.Header = frameHeader; // We need to add the header so that we can unmask it + offset += (int) _bufferPosition; + } + else + { + byte[] writebytes = new byte[_bufferPosition]; + Buffer.BlockCopy(_buffer, offset, writebytes, 0, (int) _bufferPosition); + _socketState.FrameComplete = true; + _socketState.ReceivedBytes.AddRange(writebytes); + _socketState.Header = frameHeader; + offset += (int) _bufferPosition; + } + if (_socketState.FrameComplete) + { + ProcessFrame(_socketState); + _socketState.Header.SetDefault(); + _socketState.ReceivedBytes.Clear(); + _socketState.ExpectedBytes = 0; + // do some processing + } + + } + } + if (offset > 0) + { + // If the buffer is maxed out.. we can just move the cursor. Nothing to move to the beginning. + if (offset <_buffer.Length) + Buffer.BlockCopy(_buffer, offset, _buffer, 0, _bufferPosition - offset); + _bufferPosition -= offset; + } + if (_networkContext.Stream != null && _networkContext.Stream.CanRead && !_closing) + { + _networkContext.Stream.BeginRead(_buffer, _bufferPosition, _bufferLength - _bufferPosition, OnReceive, + _socketState); + } + else + { + // We can't read the stream anymore... + } + + } + catch (IOException fail) + { + Close(string.Empty); + } + catch (ObjectDisposedException fail) + { + Close(string.Empty); + } + } + + /// + /// Sends a string to the other side + /// + /// the string message that is to be sent + public void SendMessage(string message) + { + byte[] messagedata = Encoding.UTF8.GetBytes(message); + WebSocketFrame textMessageFrame = new WebSocketFrame() { Header = WebsocketFrameHeader.HeaderDefault(), WebSocketPayload = messagedata }; + textMessageFrame.Header.Opcode = WebSocketReader.OpCode.Text; + textMessageFrame.Header.IsEnd = true; + SendSocket(textMessageFrame.ToBytes()); + + } + + public void SendData(byte[] data) + { + WebSocketFrame dataMessageFrame = new WebSocketFrame() { Header = WebsocketFrameHeader.HeaderDefault(), WebSocketPayload = data}; + dataMessageFrame.Header.IsEnd = true; + dataMessageFrame.Header.Opcode = WebSocketReader.OpCode.Binary; + SendSocket(dataMessageFrame.ToBytes()); + + } + + /// + /// Writes raw bytes to the websocket. Unframed data will cause disconnection + /// + /// + private void SendSocket(byte[] data) + { + if (!_closing) + { + try + { + + _networkContext.Stream.Write(data, 0, data.Length); + } + catch (IOException) + { + + } + } + } + + /// + /// Sends a Ping check to the other side. The other side SHOULD respond as soon as possible with a pong frame. This interleaves with incoming fragmented frames. + /// + public void SendPingCheck() + { + WebSocketFrame pingFrame = new WebSocketFrame() { Header = WebsocketFrameHeader.HeaderDefault(), WebSocketPayload = new byte[0] }; + pingFrame.Header.Opcode = WebSocketReader.OpCode.Ping; + pingFrame.Header.IsEnd = true; + _pingtime = Util.EnvironmentTickCount(); + SendSocket(pingFrame.ToBytes()); + } + + /// + /// Closes the websocket connection. Sends a close message to the other side if it hasn't already done so. + /// + /// + public void Close(string message) + { + if (_networkContext.Stream != null) + { + if (_networkContext.Stream.CanWrite) + { + byte[] messagedata = Encoding.UTF8.GetBytes(message); + WebSocketFrame closeResponseFrame = new WebSocketFrame() + { + Header = WebsocketFrameHeader.HeaderDefault(), + WebSocketPayload = messagedata + }; + closeResponseFrame.Header.Opcode = WebSocketReader.OpCode.Close; + closeResponseFrame.Header.PayloadLen = (ulong) messagedata.Length; + closeResponseFrame.Header.IsEnd = true; + SendSocket(closeResponseFrame.ToBytes()); + } + } + CloseDelegate closeD = OnClose; + if (closeD != null) + { + closeD(this, new CloseEventArgs()); + } + + _closing = true; + } + + /// + /// Processes a websocket frame and triggers consumer events + /// + /// We need to modify the websocket state here depending on the frame + private void ProcessFrame(WebSocketState psocketState) + { + if (psocketState.Header.IsMasked) + { + byte[] unmask = psocketState.ReceivedBytes.ToArray(); + WebSocketReader.Mask(psocketState.Header.Mask, unmask); + psocketState.ReceivedBytes = new List(unmask); + } + + switch (psocketState.Header.Opcode) + { + case WebSocketReader.OpCode.Ping: + PingDelegate pingD = OnPing; + if (pingD != null) + { + pingD(this, new PingEventArgs()); + } + + WebSocketFrame pongFrame = new WebSocketFrame(){Header = WebsocketFrameHeader.HeaderDefault(),WebSocketPayload = new byte[0]}; + pongFrame.Header.Opcode = WebSocketReader.OpCode.Pong; + pongFrame.Header.IsEnd = true; + SendSocket(pongFrame.ToBytes()); + break; + case WebSocketReader.OpCode.Pong: + + PongDelegate pongD = OnPong; + if (pongD != null) + { + pongD(this, new PongEventArgs(){PingResponseMS = Util.EnvironmentTickCountSubtract(Util.EnvironmentTickCount(),_pingtime)}); + } + break; + case WebSocketReader.OpCode.Binary: + if (!psocketState.Header.IsEnd) // Not done, so we need to store this and wait for the end frame. + { + psocketState.ContinuationFrame = new WebSocketFrame + { + Header = psocketState.Header, + WebSocketPayload = + psocketState.ReceivedBytes.ToArray() + }; + } + else + { + // Send Done Event! + DataDelegate dataD = OnData; + if (dataD != null) + { + dataD(this,new WebsocketDataEventArgs(){Data = psocketState.ReceivedBytes.ToArray()}); + } + } + break; + case WebSocketReader.OpCode.Text: + if (!psocketState.Header.IsEnd) // Not done, so we need to store this and wait for the end frame. + { + psocketState.ContinuationFrame = new WebSocketFrame + { + Header = psocketState.Header, + WebSocketPayload = + psocketState.ReceivedBytes.ToArray() + }; + } + else + { + TextDelegate textD = OnText; + if (textD != null) + { + textD(this, new WebsocketTextEventArgs() { Data = Encoding.UTF8.GetString(psocketState.ReceivedBytes.ToArray()) }); + } + + // Send Done Event! + } + break; + case WebSocketReader.OpCode.Continue: // Continuation. Multiple frames worth of data for one message. Only valid when not using Control Opcodes + //Console.WriteLine("currhead " + psocketState.Header.IsEnd); + //Console.WriteLine("Continuation! " + psocketState.ContinuationFrame.Header.IsEnd); + byte[] combineddata = new byte[psocketState.ReceivedBytes.Count+psocketState.ContinuationFrame.WebSocketPayload.Length]; + byte[] newdata = psocketState.ReceivedBytes.ToArray(); + Buffer.BlockCopy(psocketState.ContinuationFrame.WebSocketPayload, 0, combineddata, 0, psocketState.ContinuationFrame.WebSocketPayload.Length); + Buffer.BlockCopy(newdata, 0, combineddata, + psocketState.ContinuationFrame.WebSocketPayload.Length, newdata.Length); + psocketState.ContinuationFrame.WebSocketPayload = combineddata; + psocketState.Header.PayloadLen = (ulong)combineddata.Length; + if (psocketState.Header.IsEnd) + { + if (psocketState.ContinuationFrame.Header.Opcode == WebSocketReader.OpCode.Text) + { + // Send Done event + TextDelegate textD = OnText; + if (textD != null) + { + textD(this, new WebsocketTextEventArgs() { Data = Encoding.UTF8.GetString(combineddata) }); + } + } + else if (psocketState.ContinuationFrame.Header.Opcode == WebSocketReader.OpCode.Binary) + { + // Send Done event + DataDelegate dataD = OnData; + if (dataD != null) + { + dataD(this, new WebsocketDataEventArgs() { Data = combineddata }); + } + } + else + { + // protocol violation + } + psocketState.ContinuationFrame = null; + } + break; + case WebSocketReader.OpCode.Close: + Close(string.Empty); + + break; + + } + psocketState.Header.SetDefault(); + psocketState.ReceivedBytes.Clear(); + psocketState.ExpectedBytes = 0; + } + public void Dispose() + { + if (_networkContext != null && _networkContext.Stream != null) + { + if (_networkContext.Stream.CanWrite) + _networkContext.Stream.Flush(); + _networkContext.Stream.Close(); + _networkContext.Stream.Dispose(); + _networkContext.Stream = null; + } + + if (_request != null && _request.InputStream != null) + { + _request.InputStream.Close(); + _request.InputStream.Dispose(); + _request = null; + } + + if (_clientContext != null) + { + _clientContext.Close(); + _clientContext = null; + } + } + } + + /// + /// Reads a byte stream and returns Websocket frames. + /// + public class WebSocketReader + { + /// + /// Bit to determine if the frame read on the stream is the last frame in a sequence of fragmented frames + /// + private const byte EndBit = 0x80; + + /// + /// These are the Frame Opcodes + /// + public enum OpCode + { + // Data Opcodes + Continue = 0x0, + Text = 0x1, + Binary = 0x2, + + // Control flow Opcodes + Close = 0x8, + Ping = 0x9, + Pong = 0xA + } + + /// + /// Masks and Unmasks data using the frame mask. Mask is applied per octal + /// Note: Frames from clients MUST be masked + /// Note: Frames from servers MUST NOT be masked + /// + /// Int representing 32 bytes of mask data. Mask is applied per octal + /// + public static void Mask(int pMask, byte[] pBuffer) + { + byte[] maskKey = BitConverter.GetBytes(pMask); + int currentMaskIndex = 0; + for (int i = 0; i < pBuffer.Length; i++) + { + pBuffer[i] = (byte)(pBuffer[i] ^ maskKey[currentMaskIndex]); + if (currentMaskIndex == 3) + { + currentMaskIndex = 0; + } + else + { + currentMaskIndex++; + + } + + } + } + + /// + /// Attempts to read a header off the provided buffer. Returns true, exports a WebSocketFrameheader, + /// and an int to move the buffer forward when it reads a header. False when it can't read a header + /// + /// Bytes read from the stream + /// Starting place in the stream to begin trying to read from + /// Lenth in the stream to try and read from. Provided for cases where the + /// buffer's length is larger then the data in it + /// Outputs the read WebSocket frame header + /// Informs the calling stream to move the buffer forward + /// True if it got a header, False if it didn't get a header + public static bool TryReadHeader(byte[] pBuffer, int pOffset, int length, out WebsocketFrameHeader oHeader, + out int moveBuffer) + { + oHeader = WebsocketFrameHeader.ZeroHeader; + int minumheadersize = 2; + if (length > pBuffer.Length - pOffset) + throw new ArgumentOutOfRangeException("The Length specified was larger the byte array supplied"); + if (length < minumheadersize) + { + moveBuffer = 0; + return false; + } + + byte nibble1 = (byte)(pBuffer[pOffset] & 0xF0); //FIN/RSV1/RSV2/RSV3 + byte nibble2 = (byte)(pBuffer[pOffset] & 0x0F); // Opcode block + + oHeader = new WebsocketFrameHeader(); + oHeader.SetDefault(); + + if ((nibble1 & WebSocketReader.EndBit) == WebSocketReader.EndBit) + { + oHeader.IsEnd = true; + } + else + { + oHeader.IsEnd = false; + } + //Opcode + oHeader.Opcode = (WebSocketReader.OpCode)nibble2; + //Mask + oHeader.IsMasked = Convert.ToBoolean((pBuffer[pOffset + 1] & 0x80) >> 7); + + // Payload length + oHeader.PayloadLen = (byte)(pBuffer[pOffset + 1] & 0x7F); + + int index = 2; // LargerPayload length starts at byte 3 + + switch (oHeader.PayloadLen) + { + case 126: + minumheadersize += 2; + if (length < minumheadersize) + { + moveBuffer = 0; + return false; + } + Array.Reverse(pBuffer, pOffset + index, 2); // two bytes + oHeader.PayloadLen = BitConverter.ToUInt16(pBuffer, pOffset + index); + index += 2; + break; + case 127: // we got more this is a bigger frame + // 8 bytes - uint64 - most significant bit 0 network byte order + minumheadersize += 8; + if (length < minumheadersize) + { + moveBuffer = 0; + return false; + } + Array.Reverse(pBuffer, pOffset + index, 8); + oHeader.PayloadLen = BitConverter.ToUInt64(pBuffer, pOffset + index); + index += 8; + break; + + } + //oHeader.PayloadLeft = oHeader.PayloadLen; // Start the count in case it's chunked over the network. This is different then frame fragmentation + if (oHeader.IsMasked) + { + minumheadersize += 4; + if (length < minumheadersize) + { + moveBuffer = 0; + return false; + } + oHeader.Mask = BitConverter.ToInt32(pBuffer, pOffset + index); + index += 4; + } + moveBuffer = index; + return true; + + } + } + + /// + /// RFC6455 Websocket Frame + /// + public class WebSocketFrame + { + /* + * RFC6455 +nib 0 1 2 3 4 5 6 7 +byt 0 1 2 3 +dec 0 1 2 3 + 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 + +-+-+-+-+-------+-+-------------+-------------------------------+ + |F|R|R|R| opcode|M| Payload len | Extended payload length | + |I|S|S|S| (4) |A| (7) | (16/64) + + |N|V|V|V| |S| | (if payload len==126/127) | + | |1|2|3| |K| | + + +-+-+-+-+-------+-+-------------+ - - - - - - - - - - - - - - - + + | Extended payload length continued, if payload len == 127 | + + - - - - - - - - - - - - - - - +-------------------------------+ + | |Masking-key, if MASK set to 1 | + +-------------------------------+-------------------------------+ + | Masking-key (continued) | Payload Data | + +-------------------------------- - - - - - - - - - - - - - - - + + : Payload Data continued ... : + + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + | Payload Data continued ... | + +---------------------------------------------------------------+ + + * When reading these, the frames are possibly fragmented and interleaved with control frames + * the fragmented frames are not interleaved with data frames. Just control frames + */ + public static readonly WebSocketFrame DefaultFrame = new WebSocketFrame(){Header = new WebsocketFrameHeader(),WebSocketPayload = new byte[0]}; + public WebsocketFrameHeader Header; + public byte[] WebSocketPayload; + + public byte[] ToBytes() + { + Header.PayloadLen = (ulong)WebSocketPayload.Length; + return Header.ToBytes(WebSocketPayload); + } + + } + + public struct WebsocketFrameHeader + { + //public byte CurrentMaskIndex; + /// + /// The last frame in a sequence of fragmented frames or the one and only frame for this message. + /// + public bool IsEnd; + + /// + /// Returns whether the payload data is masked or not. Data from Clients MUST be masked, Data from Servers MUST NOT be masked + /// + public bool IsMasked; + + /// + /// A set of cryptologically sound random bytes XoR-ed against the payload octally. Looped + /// + public int Mask; + /* +byt 0 1 2 3 + 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 + +---------------+---------------+---------------+---------------+ + | Octal 1 | Octal 2 | Octal 3 | Octal 4 | + +---------------+---------------+---------------+---------------+ +*/ + + + public WebSocketReader.OpCode Opcode; + + public UInt64 PayloadLen; + //public UInt64 PayloadLeft; + // Payload is X + Y + //public UInt64 ExtensionDataLength; + //public UInt64 ApplicationDataLength; + public static readonly WebsocketFrameHeader ZeroHeader = WebsocketFrameHeader.HeaderDefault(); + + public void SetDefault() + { + + //CurrentMaskIndex = 0; + IsEnd = true; + IsMasked = true; + Mask = 0; + Opcode = WebSocketReader.OpCode.Close; + // PayloadLeft = 0; + PayloadLen = 0; + // ExtensionDataLength = 0; + // ApplicationDataLength = 0; + + } + + /// + /// Returns a byte array representing the Frame header + /// + /// This is the frame data payload. The header describes the size of the payload. + /// If payload is null, a Zero sized payload is assumed + /// Returns a byte array representing the frame header + public byte[] ToBytes(byte[] payload) + { + List result = new List(); + + // Squeeze in our opcode and our ending bit. + result.Add((byte)((byte)Opcode | (IsEnd?0x80:0x00) )); + + // Again with the three different byte interpretations of size.. + + //bytesize + if (PayloadLen <= 125) + { + result.Add((byte) PayloadLen); + } //Uint16 + else if (PayloadLen <= ushort.MaxValue) + { + result.Add(126); + byte[] payloadLengthByte = BitConverter.GetBytes(Convert.ToUInt16(PayloadLen)); + Array.Reverse(payloadLengthByte); + result.AddRange(payloadLengthByte); + } //UInt64 + else + { + result.Add(127); + byte[] payloadLengthByte = BitConverter.GetBytes(PayloadLen); + Array.Reverse(payloadLengthByte); + result.AddRange(payloadLengthByte); + } + + // Only add a payload if it's not null + if (payload != null) + { + result.AddRange(payload); + } + return result.ToArray(); + } + + /// + /// A Helper method to define the defaults + /// + /// + + public static WebsocketFrameHeader HeaderDefault() + { + return new WebsocketFrameHeader + { + //CurrentMaskIndex = 0, + IsEnd = false, + IsMasked = true, + Mask = 0, + Opcode = WebSocketReader.OpCode.Close, + //PayloadLeft = 0, + PayloadLen = 0, + // ExtensionDataLength = 0, + // ApplicationDataLength = 0 + }; + } + } + + public delegate void DataDelegate(object sender, WebsocketDataEventArgs data); + + public delegate void TextDelegate(object sender, WebsocketTextEventArgs text); + + public delegate void PingDelegate(object sender, PingEventArgs pingdata); + + public delegate void PongDelegate(object sender, PongEventArgs pongdata); + + public delegate void RegularHttpRequestDelegate(object sender, RegularHttpRequestEvnetArgs request); + + public delegate void UpgradeCompletedDelegate(object sender, UpgradeCompletedEventArgs completeddata); + + public delegate void UpgradeFailedDelegate(object sender, UpgradeFailedEventArgs faileddata); + + public delegate void CloseDelegate(object sender, CloseEventArgs closedata); + + public delegate bool ValidateHandshake(string pWebOrigin, string pWebSocketKey, string pHost); + + + public class WebsocketDataEventArgs : EventArgs + { + public byte[] Data; + } + + public class WebsocketTextEventArgs : EventArgs + { + public string Data; + } + + public class PingEventArgs : EventArgs + { + /// + /// The ping event can arbitrarily contain data + /// + public byte[] Data; + } + + public class PongEventArgs : EventArgs + { + /// + /// The pong event can arbitrarily contain data + /// + public byte[] Data; + + public int PingResponseMS; + + } + + public class RegularHttpRequestEvnetArgs : EventArgs + { + + } + + public class UpgradeCompletedEventArgs : EventArgs + { + + } + + public class UpgradeFailedEventArgs : EventArgs + { + + } + + public class CloseEventArgs : EventArgs + { + + } + + +} diff --git a/bin/HttpServer_OpenSim.dll b/bin/HttpServer_OpenSim.dll index 9cd1e0857b42badaad81e63a72ebb2c26effeb8a..fd7ad742d03a28238308f402e729fdaa007443e2 100755 GIT binary patch delta 4156 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 diff --git a/bin/HttpServer_OpenSim.pdb b/bin/HttpServer_OpenSim.pdb index d20a0c5074c5c0c5d35a409c4667a0e275481e51..f56e891d4c4e1e674ee26e75c59ec4c5bc7ebbe7 100644 GIT binary patch delta 14884 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 x_JFnc+TK(Bl4>fH6vvHTYMP>+M Date: Thu, 7 Feb 2013 12:19:54 -0500 Subject: [PATCH 6/8] * missing example module.. Oops. --- .../WebSocketEchoTest/WebSocketEchoModule.cs | 174 ++++++++++++++++++ 1 file changed, 174 insertions(+) create mode 100644 OpenSim/Region/OptionalModules/Example/WebSocketEchoTest/WebSocketEchoModule.cs diff --git a/OpenSim/Region/OptionalModules/Example/WebSocketEchoTest/WebSocketEchoModule.cs b/OpenSim/Region/OptionalModules/Example/WebSocketEchoTest/WebSocketEchoModule.cs new file mode 100644 index 0000000000..34e20b724a --- /dev/null +++ b/OpenSim/Region/OptionalModules/Example/WebSocketEchoTest/WebSocketEchoModule.cs @@ -0,0 +1,174 @@ +/* + * 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.Reflection; +using OpenSim.Framework.Servers; +using Mono.Addins; +using log4net; +using Nini.Config; +using OpenSim.Region.Framework.Interfaces; +using OpenSim.Region.Framework.Scenes; + +using OpenSim.Framework.Servers.HttpServer; + + +namespace OpenSim.Region.OptionalModules.WebSocketEchoModule +{ + + [Extension(Path = "/OpenSim/RegionModules", NodeName = "RegionModule", Id = "WebSocketEchoModule")] + public class WebSocketEchoModule : ISharedRegionModule + { + private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); + private bool enabled; + public string Name { get { return "WebSocketEchoModule"; } } + + public Type ReplaceableInterface { get { return null; } } + + + private HashSet _activeHandlers = new HashSet(); + + public void Initialise(IConfigSource pConfig) + { + enabled = true;// (pConfig.Configs["WebSocketEcho"] != null); + if (enabled) + m_log.DebugFormat("[WebSocketEchoModule]: INITIALIZED MODULE"); + } + + /// + /// This method sets up the callback to WebSocketHandlerCallback below when a HTTPRequest comes in for /echo + /// + public void PostInitialise() + { + if (enabled) + MainServer.Instance.AddWebSocketHandler("/echo", WebSocketHandlerCallback); + } + + // This gets called by BaseHttpServer and gives us an opportunity to set things on the WebSocket handler before we turn it on + public void WebSocketHandlerCallback(string path, WebSocketHttpServerHandler handler) + { + SubscribeToEvents(handler); + handler.SetChunksize(8192); + handler.NoDelay_TCP_Nagle = true; + handler.HandshakeAndUpgrade(); + } + + //These are our normal events + public void SubscribeToEvents(WebSocketHttpServerHandler handler) + { + handler.OnClose += HandlerOnOnClose; + handler.OnText += HandlerOnOnText; + handler.OnUpgradeCompleted += HandlerOnOnUpgradeCompleted; + handler.OnData += HandlerOnOnData; + handler.OnPong += HandlerOnOnPong; + } + + public void UnSubscribeToEvents(WebSocketHttpServerHandler handler) + { + handler.OnClose -= HandlerOnOnClose; + handler.OnText -= HandlerOnOnText; + handler.OnUpgradeCompleted -= HandlerOnOnUpgradeCompleted; + handler.OnData -= HandlerOnOnData; + handler.OnPong -= HandlerOnOnPong; + } + + private void HandlerOnOnPong(object sender, PongEventArgs pongdata) + { + m_log.Info("[WebSocketEchoModule]: Got a pong.. ping time: " + pongdata.PingResponseMS); + } + + private void HandlerOnOnData(object sender, WebsocketDataEventArgs data) + { + WebSocketHttpServerHandler obj = sender as WebSocketHttpServerHandler; + obj.SendData(data.Data); + m_log.Info("[WebSocketEchoModule]: We received a bunch of ugly non-printable bytes"); + obj.SendPingCheck(); + } + + + private void HandlerOnOnUpgradeCompleted(object sender, UpgradeCompletedEventArgs completeddata) + { + WebSocketHttpServerHandler obj = sender as WebSocketHttpServerHandler; + _activeHandlers.Add(obj); + } + + private void HandlerOnOnText(object sender, WebsocketTextEventArgs text) + { + WebSocketHttpServerHandler obj = sender as WebSocketHttpServerHandler; + obj.SendMessage(text.Data); + m_log.Info("[WebSocketEchoModule]: We received this: " + text.Data); + } + + // Remove the references to our handler + private void HandlerOnOnClose(object sender, CloseEventArgs closedata) + { + WebSocketHttpServerHandler obj = sender as WebSocketHttpServerHandler; + UnSubscribeToEvents(obj); + + lock (_activeHandlers) + _activeHandlers.Remove(obj); + obj.Dispose(); + } + + // Shutting down.. so shut down all sockets. + // Note.. this should be done outside of an ienumerable if you're also hook to the close event. + public void Close() + { + if (!enabled) + return; + + // We convert this to a for loop so we're not in in an IEnumerable when the close + //call triggers an event which then removes item from _activeHandlers that we're enumerating + WebSocketHttpServerHandler[] items = new WebSocketHttpServerHandler[_activeHandlers.Count]; + _activeHandlers.CopyTo(items); + + for (int i = 0; i < items.Length; i++) + { + items[i].Close(string.Empty); + items[i].Dispose(); + } + _activeHandlers.Clear(); + MainServer.Instance.RemoveWebSocketHandler("/echo"); + } + + public void AddRegion(Scene scene) + { + m_log.DebugFormat("[WebSocketEchoModule]: REGION {0} ADDED", scene.RegionInfo.RegionName); + } + + public void RemoveRegion(Scene scene) + { + m_log.DebugFormat("[WebSocketEchoModule]: REGION {0} REMOVED", scene.RegionInfo.RegionName); + } + + public void RegionLoaded(Scene scene) + { + m_log.DebugFormat("[WebSocketEchoModule]: REGION {0} LOADED", scene.RegionInfo.RegionName); + } + } +} \ No newline at end of file From a5c83f7505bf897c2d445391802f1ac7a2143d3d Mon Sep 17 00:00:00 2001 From: teravus Date: Thu, 7 Feb 2013 12:22:03 -0500 Subject: [PATCH 7/8] Websocket Echo module should not be on by default. --- .../Example/WebSocketEchoTest/WebSocketEchoModule.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/OpenSim/Region/OptionalModules/Example/WebSocketEchoTest/WebSocketEchoModule.cs b/OpenSim/Region/OptionalModules/Example/WebSocketEchoTest/WebSocketEchoModule.cs index 34e20b724a..112ba4e62f 100644 --- a/OpenSim/Region/OptionalModules/Example/WebSocketEchoTest/WebSocketEchoModule.cs +++ b/OpenSim/Region/OptionalModules/Example/WebSocketEchoTest/WebSocketEchoModule.cs @@ -55,7 +55,7 @@ namespace OpenSim.Region.OptionalModules.WebSocketEchoModule public void Initialise(IConfigSource pConfig) { - enabled = true;// (pConfig.Configs["WebSocketEcho"] != null); + enabled =(pConfig.Configs["WebSocketEcho"] != null); if (enabled) m_log.DebugFormat("[WebSocketEchoModule]: INITIALIZED MODULE"); } From af73ea909cad78eee78bd4e9d9e3a42cf8856263 Mon Sep 17 00:00:00 2001 From: Robert Adams Date: Wed, 6 Feb 2013 22:34:03 -0800 Subject: [PATCH 8/8] Change passed PhysicsParameter value from float to the more general string value --- .../PhysicsParameters/PhysicsParameters.cs | 19 +++------ .../Region/Physics/BulletSPlugin/BSParam.cs | 34 ++++++++------- .../Region/Physics/BulletSPlugin/BSScene.cs | 41 +++++++++++++++---- .../Physics/Manager/IPhysicsParameters.cs | 6 +-- 4 files changed, 57 insertions(+), 43 deletions(-) diff --git a/OpenSim/Region/OptionalModules/PhysicsParameters/PhysicsParameters.cs b/OpenSim/Region/OptionalModules/PhysicsParameters/PhysicsParameters.cs index 40f7fbc59a..3083a337ff 100755 --- a/OpenSim/Region/OptionalModules/PhysicsParameters/PhysicsParameters.cs +++ b/OpenSim/Region/OptionalModules/PhysicsParameters/PhysicsParameters.cs @@ -146,7 +146,7 @@ namespace OpenSim.Region.OptionalModules.PhysicsParameters { foreach (PhysParameterEntry ppe in physScene.GetParameterList()) { - float val = 0.0f; + string val = string.Empty; if (physScene.GetPhysicsParameter(ppe.name, out val)) { WriteOut(" {0}/{1} = {2}", scene.RegionInfo.RegionName, ppe.name, val); @@ -159,7 +159,7 @@ namespace OpenSim.Region.OptionalModules.PhysicsParameters } else { - float val = 0.0f; + string val = string.Empty; if (physScene.GetPhysicsParameter(parm, out val)) { WriteOut(" {0}/{1} = {2}", scene.RegionInfo.RegionName, parm, val); @@ -185,21 +185,12 @@ namespace OpenSim.Region.OptionalModules.PhysicsParameters return; } string parm = "xxx"; - float val = 0f; + string valparm = String.Empty; uint localID = (uint)PhysParameterEntry.APPLY_TO_NONE; // set default value try { parm = cmdparms[2]; - string valparm = cmdparms[3].ToLower(); - if (valparm == "true") - val = PhysParameterEntry.NUMERIC_TRUE; - else - { - if (valparm == "false") - val = PhysParameterEntry.NUMERIC_FALSE; - else - val = float.Parse(valparm, Culture.NumberFormatInfo); - } + valparm = cmdparms[3].ToLower(); if (cmdparms.Length > 4) { if (cmdparms[4].ToLower() == "all") @@ -224,7 +215,7 @@ namespace OpenSim.Region.OptionalModules.PhysicsParameters IPhysicsParameters physScene = scene.PhysicsScene as IPhysicsParameters; if (physScene != null) { - if (!physScene.SetPhysicsParameter(parm, val, localID)) + if (!physScene.SetPhysicsParameter(parm, valparm, localID)) { WriteError("Failed set of parameter '{0}' for region '{1}'", parm, scene.RegionInfo.RegionName); } diff --git a/OpenSim/Region/Physics/BulletSPlugin/BSParam.cs b/OpenSim/Region/Physics/BulletSPlugin/BSParam.cs index 965c3823b8..601c78c203 100755 --- a/OpenSim/Region/Physics/BulletSPlugin/BSParam.cs +++ b/OpenSim/Region/Physics/BulletSPlugin/BSParam.cs @@ -641,24 +641,6 @@ public static class BSParam return (b == ConfigurationParameters.numericTrue ? true : false); } - private static void ResetBroadphasePoolTainted(BSScene pPhysScene, float v) - { - BSScene physScene = pPhysScene; - physScene.TaintedObject("BSParam.ResetBroadphasePoolTainted", delegate() - { - physScene.PE.ResetBroadphasePool(physScene.World); - }); - } - - private static void ResetConstraintSolverTainted(BSScene pPhysScene, float v) - { - BSScene physScene = pPhysScene; - physScene.TaintedObject("BSParam.ResetConstraintSolver", delegate() - { - physScene.PE.ResetConstraintSolver(physScene.World); - }); - } - // Search through the parameter definitions and return the matching // ParameterDefn structure. // Case does not matter as names are compared after converting to lower case. @@ -722,6 +704,22 @@ public static class BSParam } } + private static void ResetBroadphasePoolTainted(BSScene pPhysScene, float v) + { + BSScene physScene = pPhysScene; + physScene.TaintedObject("BSParam.ResetBroadphasePoolTainted", delegate() + { + physScene.PE.ResetBroadphasePool(physScene.World); + }); + } + private static void ResetConstraintSolverTainted(BSScene pPhysScene, float v) + { + BSScene physScene = pPhysScene; + physScene.TaintedObject("BSParam.ResetConstraintSolver", delegate() + { + physScene.PE.ResetConstraintSolver(physScene.World); + }); + } } } diff --git a/OpenSim/Region/Physics/BulletSPlugin/BSScene.cs b/OpenSim/Region/Physics/BulletSPlugin/BSScene.cs index 6cd72f221c..f8a0c1e6ba 100644 --- a/OpenSim/Region/Physics/BulletSPlugin/BSScene.cs +++ b/OpenSim/Region/Physics/BulletSPlugin/BSScene.cs @@ -876,14 +876,39 @@ public sealed class BSScene : PhysicsScene, IPhysicsParameters // will use the next time since it's pinned and shared memory. // Some of the values require calling into the physics engine to get the new // value activated ('terrainFriction' for instance). - public bool SetPhysicsParameter(string parm, float val, uint localID) + public bool SetPhysicsParameter(string parm, string val, uint localID) { bool ret = false; + + float valf = 0f; + if (val.ToLower() == "true") + { + valf = PhysParameterEntry.NUMERIC_TRUE; + } + else + { + if (val.ToLower() == "false") + { + valf = PhysParameterEntry.NUMERIC_FALSE; + } + else + { + try + { + valf = float.Parse(val); + } + catch + { + valf = 0f; + } + } + } + BSParam.ParameterDefn theParam; if (BSParam.TryGetParameter(parm, out theParam)) { // Set the value in the C# code - theParam.setter(this, parm, localID, val); + theParam.setter(this, parm, localID, valf); // Optionally set the parameter in the unmanaged code if (theParam.onObject != null) @@ -898,16 +923,16 @@ public sealed class BSScene : PhysicsScene, IPhysicsParameters case PhysParameterEntry.APPLY_TO_NONE: // This will cause a call into the physical world if some operation is specified (SetOnObject). objectIDs.Add(TERRAIN_ID); - TaintedUpdateParameter(parm, objectIDs, val); + TaintedUpdateParameter(parm, objectIDs, valf); break; case PhysParameterEntry.APPLY_TO_ALL: lock (PhysObjects) objectIDs = new List(PhysObjects.Keys); - TaintedUpdateParameter(parm, objectIDs, val); + TaintedUpdateParameter(parm, objectIDs, valf); break; default: // setting only one localID objectIDs.Add(localID); - TaintedUpdateParameter(parm, objectIDs, val); + TaintedUpdateParameter(parm, objectIDs, valf); break; } } @@ -942,14 +967,14 @@ public sealed class BSScene : PhysicsScene, IPhysicsParameters // Get parameter. // Return 'false' if not able to get the parameter. - public bool GetPhysicsParameter(string parm, out float value) + public bool GetPhysicsParameter(string parm, out string value) { - float val = 0f; + string val = String.Empty; bool ret = false; BSParam.ParameterDefn theParam; if (BSParam.TryGetParameter(parm, out theParam)) { - val = theParam.getter(this); + val = theParam.getter(this).ToString(); ret = true; } value = val; diff --git a/OpenSim/Region/Physics/Manager/IPhysicsParameters.cs b/OpenSim/Region/Physics/Manager/IPhysicsParameters.cs index b8676bac9b..31a397c7d4 100755 --- a/OpenSim/Region/Physics/Manager/IPhysicsParameters.cs +++ b/OpenSim/Region/Physics/Manager/IPhysicsParameters.cs @@ -60,14 +60,14 @@ namespace OpenSim.Region.Physics.Manager // Set parameter on a specific or all instances. // Return 'false' if not able to set the parameter. - bool SetPhysicsParameter(string parm, float value, uint localID); + bool SetPhysicsParameter(string parm, string value, uint localID); // Get parameter. // Return 'false' if not able to get the parameter. - bool GetPhysicsParameter(string parm, out float value); + bool GetPhysicsParameter(string parm, out string value); // Get parameter from a particular object // TODO: - // bool GetPhysicsParameter(string parm, out float value, uint localID); + // bool GetPhysicsParameter(string parm, out string value, uint localID); } }