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. java中对浮点数精度的处理DecimalFormat

    DecimalFormat是一个队浮点数进行格式化输出的利器,比如我们要输出一个保留一位小数的浮点数,可以键入如下代码: DecimalFormat df = new DecimalFormat(&q ...

  2. 【英语】Bingo口语笔记(46) - 不可能的表达

  3. yii 打印sql

    $query = TableModel::find()->where([‘xxx’=>xxx]); var_dump($query->prepare(\Yii::$app->d ...

  4. 在centOS中加入本地ISO yum源

    注:本文转载自<liujun_live的博客>,感谢原博主的辛勤写作:原文地址:http://blog.sina.com.cn/s/blog_8ea8e9d50101em6f.html 在 ...

  5. 通过文件流stream下载文件

    public ActionResult ShowLocalizedXML(int id) { string orderName = ""; string xmlString = G ...

  6. mate代码详解

    1.用以说明生成工具(如MICROSOFT FRONTPAGE 4.0)等: 2.向搜索引擎说明你的网页的关键词: 3.告诉搜索引擎你的站点的主要内容: 4.告诉搜索引擎你的站点的制作的作者: 5. ...

  7. 4. 2D绘制与控件绘制

    绘制基本图形和文本 绘制图形和文本的基本方法 drawPoint(绘制点).drawLine(绘制直线).drawCircle(绘制圆) drawArc(绘制弧).drawText(绘制文本) pac ...

  8. Php 笔记2-----手机端 与 php服务器的通信

    对于 手机端 和 php服务器的通信,是不存在表单这一概念的  ,除非自己去实现, 所以通常情况下步骤是: 假定上传的是字符串. 1  手机端的流程是 把文件或者字符串,转化为 特定的流. 2 通过h ...

  9. JavaScript专业规则12条

    学习JavaScript是困难的.它发展的如此之快,以至于在任何一个特定的时刻,你都不清楚自己是否“做错了”.有些时候,感觉像是坏的部分超过了好的部分.然而,讨论这些并没有意义,JavaScript正 ...

  10. linux笔记_20150417_ubuntu 常见问题_文件_音乐播放器

    最近在学习ubuntu的过程中,遇到了一些问题,就记下来了它的解决办法.以希望对你也有用. ),至少保证周围局域网内用户可以访问.至于配置文件,内容比较少,反正对我来讲能用就ok了~不知道会不会很弱 ...