UrlRewritingNet is an Url rewriting tool for ASP .Net and Elmahis a module for logging unhandled errors.

UrlRewritingNet can set a default location for “directory” requests via defaultPage property in <urlrewritingnet> section. When a file without extension is requested the defaultPage value is appended to the original URL.

Elmah provides a handler for getting the errors summary, usually called elmah.axd. This handler also responds to the followings requests:

/elmah.axd/rss – RSS errors list feed

/elmah.axd/digestrss – RSS digest

/elmah.axd/download –  comma separated errors list

/elmah.axd/about – about page

/elmah.axd/stylesheet – the stylesheet used

/elmah.axd/detail?id=XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX – html single error summary

/elmah.axd/xml?id=XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX – xml single  error summary

/elmah.axd/json?id=XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX – json single error summary

These requests are like a request for a file with no extension. This is why the  UrlRewritingNet adds the defaultPage value which leads to the 404 Http response. First sign of this behavior is when the generated  html does not contain any CSS style applied.

To fix this situation I simply removed the defaultPage attribute from <urlrewritingnet>.

Now if the default page is not set on IIS, then will get an error when the web site is accessed only by the domain name. This situation was handled by  UrlRewritingNet, but in not a proper way, because it returns a 302 Http response(302 Found) with the location set by the defaultPage attribute value, and I think is not the best solution to give the first page when is about SEO. The drawback when the defaultPage attribute is removed is when the directories in the web site are accessed, it will give a 403.14 – Forbidden(IIS 7). But this can be handled by UrlRewritingNet using a custom UrlRewritingProvider.

During developing this custom provider I noticed it would be better to set and the HttpStatus when the url's are rewriten. So I added a new attribute to the <add/> node called httpStatusCode.

Creating a new UrlRewriterNet is very simple:

public class RootProvider : UrlRewritingProvider

{

public override RewriteRule CreateRewriteRule()

{

return new RootRule();

}

}

The class RootRule does all the logic:

public class RootRule : RewriteRule

{

public override void Initialize(UrlRewritingNet.Configuration.RewriteSettings rewriteSettings)

{

base.Initialize(rewriteSettings);

VirtualUrl = rewriteSettings.GetAttribute("virtualUrl", "");

DestinationUrl = rewriteSettings.GetAttribute("destinationUrl", "");

HttpStatusCode = rewriteSettings.GetAttribute("httpStatusCode", "");

}

public override bool IsRewrite(string requestUrl)

{

return requestUrl == VirtualUrl;

}

public override string RewriteUrl(string url)

{

if (!String.IsNullOrEmpty(HttpStatusCode))

{

HttpStatusCodeHandler handler = null;

switch (HttpStatusCode)

{

case "404" :

handler = new _404Handler(DestinationUrl);

break;

case "301":

handler = new _301Handler(DestinationUrl);

break;

case "302":

handler = new _302Handler(DestinationUrl);

break;

default:

handler = new NotImplementedHandler(DestinationUrl);

break;

}

handler.Execute();

return null;

}

return DestinationUrl;

}

public string VirtualUrl { get; set; }

public string DestinationUrl { get; set; }

public string HttpStatusCode { get; set; }

}

The  RootRule class instantiates a specific handler, depending by the http status code.

I created a base class to define the model of how a status code could be handled.

public class HttpStatusCodeHandler {

protected string destinationUrl;

protected HttpStatusCodeHandler() { }

public HttpStatusCodeHandler(string DestinationUrl) {

destinationUrl = DestinationUrl;

}

public virtual void Execute() {

throw new NotImplementedHttpStatusException();

}

}

For sample when a directory is accessed it can be used a 404 response.

public sealed class _404Handler : HttpStatusCodeHandler

{

private _404Handler() { }

public _404Handler(string DestinationUrl) : base(DestinationUrl) { }

public override void Execute()

{

HttpContext.Current.Response.Clear();

HttpContext.Current.Response.Status = "404 Not Found";

HttpContext.Current.Response.StatusDescription = "Not Found";

HttpContext.Current.Response.Write(File.ReadAllText(HttpContext.Current.Server.MapPath(destinationUrl)));

HttpContext.Current.Response.End();

}

}

The 302 and 301 reponses needs to add the Location in the header response. The location contains the new URL where the old resource exists now.

public sealed class _302Handler : HttpStatusCodeHandler

{

private _302Handler() { }

public _302Handler(string DestinationUrl) : base(DestinationUrl) { }

public override void Execute()

{

HttpContext.Current.Response.Clear();

HttpContext.Current.Response.Status = "302 Found";

HttpContext.Current.Response.StatusDescription = "Found";

HttpContext.Current.Response.AddHeader("Location", destinationUrl);

HttpContext.Current.Response.End();

}

}

public sealed class _301Handler : HttpStatusCodeHandler

{

private _301Handler() { }

public _301Handler(string DestinationUrl) : base(DestinationUrl) { }

public override void Execute()

{

HttpContext.Current.Response.Clear();

HttpContext.Current.Response.Status = "301 Moved Permanently";

HttpContext.Current.Response.StatusDescription = "Moved Permanently";

HttpContext.Current.Response.AddHeader("Location", destinationUrl);

HttpContext.Current.Response.End();

}

}

public sealed class NotImplementedHandler : HttpStatusCodeHandler

{

private NotImplementedHandler() { }

public NotImplementedHandler(string DestinationUrl) : base(DestinationUrl) { }

public override void Execute()

{

throw new NotImplementedHttpStatusException();

}

}

public class NotImplementedHttpStatusException : Exception

