Introduction:

Building modern HTTP/RESTful/RPC
services has become very easy with the new ASP.NET Web API framework.
Using ASP.NET Web API framework, you can create HTTP services which can
be accessed from browsers, machines, mobile devices and other
clients. Developing HTTP services is now become more easy for ASP.NET
MVC developer becasue ASP.NET Web API is now included in ASP.NET MVC. In
addition to developing HTTP services, it is also important to return
meaningful response to client if a resource(uri) not found(HTTP 404) for
a reason(for example, mistyped resource uri). It is also important to
make this response centralized so you can configure all of 'HTTP 404 Not
Found' resource at one place. In this article, I will show you how
to handle 'HTTP 404 Not Found' at one place.

        Description:

Let's say that you are developing
a HTTP RESTful application using ASP.NET Web API framework. In this
application you need to handle HTTP 404 errors in a centralized
location. From ASP.NET Web API point of you, you need to handle these
situations,

  • No route matched.
  • Route is matched but no {controller} has been found on route.
  • No type with {controller} name has been found.
  • No matching action method found in the selected controller due to no
    action method start with the request HTTP method verb or no action
    method with IActionHttpMethodProviderRoute implemented attribute found
    or no method with {action} name found or no method with the matching
    {action} name found.

Now, let create a ErrorController with Handle404
action method. This action method will be used in all of the above
cases for sending HTTP 404 response message to the client.

01 public class ErrorController : ApiController
02 {
03     [HttpGet, HttpPost, HttpPut, HttpDelete, HttpHead, HttpOptions, AcceptVerbs("PATCH")]
04     public HttpResponseMessage Handle404()
05     {
06         var responseMessage = new HttpResponseMessage(HttpStatusCode.NotFound);
07         responseMessage.ReasonPhrase = "The requested resource is not found";
08         return responseMessage;
09     }
10 }

You can easily change the above
action method to send some other specific HTTP 404 error response. If a
client of your HTTP service send a request to a resource(uri) and no
route matched with this uri on server then you can route the request to
the above Handle404 method using a custom route. Put this route at the
very bottom of route configuration,

1 routes.MapHttpRoute(
2     name: "Error404",
3     routeTemplate: "{*url}",
4     defaults: new { controller = "Error", action = "Handle404" }
5 );

Now you need handle the case when there is no
{controller} in the matching route or when there is no type with
{controller} name found. You can easily handle this case and route the
request to the above Handle404 method using a custom
IHttpControllerSelector. Here is the definition of a custom
IHttpControllerSelector,

01 public class HttpNotFoundAwareDefaultHttpControllerSelector : DefaultHttpControllerSelector
02 {
03     public HttpNotFoundAwareDefaultHttpControllerSelector(HttpConfiguration configuration)
04         : base(configuration)
05     {
06     }
07     public override HttpControllerDescriptor SelectController(HttpRequestMessage request)
08     {
09         HttpControllerDescriptor decriptor = null;
10         try
11         {
12             decriptor = base.SelectController(request);
13         }
14         catch (HttpResponseException ex)
15         {
16             var code = ex.Response.StatusCode;
17             if (code != HttpStatusCode.NotFound)
18                 throw;
19             var routeValues = request.GetRouteData().Values;
20             routeValues["controller"] = "Error";
21             routeValues["action"] = "Handle404";
22             decriptor = base.SelectController(request);
23         }
24         return decriptor;
25     }
26 }

Next, it is also required to pass the request to
the above Handle404 method if no matching action method found in the
selected controller due to the reason discussed above. This situation
can also be easily handled through a custom IHttpActionSelector. Here is
the source of custom IHttpActionSelector,

01 public class HttpNotFoundAwareControllerActionSelector : ApiControllerActionSelector
02 {
03     public HttpNotFoundAwareControllerActionSelector()
04     {
05     }
06  
07     public override HttpActionDescriptor SelectAction(HttpControllerContext controllerContext)
08     {
09         HttpActionDescriptor decriptor = null;
10         try
11         {
12             decriptor = base.SelectAction(controllerContext);
13         }
14         catch (HttpResponseException ex)
15         {
16             var code = ex.Response.StatusCode;
17             if (code != HttpStatusCode.NotFound && code != HttpStatusCode.MethodNotAllowed)
18                 throw;
19             var routeData = controllerContext.RouteData;
20             routeData.Values["action"] = "Handle404";
21             IHttpController httpController = new ErrorController();
22             controllerContext.Controller = httpController;
23             controllerContext.ControllerDescriptor = new HttpControllerDescriptor(controllerContext.Configuration, "Error", httpController.GetType());
24             decriptor = base.SelectAction(controllerContext);
25         }
26         return decriptor;
27     }
28 }

Finally, we need to register the custom
IHttpControllerSelector and IHttpActionSelector. Open global.asax.cs
file and add these lines,

1 configuration.Services.Replace(typeof(IHttpControllerSelector), new HttpNotFoundAwareDefaultHttpControllerSelector(configuration));
2 configuration.Services.Replace(typeof(IHttpActionSelector), new HttpNotFoundAwareControllerActionSelector());

        Summary:

