ASP.NET Core – 网站发布要做的事儿 Publish, Minification, Compression, Cache, HSTS, URL Rewrite Https & www, Close IIS Feature
前言
要发布网站需要做一些优化, 比如 cache, compression, minification 等等.
以前有写过相关的文章: Asp.net core 学习笔记 ( IIS, static file 性能优化 )
这篇作为一个大整理.
Publish Website
Visual Studio 我就不介绍了. 这里用的是 dotnet CLI
dotnet publish ProjectName.csproj -o C:\keatkeat\projects\release
project name 是 optional, -o 是 output 路径
注: 它不会把原先的 files delete 掉哦, 而且也没有 delete existing files 的 command. Github Issue,
Compression
压缩就是把 .js, .css 这类 files 做成 gzip 或者 br, 这可以大大介绍文件体积, 传输的时候就快多了.
在 service provider 做 setup, EnableForHttps 默认是 false, 开启的话要注意 security risk.
builder.Services.AddResponseCompression(options =>
{
options.EnableForHttps = true;
});
接着在 UseStaticFiles 之前运行 UseResponseCompression
app.UseResponseCompression();
app.UseStaticFiles();
效果
注: Razor page / MVC view 返回的 HTML response 也是会被 compress 的哦
Cache
在 service provider 设置
builder.Services.Configure<StaticFileOptions>(options =>
{
options.OnPrepareResponse = ctx =>
{
var cachePeriod = TimeSpan.FromDays(365 * 15).TotalSeconds.ToString();
ctx.Context.Response.Headers.Append("Cache-Control", $"public, max-age={cachePeriod}");
};
});
效果
游览器是依据 URL 做 cache 的, 像 .js .css file name 最好配上 hash, 只要内容不变缓存就一定生效, 只要内容变了 hash 也就变了, 缓存自然就无效了. 具体做法可以看这篇
缓存是 static file 而已, Razor Page response 的 HTML 是不可能被缓存的哦 (那个要缓存就要做 service worker 了)
注意: 用 chrome 测试的话, dev tools 开启会自动缓存哦, 所以要测试最好是把 dev tools 关了. 参考: stackoverflow – Google chrome css doesn't update unless clear cache
Minification
minification 就是把空格, 注释删除, 把方法变量名字变短. .js .css 我都是用 webpack 做, 可以看这篇.
Razor Page response 的 HTML 则需要用到插件 WebMarkupMin
它有多个版本, 要找对来安装哦, 每一个是不同 package 来的...
也没有找到教程, 101 篇是 2016 年写的 HTML minification using WebMarkupMin in ASP.NET Core
在 service provider 设置
builder.Services.AddWebMarkupMin(options =>
{
options.AllowMinificationInDevelopmentEnvironment = true;
}).AddHtmlMinification();
如果想在 dev mode 看到效果要记得开启 AllowMinificationInDevelopmentEnvironment 哦
接着在 RazorPages 之前 UseWebMarkupMin
app.UseWebMarkupMin();
app.MapRazorPages();
当遇上 \r\n new line and white-space: pre-line
默认情况下, minification 也会把 \r\n new line 给删除掉, 当我们需要 \r\n 的时候可以用以下 2 格方法
1. wmm:ignore
<!--wmm:ignore-->@Html.Raw("Test \nTest")<!--/wmm:ignore-->
2. 修改 default setting
services.AddWebMarkupMin().AddHtmlMinification(options =>
{
options.MinificationSettings.PreserveNewLines = true;
});
这样 \r\n 就被保留了
隐患和坑
1. 之前做 service worker 的时候有遇过一些鬼问题 Github Issus – Service Worker: Hash Mismatch
2. WhatsApp share link image no show.
这是因为 HTML minify 以后 quote 会消失. 而 WhatsApp 的 crawl 视乎不够聪明. 参考1, 参考2 (sharing debugger 是 ok 的, facebook messager 也是 ok 的, 就 WhatsApp 有问题....)
解决方法是保留 quote
services.AddWebMarkupMin().AddHtmlMinification(options =>
{
options.MinificationSettings.AttributeQuotesRemovalMode = WebMarkupMin.Core.HtmlAttributeQuotesRemovalMode.KeepQuotes;
});
HSTS
HSTS 是告诉游览器这个网站只允许 HTTPS 访问, 即使 certificate 过期了, 用户也无法 by pass. (第一次访问允许 http, 游览器得知后就不允许了, 还有一种极端的做法是 manual submit to 游览器公司, 那么连第一次都必须是 HTTPS)
最大的好处是绝对安全, 另一个是游览器会强制使用 HTTPS 访问也少了 http redirect to https 的问题 (性能慢)
ASP.NET Core 默认的 template 就有设置好这个了.
可以自定义时间, 默认是 30 days. (也就是说 30 days 内, 用户只能 HTTPS 访问)
builder.Services.Configure<HstsOptions>(options =>
{
options.MaxAge = TimeSpan.FromMinutes(1); // 默认是 30days
});
接着放到最上方 (比 ExceptionHandler 底而已), 请求一进来就处理.
var app = builder.Build();
if (!app.Environment.IsDevelopment())
{
app.UseExceptionHandler("/Error");
app.UseHsts();
}
URL Rewrite HTTPS and WWW
用户不习惯写 HTTPS and WWW 所以需要做 redirect 统一它们. 不然统计很乱.
service provider 设置 (如果项目是放到 subdomain, 拿自然就没有 redirect WWW 了哦)
builder.Services.Configure<RewriteOptions>(options =>
{
options.AddRedirectToHttpsPermanent();
options.AddRedirectToWwwPermanent();
// options.AddRedirectToHttps();
// options.AddRedirectToWww();
});
permanent 是 301, 没有 permanent 是 302 (temporary 的意思)
接着
var app = builder.Build();
if (!app.Environment.IsDevelopment())
{
app.UseExceptionHandler("/Error");
app.UseHsts();
}
app.UseRewriter();
// app.UseHttpsRedirection();
app.UseResponseCompression();
在 UseHsts 之后执行 UseRewriter
注意: 上面把默认 template 的 UseHttpsRedirection 注释掉了, 其实它和 AddRedirectToHttpsPermanent 功能是一样的.
可以选任意其中一个来用, 我这里统一让 Rewriter 负责. 参考: HTTPS Redirection Middleware alternative approach
301 Redirect for SEO
301 redirect 也是网站发布需要考虑到的. 它也是通过 Rewriter 完成的.
builder.Services.Configure<RewriteOptions>(options =>
{
options.AddRedirect(@"(zh-Hans|id|cn)\/(.*)", "/$2", (int)HttpStatusCode.Moved);
options.AddRedirect(@"(zh-Hans|id|cn)", "/", (int)HttpStatusCode.Moved);
options.AddRedirect(@"blogs\/(.*)", "/tourist-attractions/$1", (int)HttpStatusCode.Moved);
options.AddRedirect(@"blogs", "/tourist-attractions", (int)HttpStatusCode.Moved);
options.AddRedirect(@"about-us", "/about", (int)HttpStatusCode.Moved);
options.AddRedirect(@"cars\/.*", "/fleets", (int)HttpStatusCode.Moved);
options.AddRedirect(@"packages", "/pricing", (int)HttpStatusCode.Moved);
options.AddRedirect(@"testimonials", "/reviews", (int)HttpStatusCode.Moved);
});
第一个参数是 match regex, 第二个是 redirect to 可以使用 $1 配合 regex
308 或 301 都是正确的, 暂时修改地址的话就时候 302
提醒:
Close IIS Feature
上面介绍的许多功能, IIS 都有自带的. 要确保不要跑 2 轮, 必须把 IIS 相关功能移除.
web.config