{

public override string Message

{

get

{

return "NotIplementedHttpStatusException";

}

}

}

Now in the RewriteRule section I define some rules:

to define a default page when the folder “products” is accessed

<add name="products" virtualUrl="/products/" destinationUrl="/products/latest_products.asp" httpStatusCode="302"rewriteUrlParameter="ExcludeFromClientQueryString" ignoreCase="true" provider="RootProvider"/>

to say that a page is permanently moved and the new location is other page

<add name="about" virtualUrl="/pages.asp?func=get_content&amp;page_id=1" destinationUrl="/about.asp" httpStatusCode="301"rewriteUrlParameter="IncludeQueryStringForRewrite" ignoreCase="true" provider="RootProvider"/>

These redirects are very helpful when you want to keep the search engines rankings.

原文链接:http://csharpin.blogspot.com/2009/03/using-urlrewritingnet-and-elmah.html

urlrewritingnet 域名http状态302 问题(转)的更多相关文章

  1. shell+curl监控网站页面(域名访问状态),并利用sedemail发送邮件

    应领导要求,对公司几个主要站点的域名访问情况进行监控.下面分享一个监控脚本,并利用sendemail进行邮件发送. 监控脚本如下:下面是写了一个多线程的网站状态检测脚本,直接从文件中读出站点地址,然后 ...

  2. shell+curl监控网站页面(域名访问状态),并利用sendemail发送邮件

    应领导要求,对公司几个主要站点的域名访问情况进行监控.下面分享一个监控脚本,并利用sendemail进行邮件发送. 监控脚本如下:下面是写了一个多线程的网站状态检测脚本,直接从文件中读出站点地址,然后 ...

  3. Nginx 301重定向域名

    为何要使用301重定向 在网站建设中需要网页重定向的情况很多:如网页目录结构变动,网页重命名.网页的扩展名改变.网站域名改变等.如果不做重定向,用户的收藏和搜索引擎数据库中的旧地址只能让访客得到一个4 ...

  4. 三、nginx301跳转302跳转

    301跳转设置: server { listen 80; server_name downcc.com; rewrite ^/(.*) http://www.downcc.com/$1 permane ...

  5. 什么是HTTP协议?常用的状态码有哪些?

    一.HTTP简介 HTTP协议是Hyper Text Transfer Protocol(超文本传输协议)的缩写,是用于从万维网(WWW:World Wide Web )服务器传输超文本到本地浏览器的 ...

  6. 未注册wang域名批量查询工具

    一.支持规则查询 可自定义生成域名进行查询,可生成任意位数的字母数字域名,根据[声母].[韵母]生成单拼,双拼,三拼等域名,还可根据字典生成,支持全拼.首拼识别,全国城市区号.城市全拼.城市首拼.热门 ...

  7. 包含为 HTTP 定义的状态代码的值(枚举)

    using System; namespace System.Net { // 摘要: // 包含为 HTTP 定义的状态代码的值. public enum HttpStatusCode { // 摘 ...

  8. php调用whois接口域名查询

    由两部分组成,一个index.php文件,一个whois的接口文件: <html> <head> <title>域名到期查询</title> <s ...

  9. C#HttpWebResponse请求常见的状态码

    成员名称 说明 Continue 等效于 HTTP 状态 100.Continue 指示客户端可能继续其请求. SwitchingProtocols 等效于 HTTP 状态 101.Switching ...

随机推荐

  1. maven整合s2sh截图

  2. HDU 1423 Greatest Common Increasing Subsequence

    最长公共上升子序列   LCIS 看这个博客  http://www.cnblogs.com/nuoyan2010/archive/2012/10/17/2728289.html #include&l ...

  3. acess() 判断目录是否存在

    acess()功能描述: 检查调用进程是否可以对指定的文件执行某种操作. <pre lang="c" escaped="true">#include ...

  4. 两天三场Java实习生面试总结

    Java 关键字(如abstract)[详解] String[相关面试题] String.StringBuffer.StringBuilder区别 String中有没有使一个字符串反转的方法 线程的实 ...

  5. hdu 5256 LIS变形

    给一个数列,问最少修改多少个元素使数列严格递增.如果不是要求“严格”递增,那就是求最长不降子序列LIS,然后n-LIS就是答案.要严格递增也好办,输入的时候用每个数减去其下标处理一下就行了. /* * ...

  6. js 判断页面是否加载完成

    javascript代码如下: document.onreadystatechange = subSomething; //当页面加载状态改变的时候执行这个方法 function subSomethi ...

  7. Swift相关图书推荐

    Swift与Cocoa框架开发 作      者 [澳] 曼宁(Jonathon Manning),巴特菲尔德-艾迪生(Paris Buttfield 出 版 社 人民邮电出版社 出版时间 2015- ...

  8. c语言的几个重要知识点

      内存结构 这是核心中的核心,请仔细看完,充分理解,否则请不要看下一节内容. 每个程序一启动都有一个大小为4GB的内存,这个内存叫虚拟内存,是概念上的,真正能用到的,只是很小一部分,一般也就是在几百 ...

  9. C++学习之路--类的构建以及数据转换存储

    注意理解下面的代码,数据的处理与转换. 头文件: #ifndef STACK_H #define STACK_H class Stack { struct Link { void* data; Lin ...

  10. 【boost】使用lambda表达式和generate_n生成顺序序列

    程序中经常用到顺序序列(0,1,2,3,4,5,6.....),一直羡慕python有range这样的函数,而C++中通常只有用循环来处理这种初始化. 现在,结合boost库lambda(虽然差C++ ...