静态文件中间件的能力

  • 支持指定相对的路径
  • 支持目录浏览
  • 支持设置默认文档
  • 支持多目录映射

首先使用静态文件中间件

  1. // 通过这一行代码就可以访问到静态配置文件
  2. app.UseStaticFiles();

这样就可以将 wwwroot目录映射出来,这是一个默认的配置,也就是说,当我们需要使用中间件静态文件输出的时候,首选就是应该把静态文件放在 wwwroot下面
我们在这个目录下面放了几个文件:index.htmlapp.js,a 目录下面也有一个index.html和一个a.js,这两个 index.html的内容是不一样的
a/index.html

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="utf-8" />
  5. <title>/a/index</title>
  6. <script src="a.js"></script>
  7. </head>
  8. <body>
  9. <h1>这是/a/index</h1>
  10. </body>
  11. </html>

index.html

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="utf-8" />
  5. <title>静态首页</title>
  6. <script src="app.js"></script>
  7. </head>
  8. <body>
  9. <h1>这是静态首页</h1>
  10. </body>
  11. </html>

启动程序,由于我们没有指定相对路径,所以说我们的根目录是/,就代表访问到了 wwwroot,输入 index.html,可以看到 javaScript执行

  1. https://localhost:5001/index.html

23 | 静态文件中间件:前后端分离开发合并部署骚操作 - 图1
23 | 静态文件中间件:前后端分离开发合并部署骚操作 - 图2
如果把地址换一下,会得到另一个页面

  1. https://localhost:5001/a/index.html

23 | 静态文件中间件:前后端分离开发合并部署骚操作 - 图3
23 | 静态文件中间件:前后端分离开发合并部署骚操作 - 图4
如果默认情况下都是访问 index.html,怎么做呢?

  1. app.UseDefaultFiles();

这个方法还有一个重载

  1. namespace Microsoft.AspNetCore.Builder
  2. {
  3. public static class DefaultFilesExtensions
  4. {
  5. public static IApplicationBuilder UseDefaultFiles(
  6. this IApplicationBuilder app);
  7. public static IApplicationBuilder UseDefaultFiles(
  8. this IApplicationBuilder app,
  9. DefaultFilesOptions options);
  10. public static IApplicationBuilder UseDefaultFiles(
  11. this IApplicationBuilder app,
  12. string requestPath);
  13. }
  14. }

DefaultFilesOptions

  1. namespace Microsoft.AspNetCore.Builder
  2. {
  3. public class DefaultFilesOptions : SharedOptionsBase
  4. {
  5. public DefaultFilesOptions();
  6. public DefaultFilesOptions(SharedOptions sharedOptions);
  7. public IList<string> DefaultFileNames { get; set; }
  8. }
  9. }

可以设置 DefaultFileNames,默认 index.html 是在里面的,所以这里可以不输入任何参数

启动程序,访问根目录的时候,应该输出首页的 index

  1. https://localhost:5001/

访问 a 目录会输出 a 的 index
还有一种场景就是我们需要浏览我们的目录
ConfigureServices注册 AddDirectoryBrowser

  1. public void ConfigureServices(IServiceCollection services)
  2. {
  3. services.AddControllers();
  4. services.AddDirectoryBrowser();
  5. }

然后在 Configure里面启用

  1. app.UseDirectoryBrowser();

启动程序,访问根目录
23 | 静态文件中间件:前后端分离开发合并部署骚操作 - 图5

可以看到浏览器上面显示了目录的文件,当我们点击其中的一个文件的时候,实际上是访问这个文件,我们还可以浏览它的子目录
这是我们在使用 wwwroot 的情况下,实际上我们还可以使用其他的目录,把其他的目录也注册进来
我们在应用程序的 file 目录下面另外添加了一个 page.html

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="utf-8" />
  5. <title>page</title>
  6. </head>
  7. <body>
  8. <h1>page</h1>
  9. </body>
  10. </html>

我们也期望可以访问到这个文件,我们就可以这样去做

  1. app.UseStaticFiles();
  2. app.UseStaticFiles(new StaticFileOptions
  3. {
  4. // 注入我们的物理文件提供程序,把我们的当前目录加 file,就是 file 目录,赋值给我们的提供程序
  5. // 这样子的效果就是我们的 wwwroot 会优先去寻找我们的文件,如果没有的话就会执行下一个中间件
  6. // 然后在这个中间件里面再找我们的文件是否存在,如果没有的话,它会去执行后面的路由和 MVC 的 Web API 的 Controller
  7. FileProvider = new PhysicalFileProvider(Path.Combine(Directory.GetCurrentDirectory(), "file"))
  8. });

因为这里我们入参并没有设置相对路径,也就是说我们根目录对应的也是 file 这个目录,我们这里可以输出 page.html

  1. https://localhost:5001/page.html

