Files
Tor/TorWebClient/TorClient/Proxy/Connection/Connection.cs

367 lines
12 KiB
C#

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Net.Sockets;
using System.IO;
using System.Threading;
using MarketData;
using Tor.Extensions;
using System.Diagnostics;
namespace Tor.Proxy
{
/// <summary>
/// A class containing a reference to a connection proxy client.
/// </summary>
internal sealed class Connection : IDisposable
{
private readonly Client client;
private volatile bool disposed;
private Profiler profiler = new Profiler();
private ConnectionDisposedCallback disposedCallback;
private Dictionary<string, string> headers;
private string host;
private string http;
private string method;
private int port;
private byte[] post;
private Socket socket;
private Connection()
{
}
/// <summary>
/// Initializes a new instance of the <see cref="Connection"/> class.
/// </summary>
/// <param name="client">The client hosting the proxy .</param>
/// <param name="socket">The socket belonging to the connection.</param>
/// <param name="disposeCallback">A callback method raised when the connection is disposed.</param>
public Connection(Client client, Socket socket, ConnectionDisposedCallback disposeCallback)
{
this.client = client;
this.disposed = false;
this.disposedCallback = disposeCallback;
this.headers = null;
this.host = null;
this.http = "HTTP/1.1";
this.method = null;
this.port = 80;
this.post = null;
this.socket = socket;
this.GetHeaderData();
}
#region System.IDisposable
public bool IsSocketClosed()
{
if(null==socket)return true;
return socket.IsConnected()?false:true;
}
/// <summary>
/// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources.
/// </summary>
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
/// <summary>
/// Releases unmanaged and - optionally - managed resources.
/// </summary>
/// <param name="disposing"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
private void Dispose(bool disposing)
{
if (disposed)
return;
if (disposing)
{
if (socket != null)
{
try
{
socket.Shutdown(SocketShutdown.Both);
}
catch { }
socket.Dispose();
socket = null;
}
disposed = true;
if (disposedCallback != null)
disposedCallback(this);
}
}
public Profiler Profiler
{
get{return profiler;}
}
public bool IsDisposed()
{
return disposed;
}
#endregion
#region Properties
/// <summary>
/// Gets the header values provided with the request.
/// </summary>
public Dictionary<string, string> Headers
{
get { return headers; }
}
/// <summary>
/// Gets the target host of the HTTP request.
/// </summary>
public string Host
{
get { return host; }
}
/// <summary>
/// Gets the HTTP version sent with the request.
/// </summary>
public string HTTP
{
get { return http; }
}
/// <summary>
/// Gets the method requested for the HTTP request (GET, POST, PUT, DELETE, CONNECT).
/// </summary>
public string Method
{
get { return method; }
}
/// <summary>
/// Gets the target port number of the HTTP request.
/// </summary>
public int Port
{
get { return port; }
}
/// <summary>
/// Gets the POST data.
/// </summary>
public byte[] Post
{
get { return post; }
}
/// <summary>
/// Gets the socket connected to the proxy client.
/// </summary>
public Socket Socket
{
get { return socket; }
}
#endregion
/// <summary>
/// Gets the header block which was dispatched with the original socket request.
/// </summary>
/// <returns>A <see cref="System.String"/> containing the header data.</returns>
public string GetHeader()
{
StringBuilder header = new StringBuilder();
header.Append(method);
header.Append("\r\n");
foreach (KeyValuePair<string, string> value in headers)
{
if (value.Key.StartsWith("proxy", StringComparison.CurrentCultureIgnoreCase))
continue;
header.AppendFormat("{0}: {1}\r\n", value.Key, value.Value);
}
return header.Append("\r\n").ToString();
}
/// <summary>
/// Process the connection request by reading the header information from the HTTP request, and connecting to the tor server
/// and dispatching the request to the relevant host.
/// </summary>
private void GetHeaderData()
{
try
{
StringBuilder builder = new StringBuilder();
using (StreamReader reader = new StreamReader(new NetworkStream(socket, false)))
{
for (string line = reader.ReadLine(); line != null; line = reader.ReadLine())
{
builder.Append(line);
builder.Append("\r\n");
if (line.Trim().Length == 0)
break;
}
}
using (StringReader reader = new StringReader(builder.ToString()))
{
method = reader.ReadLine();
if (method == null)
{
throw new InvalidOperationException("The proxy connection did not supply a valid HTTP header");
}
headers = new Dictionary<string, string>();
for (string line = reader.ReadLine(); line != null; line = reader.ReadLine())
{
string trimmed = line.Trim();
if (trimmed.Length == 0)
break;
string[] parts = trimmed.Split(new[] { ':' }, 2);
if (parts.Length == 1)
continue;
headers[parts[0].Trim()] = parts[1].Trim();
}
if (method.StartsWith("POST", StringComparison.CurrentCultureIgnoreCase))
{
if (!headers.ContainsKey("Content-Length"))
{
throw new InvalidOperationException("The proxy connection is a POST method but contains no content length");
}
long contentLength = long.Parse(headers["Content-Length"]);
using (MemoryStream memory = new MemoryStream())
{
long read = 0;
byte[] buffer = new byte[512];
while (contentLength > read)
{
int received = socket.Receive(buffer, 0, buffer.Length, SocketFlags.None);
if (received <= 0)
throw new InvalidOperationException("The proxy connection was terminated while reading POST data");
memory.Write(buffer, 0, received);
read += received;
}
post = memory.ToArray();
}
}
if (method.StartsWith("CONNECT", StringComparison.CurrentCultureIgnoreCase))
{
string[] connectTargets = method.Split(' ');
if (connectTargets.Length < 3)
throw new InvalidOperationException("The proxy connection supplied a CONNECT command with insufficient parameters");
http = connectTargets[2];
string connectTarget = connectTargets[1];
string[] connectParams = connectTarget.Split(':');
if (connectParams.Length == 2)
{
host = connectParams[0];
port = int.Parse(connectParams[1]);
}
else
{
host = connectParams[0];
port = 443;
}
}
else
{
if (!headers.ContainsKey("Host"))
throw new InvalidOperationException("The proxy connection did not supply a connection host");
string connectTarget = headers["Host"];
string[] connectParams = connectTarget.Split(':');
if (connectParams.Length == 1)
host = connectParams[0];
else
{
host = connectParams[0];
port = int.Parse(connectParams[1]);
}
}
if(null!=host)
{
MDTrace.WriteLine(LogLevel.DEBUG,String.Format("Host: {0}:{1}",host,port));
}
return;
}
}
catch (Exception)
{
headers = new Dictionary<string, string>();
return;
// throw new TorException("The proxy connection failed to process", exception);
}
}
/// <summary>
/// Writes a buffer of data to the connected client socket.
/// </summary>
/// <param name="data">The data to send to the client socket.</param>
public void Write(string data)
{
Write(Encoding.ASCII.GetBytes(data));
}
/// <summary>
/// Writes a buffer of data to the connected client socket.
/// </summary>
/// <param name="data">The data to send to the client socket.</param>
/// <param name="parameters">An optional list of parameters to format into the data.</param>
public void Write(string data, params object[] parameters)
{
data = string.Format(data, parameters);
Write(Encoding.ASCII.GetBytes(data));
}
/// <summary>
/// Writes a buffer of data to the connected client socket.
/// </summary>
/// <param name="buffer">The data to send to the client socket.</param>
public void Write(byte[] buffer)
{
socket.Send(buffer, 0, buffer.Length, SocketFlags.None);
}
}
/// <summary>
/// A delegate event handler representing a method raised when a connection is disposed.
/// </summary>
/// <param name="connection">The connection which was disposed.</param>
internal delegate void ConnectionDisposedCallback(Connection connection);
}