如何使用 TcpClient.ConnectAsync() 的代理?

How to use Proxy with TcpClient.ConnectAsync()?

.NET 中的 HTTP 代理支持实际上并不支持较低级别 类,如 TcpClient 或 Socket。但是我想通过支持 'CONNECT' 命令的 HTTP 代理连接一个 TCPServer (ip, port)。

所以我需要执行以下步骤:

  1. 连接到代理。
  2. 发送CONNECT Host:Port HTTP/1.1<CR><LF>
  3. 发送<CR><LF>
  4. 等待一行回复。如果包含HTTP/1.X 200,则连接成功。
  5. 阅读更多响应行,直到收到空行。
  6. 它通过代理与外界相连。与代理进行任何可能的数据交换。

实际上我是在没有代理的情况下这样做的

    TcpClient _client;
    NetworkStream _stream;

    public static async Task<bool> ConnectAsync(string hostname, int port)
    {
        _client = new TcpClient();
        await _client.ConnectAsync(hostname, port).ConfigureAwait(false);
        _stream = conn._client.GetStream();

        ..... Do some stuff

        // Connexion OK
        return true;
    }

如何在连接 TcpClient 之前使用代理和凭据?

我根据.NET: Connecting a TcpClient through an HTTP proxy with authentication and Bypass the proxy using TcpClient

找到了解决方案
TcpClient _client;
NetworkStream _stream;

public TcpClient ProxyTcpClient(string targetHost, int targetPort, string httpProxyHost, int httpProxyPort, string proxyUserName, string proxyPassword)
{
        const BindingFlags Flags = BindingFlags.NonPublic | BindingFlags.Instance;
        Uri proxyUri = new UriBuilder
        {
            Scheme = Uri.UriSchemeHttp,
            Host = httpProxyHost,
            Port = httpProxyPort
        }.Uri;
        Uri targetUri = new UriBuilder
        {
             Scheme = Uri.UriSchemeHttp,
             Host = targetHost,
             Port = targetPort
        }.Uri;

        WebProxy webProxy = new WebProxy(proxyUri, true);
        webProxy.Credentials = new NetworkCredential(proxyUserName, proxyPassword);
        WebRequest request = WebRequest.Create(targetUri);
        request.Proxy = webProxy;
        request.Method = "CONNECT";
        HttpWebResponse response = (HttpWebResponse)request.GetResponse();
        Stream responseStream = response.GetResponseStream();
        Type responseType = responseStream.GetType();
        PropertyInfo connectionProperty = responseType.GetProperty("Connection", Flags);
        var connection = connectionProperty.GetValue(responseStream, null);
        Type connectionType = connection.GetType();
        PropertyInfo networkStreamProperty = connectionType.GetProperty("NetworkStream", Flags);
        NetworkStream networkStream = (NetworkStream)networkStreamProperty.GetValue(connection, null);
        Type nsType = networkStream.GetType();
        PropertyInfo socketProperty = nsType.GetProperty("Socket", Flags);
        Socket socket = (Socket)socketProperty.GetValue(networkStream, null);

        return new TcpClient { Client = socket };
}

public static async Task<bool> ConnectAsync(string hostname, int port)
{
        _client = ProxyTcpClient("IPTargetHost", 1234, "IPProxyHost", 5678, "Userproxy", "Userppwd");
        _stream = conn._client.GetStream();

        ..... Do some stuff

        // Connexion OK
        return true;
}

我们已经设法使用 .Net 的 Socket 实现了它。 Nuget 包名为 Filemail.ProxiedTcpClient。代码非常简单:

public static TcpClient CreateProxied(Uri proxy, Uri destination)
{
    var socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

    socket.Connect(proxy.Host, proxy.Port);

    var connectMessage = Encoding.UTF8.GetBytes($"CONNECT {destination.Host}:{destination.Port} HTTP/1.1{Environment.NewLine}{Environment.NewLine}");
    socket.Send(connectMessage);

    byte[] receiveBuffer = new byte[1024];
    var received = socket.Receive(receiveBuffer);

    var response = ASCIIEncoding.ASCII.GetString(receiveBuffer, 0, received);

    if (!response.Contains("200 OK"))
    {
        throw new Exception($"Error connecting to proxy server {destination.Host}:{destination.Port}. Response: {response}");
    }

    return new TcpClient
    {
        Client = socket
    };
}

在这里投稿:https://github.com/filemail/ProxiedTcpClient

如果代理需要基本身份验证,要扩展 Rychu's 答案,请将 header 添加到连接消息中

var auth = Convert.ToBase64String(Encoding.UTF8.GetBytes($"{proxy.Login}:{proxy.Password}"));
var connectMessage = Encoding.UTF8.GetBytes($"CONNECT {host}:{port} HTTP/1.1\nProxy-Authorization: Basic {auth}\n\n");
socket.Send(connectMessage);