我正在尝试创建一个系统来使用 MVC3 和 IIS URL Rewrite 2 从 GridFS 提供图像及其调整大小的版本。经过测试,我意识到直接从文件系统提供图像比使用 GridFS 文件流提供图像快 10 倍。然后我决定将原始文件保留在 GridFS 中,并使用 Url Rewrite 2 和 Asp.Net 处理程序的组合在服务器本地文件系统上创建原始文件的副本和调整大小的版本。
以下是我用于提供原始版本和调整大小版本的重写规则:
<rule name="Serve Resized Image" stopProcessing="true">
<match url="images/([a-z]+)/[a-f0-9]+/[a-f0-9]+/[a-f0-9]+/([a-f0-9]+)-([a-f0-9]+)-([a-f0-9]+)-([0-9]+)\.(.+)" />
<conditions logicalGrouping="MatchAll" trackAllCaptures="false">
<add input="{REQUEST_FILENAME}" matchType="IsFile" negate="true" />
</conditions>
<action type="Rewrite" url="/Handlers/ImageResizer.ashx?Uri={REQUEST_URI}&Type={R:1}&Id={R:2}&Width={R:3}&Height={R:4}&ResizeType={R:5}&Extension={R:6}" appendQueryString="false" logRewrittenUrl="true" />
</rule>
<rule name="Serve Original Image" stopProcessing="true">
<match url="images/([a-z]+)/[a-f0-9]+/[a-f0-9]+/[a-f0-9]+/([a-f0-9]+)\.(.+)" />
<conditions logicalGrouping="MatchAll" trackAllCaptures="false">
<add input="{REQUEST_FILENAME}" matchType="IsFile" negate="true" />
</conditions>
<action type="Rewrite" url="/Handlers/Images.ashx?Uri={REQUEST_URI}&Type={R:1}&Id={R:2}&Extension={R:3}" appendQueryString="false" logRewrittenUrl="true" />
</rule>
如您所见,重写引擎检查文件是否存在于文件系统上,如果不存在。重写 url 并将请求发送到处理程序。处理程序为流提供服务并将文件写入文件系统。在下一个请求中,文件直接从文件系统提供。我通过拆分文件的 24 个字符 ID(MongoDB 对象 ID 作为字符串)将文件分隔到文件夹中,以避免同一文件夹中出现数十万张图像。
这是一个示例原始图像请求:
http://localhost/images/test/50115c53/1f37e409/4c7ab27d/50115c531f37e4094c7ab27d.jpg
这个和调整大小的版本没有任何问题。
由于这个 URL 太长并且有重复,我决定再次使用重写引擎来缩短 URL 以自动生成文件夹名称。这是我放在顶部的规则:
<rule name="Short Path for Images">
<match url="images/([a-z]+)/([a-f0-9]{8})([a-f0-9]{8})([a-f0-9]{8})(.+)" />
<action type="Rewrite" url="images/{R:1}/{R:2}/{R:3}/{R:4}/{R:2}{R:3}{R:4}{R:5}" appendQueryString="false" logRewrittenUrl="true"></action>
</rule>
当我使用此规则请求图像时,例如使用以下 URL:
http://localhost/images/test/50115c531f37e4094c7ab27d.jpg
如果图像已经在文件系统上,它只会提供图像,否则我会收到以下错误:
HTTP 错误 500.50 - URL 重写模块错误。由于发生内部服务器错误,无法显示该页面。
我已经检查了请求的 IIS 日志文件条目。它不显示任何细节,除了:
2012-08-02 14:44:51 127.0.0.1 GET /images/test/50115c531f37e4094c7ab27d.jpg - 80 - 127.0.0.1 Mozilla/5.0+(Windows+NT+6.1;+WOW64)+AppleWebKit/537.1+(KHTML,+like+Gecko)+Chrome/21.0.1180.60+Safari/537.1 500 50 161 37
另一方面,成功的请求会记录重写的 URL,如下所示:
GET /Handlers/ImageResizer.ashx Uri=/images/test/50115c53/1f37e409/4c7ab27d/50115c531f37e4094c7ab27d-1f4-1f4-2.jpg&Type=test&Id=50115c531f37e4094c7ab27d&Width=1f4&Height=1f4&ResizeType=2&Extension=jpg
Elmah 和 EventLog 也没有显示任何内容。在我的控制器方法的顶部添加了一个文件系统记录器,它不会记录这些特别有问题的请求。
任何人都可以建议一种解决方法来让它工作吗?
编辑:在RuslanY关于Failed Request Tracing的建议之后,我设法确定了错误:
ModuleName: RewriteModule
Notification: 1
HttpStatus: 500
HttpReason: URL Rewrite Module Error.
HttpSubStatus: 50
ErrorCode: 2147942561
ConfigExceptionInfo:
Notification: BEGIN_REQUEST
ErrorCode: The specified path is invalid. (0x800700a1)
可以在此处查看整个跟踪结果(仅限 IE)
不幸的是,这仍然没有让我找到解决方案,因为当文件系统上存在文件时,第二条规则(因此缩短规则)正在工作。