<?xml version="1.0" encoding="utf-8"?>
<configuration>
<location path="." inheritInChildApplications="false">
<system.webServer>
<handlers>
<add name="aspNetCore" path="*" verb="*" modules="AspNetCoreModuleV2" resourceType="Unspecified" />
</handlers>
<!-- for debug -->
<aspNetCore processPath="dotnet" arguments=".\TestMiniCompressCacheHttpsWww.dll" stdoutLogEnabled="false" stdoutLogFile=".\logs\stdout" hostingModel="inprocess" />
<!-- <aspNetCore processPath="dotnet" arguments=".\TestMiniCompressCacheHttpsWww.dll" stdoutLogEnabled="false" stdoutLogFile=".\logs\stdout" hostingModel="inprocess" /> -->
<modules>
<remove name="RewriteModule" />
<remove name="AspNetCoreModule" />
<!-- <remove name="RequestFilteringModule" /> -->
<remove name="iisnode" />
<!-- <remove name="ProtocolSupportModule" /> -->
<!-- <remove name="IsapiModule" /> -->
<remove name="IpRestrictionModule" />
<!-- <remove name="IsapiFilterModule" /> -->
<remove name="HttpRedirectionModule" />
<remove name="DynamicIpRestrictionModule" />
<!-- <remove name="DirectoryListingModule" /> -->
<!-- <remove name="DefaultDocumentModule" /> -->
<remove name="CorsModule" />
<!-- <remove name="ConfigurationValidationModule" /> -->
<remove name="ApplicationInitializationModule" />
<remove name="WebSocketModule" />
<remove name="AnonymousIdentification" />
<remove name="DefaultAuthentication" />
<remove name="FileAuthorization" />
<remove name="FormsAuthentication" />
<remove name="Profile" />
<remove name="OutputCache" />
<remove name="UrlAuthorization" />
<remove name="RoleManager" />
<remove name="ScriptModule-4.0" />
<remove name="UrlMappingsModule" />
<remove name="UrlRoutingModule-4.0" />
<remove name="WindowsAuthentication" />
<remove name="Session" />
<!-- <remove name="StaticFileModule" /> -->
<!-- <remove name="StaticCompressionModule" /> -->
</modules>
<!-- for debug -->
<!--<httpErrors errorMode="Detailed" />-->
<httpErrors errorMode="DetailedLocalOnly" />
</system.webServer>
</location>
</configuration>
Folder & File Permission
IIS 的 Application 要读写 File 是需要额外 permission 的。
不够 permission 就会报错 Access to the path。
举个例子
var tempFolder = Path.Combine(env.WebRootPath, "uploaded-files");
var bytes = Encoding.UTF8.GetBytes("Hello World");
await System.IO.File.WriteAllBytesAsync($@"{tempFolder}\test.txt", bytes);
如果 test 不存在,或许你可以成功创建它,但是如果 test 已存在,它需要 override,这个可能就会报错 permission 了。
我一般上会直接把 wwwroot folder 添加所以权限给 IIS_IUSRS
wwwroot folder > right click > Properties > Security > Edit > Add > IIS_IUSRS > OK > tick Full control > Apply > OK
ASP.NET Core – 网站发布要做的事儿 Publish, Minification, Compression, Cache, HSTS, URL Rewrite Https & www, Close IIS Feature的更多相关文章
- ASP.NET Core 网站发布到Linux服务器(转)
出处;ASP.NET Core 网站发布到Linux服务器 长期以来,使用.NET开发的应用只能运行在Windows平台上面,而目前国内蓬勃发展的互联网公司由于成本的考虑,大量使用免费的Linux平台 ...
- ASP.NET Core 网站发布到Linux服务器
长期以来,使用.NET开发的应用只能运行在Windows平台上面,而目前国内蓬勃发展的互联网公司由于成本的考虑,大量使用免费的Linux平台,这就使得.NET空有一身绝技但无法得到广大的施展空间,.N ...
- ASP.NET Core 网站在Docker中运行
Docker作为新一代的虚拟化方式,未来肯定会得到广泛的应用,传统虚拟机的部署方式要保证开发环境.测试环境.UAT环境.生产环境的依赖一致性,需要大量的运维人力,使用Docker我们可以实现一次部署, ...
- docker 初识之二(简单发布ASP.NET Core 网站)
在发布ASP.NET Core网站以前,先介绍一下DaoCloud 一个免费的docker云容器服务平台.登陆官方网站,创建一台docker主机,这台主机有120分钟的使用时间,对于鄙人学习使用正好合 ...
- .Net Core 3 骚操作 之 用 Windows 桌面应用开发 Asp.Net Core 网站
前言 曾经在开发 Asp.Net 网站时就在想,为什么一定要把网站挂到 IIS 上?网站项目的 Main 函数哪儿去了?后来才知道这个 Main 函数在 w3wp.exe 里,这也是 IIS 的主进程 ...
- Windows平台部署 Asp.Net Core 3.1.0,将 ASP.NET Core 应用发布到 IIS ,使用 IIS 在 Windows 上托管 ASP.NET Core
第一部分:本教程介绍如何在 IIS 服务器上托管 ASP.NET Core 应用. 官方文档地址:https://docs.microsoft.com/zh-cn/aspnet/core/tutori ...
- 将asp.net core站点发布到IIS上遇到的问题
今天第一次将整个 asp.net core 站点发布到 IIS 上,以前都是发布到 Linux 服务器上. 开始使用 dotnet publish -c release 命令发布,用浏览器访问站点时出 ...
- ASP.NET CORE网站部署到 windows server 的IIS 上去
章基于我自己经验的一个总结,在windows服务器上部署asp.net core网站.环境是 windows server 2012数据中心版本 第一步先安装 IIS 服务器 接下来就是一路下一步,然 ...
- asp.net core 使用 TestServer 来做集成测试
asp.net core 使用 TestServer 来做集成测试 Intro 之前我的项目里的集成测试是随机一个端口,每次都真实的启动一个 WebServer,之前也有看到过微软文档上 TestSe ...
- ASP.NET Core下发布网站
一.windows下发布到IIS 1.前奏:IIS上的准备 (1)IIS 必须安装AspNetCoreModule 模块 下载地址:(DotNetCore.2.0.3-WindowsHosting-a ...
随机推荐
- Git 奇幻之旅⌛️续集
第十二天:暂存未完成的修改 小明和小红在开发一个新功能时,他们需要切换到另一个分支去修复一个紧急的 bug .但是他们的当前分支上还有一些未完成的修改,他们不想提交这些修改,也不想丢弃这些修改.有一天 ...
- 解锁网络无限可能:揭秘微软工程师力作——付费代理IP池深度改造与实战部署指南
基于付费代理的代理IP池 项目来源 此项目为微软某个工程师构建的代理IP池,我对此进行了改造.可以用于生产环境中的爬虫项目 阅读前建议 阅读我之前发布的爬虫基础的文章,了解代理如何获取.使用等. 分为 ...
- Asp .Net Core 系列:基于 T4 模板生成代码
目录 简介 组成部分 分类 Visual Studio 中使用T4模板 创建T4模板文件 2. 编写T4模板 3. 转换模板 中心控制Manager 根据 MySQL 数据生成 实体 简介 T4模板, ...
- oeasy教您玩转vim - 13 - # 大词小词
大词小词 回忆上节课内容 我们上次学习了 e e 代表 end 词尾 自有跳跃 还可以成倍次数的跳跃 但其实我是想以一个一个属性地跳跃,有没有方法呢? 查询帮助 没思路的话我们还是得继续查询 :h w ...
- C# 实现Eval(字符串表达式)的三种方法
一.背景 假如给定一个字符串表达式"-12 * ( - 2.2 + 7.7 ) - 44 * 2",让你计算结果,熟悉JavaScript的都知道有个Eval函数可以直接进行计算, ...
- 【转载】 新版 Kite为啥这么火,问就俩字『好用』
本文转自: https://blog.csdn.net/qq_28168421/article/details/102927311 ---------------------------------- ...
- A3C与GA3C的收敛性分析
G-A3C的代码: https://gitee.com/devilmaycry812839668/gpu_a3c 论文: <Reinforcement Learning thorugh Asyn ...
- 读论文《Reinforced Attention for Few-Shot Learning and Beyond》
2022年4月22日,实验室开组会,我讲了论文<Reinforced Attention for Few-Shot Learning and Beyond>,最近整理资料又再读了一遍,这里 ...
- 【EF Core】自动生成的字段值
自动生成字段值,咱们首先想到的是主键列(带 IDENTITY 的主键).EF Core 默认的主键配置也是启用 Identity 自增长的,而且可以自动标识主键.前提是代表主键的实体属性名要符合以下规 ...
- CF1697C
C. awoo's Favorite Problem 首先,检查两个字符串中所有字母的计数是否相同. 然后考虑下面的重述.字符串s中的字母 b是静止的.而字母a和c则在字符串中移动.第一种移动是将字母 ...