30

我目前正在使用几个委托处理程序(派生自 的类DelegatingHandler)在请求发送之前对其进行处理,例如验证签名等。这一切都非常好,因为我不必在所有调用上重复签名验证(例如)。

我想对来自同一个 Web 请求的响应使用相同的原则。响应中是否有类似于 DelegatingHandler 的东西?一种在响应返回到方法之前捕获响应的方法,在某种程度上?

附加信息:我正在使用HttpClient.PutAsync(...)

4

2 回答 2

50

是的。您可以在后续任务中执行此操作。

我在这里解释一下。

例如,这段代码(来自上面的博客)跟踪请求 URI 并在响应中添加一个虚拟标头。

public class DummyHandler : DelegatingHandler
{
    protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
    {
        // work on the request 
       Trace.WriteLine(request.RequestUri.ToString());

       var response = await base.SendAsync(request, cancellationToken);
       response.Headers.Add("X-Dummy-Header", Guid.NewGuid().ToString());
       return response;
    }
}
于 2012-08-15T14:01:21.213 回答
21

这是拦截请求和响应的示例。重写的方法 SendAsync 用于捕获原始请求,而名为 ResponseHandler 的方法用于捕获响应。

捕获原始请求和响应的示例

using System.Net.Http;
using System.Threading.Tasks;
namespace webAPI_Test
{
    public class MessageInterceptor : DelegatingHandler
    {
        protected override System.Threading.Tasks.Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, System.Threading.CancellationToken cancellationToken)
        {
            // CATCH THE REQUEST BEFORE SENDING TO THE ROUTING HANDLER
            var headers = request.ToString();
            var body = request.Content.ReadAsStringAsync().Result;
            var fullRequest = headers + "\n" + body;

            // SETUP A CALLBACK FOR CATCHING THE RESPONSE - AFTER ROUTING HANDLER, AND AFTER CONTROLLER ACTIVITY
            return base.SendAsync(request, cancellationToken).ContinueWith(
                        task =>
                        {
                            // GET THE COPY OF THE TASK, AND PASS TO A CUSTOM ROUTINE
                            ResponseHandler(task);

                            // RETURN THE ORIGINAL RESULT
                            var response = task.Result;
                            return response;
                        }
            );
        }

        public void ResponseHandler(Task<HttpResponseMessage> task)
        {
            var headers = task.Result.ToString();
            var body = task.Result.Content.ReadAsStringAsync().Result;

            var fullResponse = headers + "\n" + body;
        }
    }
}

要使用此方法,需要将类标识并注册为 MessageHandler。我将以下行添加到我的 Global.asax 文件中...

示例如何注册新的 MessageInterceptor 类

GlobalConfiguration.Configuration.MessageHandlers.Add(new MessageInterceptor());

这是我完整的 Global.asax 文件。注意 MessageInterceptor 是如何被引用的......

显示 MessageInterceptor 集成的 Global.asax 完整版

using System.Web.Http;
using System.Web.Mvc;
using System.Web.Optimization;
using System.Web.Routing;
namespace webAPI_Test
{
    // Note: For instructions on enabling IIS6 or IIS7 classic mode, 
    // visit http://go.microsoft.com/?LinkId=9394801

    public class WebApiApplication : System.Web.HttpApplication
    {
        protected void Application_Start()
        {
            AreaRegistration.RegisterAllAreas();

            WebApiConfig.Register(GlobalConfiguration.Configuration);
            FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
            RouteConfig.RegisterRoutes(RouteTable.Routes);
            BundleConfig.RegisterBundles(BundleTable.Bundles);

            GlobalConfiguration.Configuration.MessageHandlers.Add(new MessageInterceptor());
        }
    }
}
于 2013-07-29T01:23:05.677 回答