asp.net core中负载均衡场景下http重定向https的问题

上周欣喜地发现,微软官方终于针对 asp.net core 在使用负载均衡的情况下从 http 强制重定向至 https 的问题提供了解决方法。

app.UseForwardedHeaders(new ForwardedHeadersOptions
{ForwardedHeaders = ForwardedHeaders.XForwardedProto
});var options = new RewriteOptions().AddRedirectToHttpsPermanent();
app.UseRewriter(options);

但实际使用之后,欣喜变成了失望 —— 微软对这个问题的认识角度和我们不一样,造成这个方法对我们不适用,不得不继续使用我们的土方法。

为什么会这样?请看下面的分解。 

AddRedirectToHttpsPermanent 早就在 BasicMiddleware 的 RedirectToHttpsRule 中实现了,它的逻辑很简单 —— 判断当前请求是否是https,如果不是就进行重定向。

if (!context.HttpContext.Request.IsHttps)
{    //...}

这个直接了当的判断在使用负载均衡的场景下不仅不会发挥应有的作用,而且会产生致命的副作用 —— 让请求进入重定向死循环(ERR_TOO_MANY_REDIRECTS)。因为不管客户端的请求是 http 还是 https ,负载均衡与后端服务器之间始终是 http(当然你可以用https,但那是吃饱了撑着还浪费粮食)。如果负载均衡不额外提供这个信息,在后端服务器的眼里始终只有 http 没有 https ,http 重定向 https 根本无法实现。

从负载均衡的角度,为了解决这个问题,通常会通过一个另外的专用的请求头抓发这个信息,它的名字叫"X-Forwarded-Proto"。

从 asp.net core 的角度,要解决这个问题,需要弥补 Request.IsHttps 与 X-Forwarded-Proto 之间的鸿沟。于是微软实现了上面的 app.UseForwardedHeaders() ,实际是由 ForwardedHeadersMiddleware 完成这个任务 —— 根据 X-Forwarded-Proto 设置 Scheme(Request.IsHttps 就是基于 Scheme 进行判断的)。

if (checkProto && i < forwardedProto.Length)
{    set.Scheme = forwardedProto[forwardedProto.Length - i - 1];
}

到此为止,微软完美地解决了这个问题,RedirectToHttpsRule 不用修改1行代码。

但是在实际使用时,我们发现一个大问题,大到我们必须弃用这个看似完美的解决方法。

微软解决 http to https 问题的思路是这样:只要请求不是 https 的,就强制跳转到 https(这个没问题),其他一概不管,不管这个请求是不是来自负载均衡转发的(这个不够贴心)。

而我们要解决的问题是:只有在负载均衡转发的原始请求是 http 的情况下,才强制跳转至 https 。比如在服务器本机访问,比如来自其他docker容器的访问,如果这也跳转,那每台服务器(或者docker容器)都要部署https证书,多麻烦。

一个是只要不是 https ,就跳转;一个是只有是转发的 http ,才跳转。 就是因为这个对问题理解的差异,我们不得不放弃采用微软的官方解决方法,继续使用我们不太优雅的土方法。

RedirectToProxiedHttpsRule

public class RedirectToProxiedHttpsRule : RedirectToHttpsRule
{ 
public RedirectToProxiedHttpsRule(){
base.StatusCode = StatusCodes.Status301MovedPermanently;
base.SSLPort = null;}

public override void ApplyRule(RewriteContext context){
var key = "X-Forwarded-Proto";
var request = context.HttpContext.Request;
if (request.Headers.ContainsKey(key)){
if (request.Headers[key].FirstOrDefault() == "http"){
base.ApplyRule(context);}}} }


RewriteOptionsExtensions


public static class RewriteOptionsExtensions
{   
public static RewriteOptions AddRedirectForwardedHttpToHttps(this RewriteOptions options){options.Rules.Add(new RedirectToProxiedHttpsRule());
return options;} }


在 Startup 中使用

var options = new RewriteOptions().AddRedirectForwardedHttpToHttps();
app.UseRewriter(options);

原文地址:http://www.cnblogs.com/dudu/p/7044923.html


.NET社区新闻,深度好文,微信中搜索dotNET跨平台或扫描二维码关注

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/325131.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

《走遍中国》珍藏版(七)

Console.WriteLine("上一本书已经看完&#xff0c;从今天开始看下一本"); System.out.println("《走遍中国》珍藏版"); System.out.println("这本书是关于地理的"); System.out.println("地理往往是和历史进行紧密联系的");鸣沙山群峰…

JavaFX UI控件教程(四)之Button

翻译自 Button Button通过JavaFX API提供的类使开发人员能够在用户单击按钮时处理操作。该Button班是的扩展Labeled类。它可以显示文本&#xff0c;图像或两者。图3-1显示了具有各种效果的按钮。在本章中&#xff0c;您将学习如何创建这些按钮类型。 图3-1按钮类型 创建一个按…

角色管理

角色管理 MySQL8新增的功能 – 创建角色 role create role ‘dev_read’,‘dev_write’ – 给角色分配权限 grant select on mydb.* to ‘dev_read’ grant insert,update,delete on mydb.* to ‘dev_write’ – 创建用户 create user ‘user1’‘localhost’ identified b…

ssl1056-金明的预算方案【dp之有依赖的背包】

这道题卡了挺久的QAQ现在才搞定 Description   金明今天很开心&#xff0c;家里购置的新房就要领钥匙了&#xff0c;新房里有一间金明自己专用的很宽敞的房间。更让他高兴的是&#xff0c;妈妈昨天对他说&#xff1a;“你的房间需要购买哪些物品&#xff0c;怎么布置&#x…

《走遍中国》珍藏版(八)

Console.WriteLine("上一本书已经看完&#xff0c;从今天开始看下一本"); System.out.println("《走遍中国》珍藏版"); System.out.println("这本书是关于地理的"); System.out.println("地理往往是和历史进行紧密联系的");天地之间&a…

.Net Core应用搭建的分布式邮件系统设计

本篇分享的是由NetCore搭建的分布式邮件系统&#xff0c;主要采用NetCore的Api和控制台应用程序&#xff0c;由于此系统属于公司的所以这里只能分享设计图和一些单纯不设计业务的类或方法&#xff1b; 为什么要在公司中首例采用NetCore做开发 为什么要在公司中首例采用NetCore…

Java标识符与命名规则

标识符 标识符 Java 对各种变量、方法和类等要素命名时使用的字符序列称为标识符 技巧&#xff1a;凡是自己可以起名字的地方都叫标识符。 定义合法标识符规则 由26个英文字母大小写&#xff0c;0-9 &#xff0c;_或$组成 数字不可以开头。 不可以使用关键字和保留字&am…

JavaFX UI控件教程(五)之Radio Button

翻译自 Radio Button 本章讨论单选按钮控件和RadioButton类&#xff0c;该类的一个专门实现ToggleButton。 可以选择或取消选择单选按钮控件。通常&#xff0c;单选按钮组合成一个组&#xff0c;其中一次只能选择一个按钮。此行为将它们与切换按钮区分开来&#xff0c;因为组…

查询练习

简单查询 ①求js9901班的学生学号和姓名。 select s_no,s_name from student where class_no ‘js9901’ ②求计算机系的所有班级号和班级名。 select class_no,class_name from class where class_dept ‘计算机系’ ③求选修了课程的学生学号。 select s_no from choice ④…

《走遍中国》珍藏版(九)

Console.WriteLine("上一本书已经看完&#xff0c;从今天开始看下一本"); System.out.println("《走遍中国》珍藏版"); System.out.println("这本书是关于地理的"); System.out.println("地理往往是和历史进行紧密联系的");那一天&…

ssl1236-逃亡准备【dp之多重背包】

Description在《Harry Potter and the Deathly Hallows》中&#xff0c;Harry Potter他们一起逃亡&#xff0c;现在有许多的东西要放到赫敏的包里面&#xff0c;但是包的大小有限&#xff0c;所以我们只能够在里面放入非常重要的物品&#xff0c;现在给出该种物品的数量、体积、…

.NET开源MSSQL、Redis监控产品Opserver之Redis配置

Redis监控数据实例的加载可以查看Opserver.Core项目data/Redis文件夹下的RedisModule.cs,我加了点注释 /// <summary>/// 加载Redis连接 /// </summary>/// <returns></returns>private static List<RedisConnectionInfo> LoadRedisConn…

使用变量注意事项

Java中每个变量必须先声明&#xff0c;后使用(否则会编译失败) 使用变量名来访问这块区域的数据 变量的作用域&#xff1a;其定义所在的一对{ }内 变量只有在其作用域内才有效 同一个作用域内&#xff0c;不能定义重名的变量

《走遍中国》珍藏版(十)

Console.WriteLine("上一本书已经看完&#xff0c;从今天开始看下一本"); System.out.println("《走遍中国》珍藏版"); System.out.println("这本书是关于地理的"); System.out.println("地理往往是和历史进行紧密联系的");三峡是万里…

JavaFX UI控件教程(六)之Toggle Button

翻译自 Toggle Button 在本章中&#xff0c;您将了解ToggleButton该类&#xff0c;这是通过JavaFX API提供的另一种类型的按钮。 可以将两个或多个切换按钮组合成一个组&#xff0c;其中一次只能选择一个按钮&#xff0c;或者不需要选择。图5-1是组合了三个切换按钮的应用程…

ssl1377-竞赛真理【dp之分组背包】

其实这题很简单&#xff0c;我也不想多讲&#xff0c;但没办法老师要我们发博客╭(╯^╰)╮ Description TENSHI在经历了无数次学科竞赛的失败以后&#xff0c;得到了一个真理&#xff1a;做一题就要对一题&#xff01;但是要完全正确地做对一题是要花很多时间&#xff08;包括…

配置mybatis

配置mybatis.xml <?xml version"1.0" encoding"UTF-8" ?> <!DOCTYPE configurationPUBLIC "-//mybatis.org//DTD Config 3.0//EN""http://mybatis.org/dtd/mybatis-3-config.dtd"><configuration><environmen…

窥探ASP.Net MVC底层原理 实现跨越Session的分布式TempData

Hi,guys!Long time no see! 1、问题的引出 我相信大家在项目中都使用过TempData&#xff0c;TempData是一个字典集合&#xff0c;一般用于两个请求之间临时缓存数据或者页面之间传递消息。也都知道TempData是用Session来实现的&#xff0c;既然是用Session来实现的&#xff0…

某同学正为自己安装不上sqlserver数据库而愁眉苦脸,使用朋友给的方法顿时喜笑颜开,那么朋友到底出了个什么样的方法呢?...

小故事引入&#xff1a;张同学下一节课就要开始学习使用jdbc访问Sql Server数据库了&#xff0c;部分同学由于刚换电脑&#xff0c;导致没有安装数据库&#xff0c;于是同学们按照老师给的方法&#xff08;文末提供老师的方法&#xff09;&#xff0c;将5G多的Sqlserver的安装包…