HttpClient永远不会在IIS上发送请求

phu*_*uzi 0 c# asp.net iis reverse-proxy asp.net-web-api

我有一个奇怪的问题,并尝试各种各样的事情来实现这一点.

我在我的Web API项目中有一个反向代理委托处理程序,它连接到内部资源,文件等的拦截请求,从我们的外部站点到我们DMZ内部的内部站点......

using System;
using System.Configuration;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Net.Http.Formatting;
using System.Threading;
using System.Threading.Tasks;
using System.Web.Http;

namespace Resources.API
{
    public class ProxyHandler : DelegatingHandler
    {
        protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
        {
            var routes = new[]{
                "/api/videos",
                "/api/documents"
            };

            // check whether we need to proxy this request
            var passThrough = !routes.Any(route => request.RequestUri.LocalPath.StartsWith(route));
            if (passThrough)
                return await base.SendAsync(request, cancellationToken);

            // got a hit forward the request to the proxy Web API
            return await ForwardRequest(request, cancellationToken);
        }

        private static async Task<HttpResponseMessage> ForwardRequest(HttpRequestMessage request, CancellationToken cancellationToken)
        {
            //Clone the request and forward to the internal proxy site
            var proxyUrl = ConfigurationManager.AppSettings["ProxyUrl"];
            var baseUri = new UriBuilder(proxyUrl);

            //clone the requestUri and point it at the proxy site
            var forwardedUri = new UriBuilder(request.RequestUri)
            {
                Scheme = baseUri.Scheme,
                Host = baseUri.Host,
                Port = baseUri.Port
            };

            var forwardRequest = new HttpRequestMessage(request.Method, forwardedUri.Uri);

            if (request.Method == HttpMethod.Post || request.Method == HttpMethod.Put)
            {
                var stream = new MemoryStream();
                await request.Content.CopyToAsync(stream);
                stream.Seek(0, SeekOrigin.Begin);
                forwardRequest.Content = new StreamContent(stream);

                //copy the content headers
                foreach (var header in request.Content.Headers)
                {
                    forwardRequest.Content.Headers.TryAddWithoutValidation(header.Key, header.Value);
                }
            };

            forwardRequest.Version = request.Version;

            foreach (var prop in request.Properties)
            {
                forwardRequest.Properties.Add(prop);
            }

            foreach (var header in request.Headers)
            {
                forwardRequest.Headers.TryAddWithoutValidation(header.Key, header.Value);
            }

            var client = new HttpClient(new HttpClientHandler(), disposeHandler: false);
            var task = await Task.Factory
               .StartNew(async () => await client.SendAsync(forwardRequest, HttpCompletionOption.ResponseHeadersRead,
                   cancellationToken),
                   CancellationToken.None,
                   TaskCreationOptions.LongRunning,
                   TaskScheduler.Default);
            try
            {
                task.Wait(cancellationToken);
            }
            catch (Exception e)
            {
                return new HttpResponseMessage(HttpStatusCode.InternalServerError)
                {
                    Content =
                        new ObjectContent<HttpError>(new HttpError(e, includeErrorDetail: true),
                            new JsonMediaTypeFormatter())
                };
            }

            return task.Result;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

编辑:还试过等待并返回任务...

            try
            {
                return await task;
            }
Run Code Online (Sandbox Code Playgroud)

这适用于IIS Express 8.0,但不适用于Windows 7 Professional(我的开发机器)上的IIS 7.5或Windows Server 2012上的IIS 8.0.

创建者HttpClient永远不会通过网络实际发送请求(由Fiddler检查)并最终超时并抛出AggregateException一个孩子TaskCanceledException.

设置一个断点task.Wait,我注意到,由于某种原因,断点被点击10次,而不是一次通过IIS Express运行.

我已经尝试了各种各样的东西尝试让它工作,包括搜索谷歌和SO,但似乎没有任何工作.

谁知道为什么会这样?或者可以解释我做错了什么?

phu*_*uzi 7

弄清楚了.必须更改Host请求的标头才能正确发送.它基本上忽略了RequestUri并使用Host标头来决定实际发送请求的位置.

forwardRequest.Headers.Host = forwardRequest.RequestUri.Host;
Run Code Online (Sandbox Code Playgroud)

现在就像一个魅力,IIS现在将适当地发送请求.仍然让我想知道为什么IIS Express似乎不需要Host更改标题!

完整的代码......增加了X-Forwarded-ForX-Forwarded-Host,以及良好的措施.

using System;
using System.Configuration;
using System.Diagnostics;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Net.Http.Formatting;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Web;
using System.Web.Http;

namespace Resources.API
{
    public class ProxyHandler : DelegatingHandler
    {
        protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
        {
            var routes = new[]{
                "/api/videos",
                "/api/documents"
            };

            // check whether we need to proxy this request
            var passThrough = !routes.Any(route => request.RequestUri.LocalPath.StartsWith(route));
            if (passThrough)
                return await base.SendAsync(request, cancellationToken);

            // got a hit forward the request to the proxy Web API
            //return GetResponseFromProxy(request);

            //Nicer method using HttpClient - but it doesn't work on IIS!
            return await ForwardRequest(request, cancellationToken);
        }

        private static async Task<HttpResponseMessage> ForwardRequest(HttpRequestMessage request, CancellationToken cancellationToken)
        {
            //Clone the request and forward to the internal proxy site
            var proxyUrl = ConfigurationManager.AppSettings["ProxyUrl"];
            var baseUri = new UriBuilder(proxyUrl);

            //clone the requestUri and point it at the proxy site
            var forwardedUri = new UriBuilder(request.RequestUri)
            {
                Scheme = baseUri.Scheme,
                Host = baseUri.Host,
                Port = baseUri.Port
            };

            var forwardRequest = new HttpRequestMessage(request.Method, forwardedUri.Uri);

            if (request.Method == HttpMethod.Post || request.Method == HttpMethod.Put)
            {
                var stream = new MemoryStream();
                await request.Content.CopyToAsync(stream);
                stream.Seek(0, SeekOrigin.Begin);
                forwardRequest.Content = new StreamContent(stream);

                //copy the content headers
                foreach (var header in request.Content.Headers)
                {
                    forwardRequest.Content.Headers.TryAddWithoutValidation(header.Key, header.Value);
                }
            };

            forwardRequest.Version = request.Version;

            foreach (var prop in request.Properties)
            {
                forwardRequest.Properties.Add(prop);
            }

            foreach (var header in request.Headers)
            {
                forwardRequest.Headers.TryAddWithoutValidation(header.Key, header.Value);
            }

            // Don't forget to change the Host header to refer to the proxy
            forwardRequest.Headers.Host = forwardRequest.RequestUri.Host;

            //Add the relevant X-Forwarded headers
            var xForwardedHost = request.Headers.Host;
            forwardRequest.Headers.Add("X-Forwarded-Host", xForwardedHost);

            var xForwardedFor = HttpContext.Current.Request.UserHostAddress;
            forwardRequest.Headers.Add("X-Forwarded-For", xForwardedFor);

            var client = new HttpClient(new HttpClientHandler(), disposeHandler: false);

            try
            {
                return await client.SendAsync(forwardRequest, HttpCompletionOption.ResponseHeadersRead,
                    cancellationToken);
            }
            catch (Exception e)
            {
                return new HttpResponseMessage(HttpStatusCode.InternalServerError)
                {
                    Content =
                        new ObjectContent<HttpError>(new HttpError(e, includeErrorDetail: true),
                            new JsonMediaTypeFormatter())
                };
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)