In addition to building an
application for HTTP services, it is also important to send
meaningful centralized information in response when something goes
wrong, for example 'HTTP 404 Not Found' error.  In this article, I
showed you how to handle 'HTTP 404 Not Found' error in a centralized
location. Hopefully you will enjoy this article too.

Handling HTTP 404 Error in ASP.NET Web API的更多相关文章

  1. Global Error Handling in ASP.NET Web API 2(webapi2 中的全局异常处理)

    目前,在Web API中没有简单的方法来记录或处理全局异常(webapi1中).一些未处理的异常可以通过exception filters进行处理,但是有许多情况exception filters无法 ...

  2. Exception Handling in ASP.NET Web API webapi异常处理

    原文:http://www.asp.net/web-api/overview/error-handling/exception-handling This article describes erro ...

  3. Exception Handling in ASP.NET Web API

    public static void RegisterGlobalFilters(GlobalFilterCollection filters) { filters.Add(new HandleErr ...

  4. Asp.net web api部署在某些服务器上老是404

    asp.net web api部署在Windows服务器上后,按照WebAPI定义的路由访问,老是出现404,但定义一个静态文件从站点访问,却又OK. 这时,便可以确定是WebAPI路由出了问题,经调 ...

  5. "Asp.Net Web Api MediaTypeFormatter Error for x-www-formurlencoded data" 解决方法

    遇到标题中所说的问题原因是使用 jQuery AJAX 以 POST 方式调用 Asp.Net Web API .解决办法请看以下代码中有注释的部分. public static class WebA ...

  6. 【ASP.NET Web API教程】4.3 ASP.NET Web API中的异常处理

    原文:[ASP.NET Web API教程]4.3 ASP.NET Web API中的异常处理 注:本文是[ASP.NET Web API系列教程]的一部分,如果您是第一次看本系列教程,请先看前面的内 ...

  7. Professional C# 6 and .NET Core 1.0 - Chapter 42 ASP.NET Web API

    本文内容为转载,重新排版以供学习研究.如有侵权,请联系作者删除. 转载请注明本文出处: -------------------------------------------------------- ...

  8. 初试ASP.NET Web API/MVC API(附Demo)

    写在前面 HTTP RESTful 创建Web API 调用Web API 运行截图及Demo下载 ASP.NET Web API是​​一个框架,可以很容易构建达成了广泛的HTTP服务客户端,包括浏览 ...

  9. ASP.NET Web API 异常日志记录

    如果在 ASP.NET MVC 应用程序中记录异常信息,我们只需要在 Global.asax 的 Application_Error 中添加代码就可以了,比如: public class MvcApp ...

随机推荐

  1. 淘宝(taobao)HSF框架

    一.背景 随着网站访问量增加,仅仅靠增加机器已不能满足系统的要求,于是需要对应用系统进行垂直拆分和水平拆分.在拆分之后,各个被拆分的模块如何通信?如何保证 性能?如何保证各个应用都以同样的方式交互?这 ...

  2. lintcode:Flip Bits 将整数A转换为B

    题目: 将整数A转换为B 如果要将整数A转换为B,需要改变多少个bit位? 样例 如把31转换为14,需要改变2个bit位. ()10=()2 ()10=()2 挑战 你能想出几种方法? 解题: A- ...

  3. 只有innoDB才允许使用外键

    1.只有InnoDB引擎才允许使用外键,所以,我们的数据表必须使用InnoDB引擎. 2.注意: 1.必须使用InnoDB引擎: 2.外键必须建立索引(INDEX): 3.外键绑定关系这里使用了“ O ...

  4. ADO.NET入门教程(三) 连接字符串,你小觑了吗?

    出处:http://www.cnblogs.com/liuhaorain/archive/2012/02/12/2347914.html 摘要 ADO.NET强大的优势在于对不同的数据源提供一致的访问 ...

  5. [iOS]iPhone进行真机测试(基础版)

    买完688个人开发者账号之后,如何进行真机测试呢??看下面 1.打开https://developer.apple.com 然后,输入我们买过688点那个App ID帐号和密码哦!!一定是要支付过的! ...

  6. QTP公开课视频-持续更新中。。。

    以下是视频的下载地址: http://pan.baidu.com/share/link?shareid=1760499709&uk=3711405498

  7. Android开发之源码:多次点击事件的原理和实现

    多次点击事件 多次点击事件原理:最后一次点击事件与第一次点击事件的时间间隔是否小于某个时间,当小于的时候,就认为这是一个多次点击事件. Android源码实现效果: import android.ap ...

  8. [HDOJ2874]Connections between cities(LCA, 离线tarjan)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2874 这题有不连通的情况,特别注意. 觉得是存query的姿势不对,用前向星存了一遍,还是T…… /* ...

  9. javascript插件编写小结

    写JS插件,最好是先通过HTML方式将展示结果显示出来,然后再封装成JS插件,将其画出来.JS模板如下: (function($){ $.fn.fnName = function(options){ ...

  10. SQL2008 R2 主从数据库同步设置

    一.准备工作: 主数据库服务器: OS:Windows Server 2008 R2    DB: SQL Server 2008 R2 Hostname : CXMasterDB IP: 192.1 ...