可以使用ASP.Net路由(而不是MVC)来提供静态文件吗?
说我想路由
http://domain.tld/static/picture.jpg
到
http://domain.tld/a/b/c/picture.jpg
我希望动态地执行它,因为重写的URL是动态计算的。我无法一劳永逸地建立静态路线。
无论如何,我可以创建这样的路线:
routes.Add(
"StaticRoute", new Route("static/{file}", new FileRouteHandler())
);
在FileRouteHandler.ProcessRequest
方法中,我可以重写从/static/picture.jpg
到/a/b/c/picture.jpg
的路径。然后我想为静态文件创建一个处理程序。 ASP.NET为此目的使用StaticFileHandler
。不幸的是,这堂课是内部的。我试图使用反射创建处理程序,它实际上有效:
Assembly assembly = Assembly.GetAssembly(typeof(IHttpHandler));
Type staticFileHandlerType = assembly.GetType("System.Web.StaticFileHandler");
ConstructorInfo constructorInfo = staticFileHandlerType.GetConstructor(BindingFlags.NonPublic | BindingFlags.Instance, null, Type.EmptyTypes, null);
return (IHttpHandler) constructorInfo.Invoke(null);
但使用内部类型似乎不是正确的解决方案。另一个选择是实现我自己的StaticFileHandler
,但正确地执行此操作(支持范围和etags等HTTP内容)并非易事。
如何在ASP.NET中处理静态文件的路由?
答案 0 :(得分:51)
为什么不使用IIS来执行此操作?您可以创建重定向规则,以便在请求进入您的应用程序之前将任何请求从第一个路由指向第二个路由。因此,这将是一种更快速的重定向请求的方法。
假设你有IIS7 +,你会做类似......
的事情<rule name="Redirect Static Images" stopProcessing="true">
<match url="^static/?(.*)$" />
<action type="Redirect" url="/a/b/c/{R:1}" redirectType="Permanent" />
</rule>
或,如果您不需要重定向,请按照@ ni5ni6的建议:
<rule name="Rewrite Static Images" stopProcessing="true">
<match url="^static/?(.*)$" />
<action type="Rewrite" url="/a/b/c/{R:1}" />
</rule>
编辑2015-06-17 for @RyanDawkins :
如果您想知道重写规则的去向,请在web.config
文件中找到其位置的地图。
<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<system.webServer>
<rewrite>
<rules>
<!-- rules go below -->
<rule name="Redirect Static Images" stopProcessing="true">
<match url="^static/?(.*)$" />
<action type="Redirect" url="/a/b/c/{R:1}" redirectType="Permanent" />
</rule>
</rules>
</rewrite>
</system.webServer>
</configuration>
答案 1 :(得分:41)
在深入研究这个问题几个小时之后,我发现只需添加忽略规则就可以获得静态文件。
在RegisterRoutes(RouteCollection路由)中,添加以下忽略规则:
routes.IgnoreRoute("{file}.js");
routes.IgnoreRoute("{file}.html");
答案 2 :(得分:10)
我遇到了类似的问题。我最终使用HttpContext.RewritePath:
public class MyApplication : HttpApplication
{
private readonly Regex r = new Regex("^/static/(.*)$", RegexOptions.IgnoreCase);
public override void Init()
{
BeginRequest += OnBeginRequest;
}
protected void OnBeginRequest(object sender, EventArgs e)
{
var match = r.Match(Request.Url.AbsolutePath);
if (match.Success)
{
var fileName = match.Groups[1].Value;
Context.RewritePath(string.Format("/a/b/c/{0}", fileName));
}
}
}
答案 3 :(得分:6)
我提出了使用内部StaticFileHandler
的替代方法。在IRouteHandler
我致电HttpServerUtility.Transfer
:
public class FileRouteHandler : IRouteHandler {
public IHttpHandler GetHttpHandler(RequestContext requestContext) {
String fileName = (String) requestContext.RouteData.Values["file"];
// Contrived example of mapping.
String routedPath = String.Format("/a/b/c/{0}", fileName);
HttpContext.Current.Server.Transfer(routedPath);
return null; // Never reached.
}
}
这是一个黑客。 IRouteHandler
应该返回IHttpHandler
而不是中止并转移当前请求。但是,它确实实现了我想要的目标。
使用内部StaticFileHandler
也有点像黑客,因为我需要反思才能访问它,但至少有一些documentation on StaticFileHandler
on MSDN使它成为一个稍微“官方”的类。不幸的是,我认为不可能在部分信任环境中反思内部类。
我会坚持使用StaticFileHandler
因为我认为在可预见的将来它不会从ASP.NET中删除。
答案 4 :(得分:4)
您需要添加TransferRequestHandler来处理静态文件。请参阅以下答案 https://stackoverflow.com/a/21724783/22858