我最近使用Titanium-Web-Proxy编写了一个反向代理。
浏览器通过IP 127.0.0.1、80端口访问反向代理,反向代理将浏览器的请求转发到IP 127.0.0.1、2366端口的IIS服务器。
+---------+ Request +---------------+ +------------+
| +-------------> | | Request | |
| | | Reverse Proxy +---------------> | Web Server |
| | | | | |
| Browser | | 127.0.0.1 | | 127.0.0.1 |
| | | | Response | |
| | Response | 80 | <---------------+ 2366 |
| | <-------------+ | | |
+---------+ +---------------+ +------------+
当我测试它时,反向代理没有像我预期的那样工作,浏览器返回一个 HTTP 400 错误。
Bad Request - Invalid Hostname
HTTP Error 400. The request hostname is invalid.
我尝试重写127.0.0.1:2366
为localhost:2366
并重新测试,错误是一样的。
后来我尝试更改modifiedUri
为http://example.com但收到 404 Not Found 错误。
var modifiedUri = new UriBuilder("http://example.com/");
这是我的代码,请不要介意我糟糕的编码水平。
public class Startup
{
private readonly ProxyServer proxyServer;
private readonly IDictionary<Guid, HeaderCollection> requestHeaderHistory = new ConcurrentDictionary<Guid, HeaderCollection>();
private readonly IDictionary<Guid, HeaderCollection> responseHeaderHistory = new ConcurrentDictionary<Guid, HeaderCollection>();
public Startup()
{
proxyServer = new ProxyServer()
{
TrustRootCertificate = true,
ForwardToUpstreamGateway = true,
};
}
public void Start()
{
proxyServer.BeforeRequest += OnRequest;
proxyServer.BeforeResponse += OnResponse;
var transparentProxyEndPoint = new TransparentProxyEndPoint(IPAddress.Loopback, 80, false)
{
};
proxyServer.AddEndPoint(transparentProxyEndPoint);
proxyServer.Start();
}
public void Stop()
{
proxyServer.BeforeRequest -= OnRequest;
proxyServer.BeforeResponse -= OnResponse;
proxyServer.Stop();
}
public async Task OnRequest(object sender, SessionEventArgs e)
{
var requestUri = e.WebSession.Request.RequestUri;
var modifiedUri = new UriBuilder("http", "127.0.0.1", 2366, requestUri.AbsolutePath);
e.WebSession.Request.RequestUri = modifiedUri.Uri;
requestHeaderHistory[e.Id] = e.WebSession.Request.RequestHeaders;
if (e.WebSession.Request.HasBody)
{
var bodyBytes = await e.GetRequestBody();
await e.SetRequestBody(bodyBytes);
string bodyString = await e.GetRequestBodyAsString();
await e.SetRequestBodyString(bodyString);
}
}
public async Task OnResponse(object sender, SessionEventArgs e)
{
responseHeaderHistory[e.Id] = e.WebSession.Response.ResponseHeaders;
if (e.WebSession.Request.Method == "GET" || e.WebSession.Request.Method == "POST")
{
if (e.WebSession.Response.ResponseStatusCode == (int)HttpStatusCode.OK)
{
if (e.WebSession.Response.ContentType != null && e.WebSession.Response.ContentType.Trim().ToLower().Contains("text/html"))
{
var bodyBytes = await e.GetResponseBody();
await e.SetResponseBody(bodyBytes);
string body = await e.GetResponseBodyAsString();
await e.SetResponseBodyString(body);
}
}
}
}
}