23 | 静态文件中间件:前后端分离开发合并部署骚操作 - 图6
我们的 page.html 就可以访问到了
还有一种情况是我们希望把我们的静态目录映射为某一个特定的 URL 地址目录下面,我们可以这样去做

  1. app.UseStaticFiles();
  2. app.UseStaticFiles(new StaticFileOptions
  3. {
  4. // 我们希望把我们的静态目录映射为某一个特定的 URL 地址目录下面
  5. RequestPath = "/files",
  6. // 注入我们的物理文件提供程序,把我们的当前目录加 file,就是 file 目录,赋值给我们的提供程序
  7. // 这样子的效果就是我们的 wwwroot 会优先去寻找我们的文件,如果没有的话就会执行下一个中间件
  8. // 然后在这个中间件里面再找我们的文件是否存在,如果没有的话,它会去执行后面的路由和 MVC 的 Web API 的 Controller
  9. FileProvider = new PhysicalFileProvider(Path.Combine(Directory.GetCurrentDirectory(), "file"))
  10. });

访问以下路径就可以看到我们的静态文件页面

  1. https://localhost:5001/files/page.html

也就是说我们可以把任意的文件目录映射为任意的 URL 地址

这里还有一个比较特殊的用法
一般情况下,我们前后端分离的架构,前端会编译成一个 index.html 文件和若干个 CSS 文件和 JavaScript 和图片文件
CSS 文件和 JavaScript 和图片文件一般会部署在 CDN 服务器上,这个 index 文件就需要我们建立一个宿主来 host 它
并且前端的一般路由的话,我们现在都会用 HTML5 的 History 的路由模式
这个时候前端就会对后端有一个特殊的诉求,除了 API 的请求以外,其他的请求的响应都应该是 index.html 这个静态文件
要达到这个目的,我们可以借助我们的中间件的执行原理来实现
首先假设我们的 index.html 就是我们前端编译好的静态文件,我们放置在 wwwroot 下面,前端编译的任何文件都放在 wwwroot 下面
然后我们再做一件事件就是 UseStaticFiles,我们把目录访问整个去掉

  1. //services.AddDirectoryBrowser();

首先映射静态文件

  1. app.UseStaticFiles();

静态文件映射出来之后实际上还有一个诉求,就是当我们访问其他特殊的页面地址的时候,比如说 /order/get 这样子的页面的时候,也应该响应我们的静态文件
这个时候我们可以把这样一段逻辑加入进来

  1. // 判断我们当前的请求是否满足条件
  2. app.MapWhen(context =>
  3. {
  4. // 如果我们的请求不是以 API 开头的请求
  5. return !context.Request.Path.Value.StartsWith("/api");
  6. }, appBuilder =>
  7. {
  8. // 如果满足条件,我就走我下面这一段中间件的逻辑
  9. var option = new RewriteOptions();
  10. // 重写为 /index.html
  11. option.AddRewrite(".*", "/index.html", true);
  12. appBuilder.UseRewriter(option);
  13. // 重写完之后再使用我们的静态文件中间件
  14. appBuilder.UseStaticFiles();
  15. });

这样子可以达到一个效果就是我们访问任意的非 API 目录的时候,我们都可以得到 index.html

启动程序

  1. https://localhost:5001/api/weatherforecast

可以正常访问
API 的请求我们都是让它通过的,不是 API 的时候才会拦截
这个时候如果访问

  1. https://localhost:5001/order

会发现获得的是静态文件
如果说静态文件是存在的,这个时候实际上会响应原有的静态文件,比如说访问

  1. https://localhost:5001/a/index.html

这样子就可以发现我们能让静态文件的目录正常工作,并且能将其他的我们需要的地址都重定向到 index.html
当然这里还有另外一种写法,就是不用 UseRewriter 的方式,而是用 Run 的方式,也是就用断路器的方式

  1. // 判断我们当前的请求是否满足条件
  2. app.MapWhen(context =>
  3. {
  4. // 如果我们的请求不是以 API 开头的请求
  5. return !context.Request.Path.Value.StartsWith("/api");
  6. }, appBuilder =>
  7. {
  8. //// 如果满足条件,我就走我下面这一段中间件的逻辑
  9. //var option = new RewriteOptions();
  10. //// 重写为 /index.html
  11. //option.AddRewrite(".*", "/index.html", true);
  12. //appBuilder.UseRewriter(option);
  13. //// 重写完之后再使用我们的静态文件中间件
  14. //appBuilder.UseStaticFiles();
  15. appBuilder.Run(async c =>
  16. {
  17. // 读取静态文件,并且输出给我们的 Response
  18. var file = env.WebRootFileProvider.GetFileInfo("index.html");
  19. c.Response.ContentType = "text/html";
  20. using (var fileStream = new FileStream(file.PhysicalPath, FileMode.Open, FileAccess.Read))
  21. {
  22. await StreamCopyOperation.CopyToAsync(fileStream, c.Response.Body, null, BufferSize, c.RequestAborted);
  23. }
  24. });
  25. });

这种写法有一个缺点就是,没办法像静态文件中间件那样,输出正确的 Http 请求头
对比一下两种方式的输出的请求头的不同
启动程序,访问

  1. https://localhost:5001/order

打开调试工具,可以看到对 order 的我们的响应头就只有 4 个
23 | 静态文件中间件:前后端分离开发合并部署骚操作 - 图7
其他的静态文件,响应头会多出来 etag,data,last-modified
23 | 静态文件中间件:前后端分离开发合并部署骚操作 - 图8
这些的话就是我们关于 HTTP缓存可以用到的头,所以说我们还是推荐使用上面这种方式,静态中间件的方式,而不是自己输出文件的方式