常见跨域解决方案以及Ocelot 跨域配置

常见跨域解决方案以及Ocelot 跨域配置

Intro

我们在使用前后端分离的模式进行开发的时候,如果前端项目和api项目不是一个域名下往往会有跨域问题。今天来介绍一下我们在Ocelot网关配置的跨域。

什么是跨域

跨域:

浏览器对于javascript的同源策略的限制,例如a.cn下面的js不能调用b.cn中的js,对象或数据(因为a.cn和b.cn是不同域),所以跨域就出现了.

上面提到的,同域的概念又是什么呢??? 简单的解释就是相同域名,端口相同,协议相同

同源策略:

请求的url地址,必须与浏览器上的url地址处于同域上,也就是域名,端口,协议相同.

比如:我在本地上的域名是study.cn,请求另外一个域名一段数据

640?wx_fmt=png

这个时候在浏览器上会报错:

640?wx_fmt=png

这个就是同源策略的保护,如果浏览器对javascript没有同源策略的保护,那么一些重要的机密网站将会很危险~

study.cn/json/jsonp/jsonp.html

当协议、子域名、主域名、端口号中任意一个不相同时,都算作不同域。不同域之间相互请求资源,就算作“跨域”。

请求地址形式结果
http://study.cn/test/a.html同一域名,不同文件夹成功
http://study.cn/json/jsonp/jsonp.html同一域名,统一文件夹成功
http://a.study.cn/json/jsonp/jsonp.html不同域名,文件路径相同失败
http://study.cn:8080/json/jsonp/jsonp.html同一域名,不同端口失败
https://study.cn/json/jsonp/jsonp.html同一域名,不同协议失败

跨域几种常见的解决方案

解决跨域问题有几种常见的解决方案:

跨域资源共享(CORS)

通过在服务器端配置 CORS 策略即可,每门语言可能有不同的配置方式,但是从本质上来说,最终都是在需要配置跨域的资源的Response上增加允许跨域的响应头,以实现浏览器跨域资源访问,详细可以参考MDN上的这篇CORS介绍

JSONP

JSONP 方式实际上返回的是一个callbak,通常为了减轻web服务器的负载,我们把js、css,img等静态资源分离到另一台独立域名的服务器上,在html页面中再通过相应的标签从不同域名下加载静态资源,而被浏览器允许,基于此原理,我们可以通过动态创建script,再请求一个带参网址实现跨域通信。

原生实现:

  1. <script>

  2. var script = document.createElement('script');

  3. script.type = 'text/javascript';


  4. // 传参一个回调函数名给后端,方便后端返回时执行这个在前端定义的回调函数

  5. script.src = 'http://www.domain2.com:8080/login?user=admin&callback=handleCallback';

  6. document.head.appendChild(script);


  7. // 回调执行函数

  8. function handleCallback(res) {

  9. alert(JSON.stringify(res));

  10. }

  11. </script>

服务端返回如下(返回时即执行全局函数):

  1. handleCallback({"status": true, "user": "admin"})

jquery ajax

  1. $.ajax({

  2. url: 'http://www.domain2.com:8080/login',

  3. type: 'get',

  4. dataType: 'jsonp', // 请求方式为jsonp

  5. jsonpCallback: "handleCallback", // 自定义回调函数名

  6. data: {}

  7. });

后端 node.js 代码示例:

  1. var querystring = require('querystring');

  2. var http = require('http');

  3. var server = http.createServer();


  4. server.on('request', function(req, res) {

  5. var params = qs.parse(req.url.split('?')[1]);

  6. var fn = params.callback;


  7. // jsonp返回设置

  8. res.writeHead(200, { 'Content-Type': 'text/javascript' });

  9. res.write(fn + '(' + JSON.stringify(params) + ')');


  10. res.end();

  11. });


  12. server.listen('8080');

  13. console.log('Server is running at port 8080...');

JSONP 只支持 GET 请求

代理

前端代理

在现代化的前端开发的时候往往可以配置开发代理服务器,实际作用相当于做了一个请求转发,但实际请求的api地址是没有跨域问题的,然后由实际请求的api服务器转发请求到实际的存在跨域问题的api地址。

angular 配置开发代理可以参考 angular反向代理配置

后端代理

后端可以通过一个反向代理如(nginx),统一暴露一个服务地址,然后为所有的请求设置跨域配置,配置 CORS 响应头,Ocelot是ApiGateway,也可以算是api的反向代理,但不仅仅如此。

Ocelot 跨域配置

示例代码:

  1. app.UseOcelot((ocelotBuilder, pipelineConfiguration) =>

  2. {

  3. // This is registered to catch any global exceptions that are not handled

  4. // It also sets the Request Id if anything is set globally

  5. ocelotBuilder.UseExceptionHandlerMiddleware();

  6. // Allow the user to respond with absolutely anything they want.

  7. if (pipelineConfiguration.PreErrorResponderMiddleware != null)

  8. {

  9. ocelotBuilder.Use(pipelineConfiguration.PreErrorResponderMiddleware);

  10. }

  11. // This is registered first so it can catch any errors and issue an appropriate response

  12. ocelotBuilder.UseResponderMiddleware();

  13. ocelotBuilder.UseDownstreamRouteFinderMiddleware();

  14. ocelotBuilder.UseDownstreamRequestInitialiser();

  15. ocelotBuilder.UseRequestIdMiddleware();

  16. ocelotBuilder.UseMiddleware<ClaimsToHeadersMiddleware>();

  17. ocelotBuilder.UseLoadBalancingMiddleware();

  18. ocelotBuilder.UseDownstreamUrlCreatorMiddleware();

  19. ocelotBuilder.UseOutputCacheMiddleware();

  20. ocelotBuilder.UseMiddleware<HttpRequesterMiddleware>();

  21. // cors headers

  22. ocelotBuilder.Use(async (context, next) =>

  23. {

  24. if (!context.DownstreamResponse.Headers.Exists(h => h.Key == HeaderNames.AccessControlAllowOrigin))

  25. {

  26. var allowedOrigins = Configuration.GetAppSetting("AllowedOrigins").SplitArray<string>();

  27. context.DownstreamResponse.Headers.Add(new Header(HeaderNames.AccessControlAllowOrigin, allowedOrigins.Length == 0 ? new[] { "*" } : allowedOrigins));

  28. context.DownstreamResponse.Headers.Add(new Header(HeaderNames.AccessControlAllowHeaders, new[] { "*" }));

  29. context.DownstreamResponse.Headers.Add(new Header(HeaderNames.AccessControlRequestMethod, new[] { "*" }));

  30. }

  31. await next();

  32. });

  33. })

  34. .Wait();

这里扩展了一个 Ocelot pipeline 的配置,这样我们可以直接很方便的直接在 Startup 里配置 Ocelot 的请求管道。

核心代码:

  1. ocelotBuilder.Use(async (context, next) =>

  2. {

  3. if (!context.DownstreamResponse.Headers.Exists(h => h.Key == HeaderNames.AccessControlAllowOrigin))

  4. {

  5. var allowedOrigins = Configuration.GetAppSetting("AllowedOrigins").SplitArray<string>();

  6. context.DownstreamResponse.Headers.Add(new Header(HeaderNames.AccessControlAllowOrigin, allowedOrigins.Length == 0 ? new[] { "*" } : allowedOrigins));

  7. context.DownstreamResponse.Headers.Add(new Header(HeaderNames.AccessControlAllowHeaders, new[] { "*" }));

  8. context.DownstreamResponse.Headers.Add(new Header(HeaderNames.AccessControlRequestMethod, new[] { "*" }));

  9. }

  10. await next();

  11. });

在 HttpRequester 中间件后面添加这个中间件在响应中增加跨域请求头配置,这里先判断了一下下面的api有没有配置,如果已经配置则不再配置,使用下游api的跨域配置,这样一来,只需要在网关配置指定的允许跨域访问的源即使下游api没有设置跨域也是可以访问了

需要说明一下的是如果想要这样配置需要 Ocelot 13.2.0 以上的包,因为之前 HttpRequester 这个中间件没有调用下一个中间件,详见 https://github.com/ThreeMammals/Ocelot/pull/830

Reference

  • https://developer.mozilla.org/zh-CN/docs/Web/HTTP/AccesscontrolCORS

  • https://segmentfault.com/a/1190000011145364

  • https://github.com/ThreeMammals/Ocelot/pull/830

.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com 

640?wx_fmt=jpeg


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

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

相关文章

E - Flow Gym - 102471E

E - Flow Gym - 102471E 题意&#xff1a; n个点&#xff0c;m条边&#xff0c;从点1到点n有k条相同长度的路径&#xff0c;每个边都有对应的容量&#xff0c;你可以进行任意次操作&#xff0c;每次将一个边的容量-1&#xff0c;将另一个边的容量1&#xff0c;问最少操作多少…

解读大内老A的《.NET Core框架本质》

老A说的一句话让我很受启发&#xff0c;想要深入了解框架&#xff0c;你要把精力聚焦在架构设计的层面来思考问题。而透彻了解底层原理&#xff0c;最好的笨办法就是根据原理对框架核心进行重建或者说再造。看起来没有捷径&#xff0c;也是最快的捷径。相信很多读者已经看过老A…

2019 ICPC Asia-East Continent Final

2019 ICPC Asia-East Continent Final 题号题目知识点ACity贪心BBlack and WhiteCDirichlet kkk-th rootDFireEFlow思维FGameGHappinessHKing随机思维IMoonJPermutationKAll Pair Maximum FlowLTravelMValue状压贪心

Mono和.NET Core 从比翼双飞到合体

.NET 开源之路就是 Mono和.NET Core 从比翼双飞到合体&#xff1a;2001年12月-2002年2月。一个新的平台诞生了。与惠普、英特尔和其他公司一起, 创建了 ECMA-335 标准, 该标准定义了支持多种编程语言的公共语言基础结构&#xff0c;C# 和 Visual Basic. Net。 F # 于2007年晚些…

7-3 打怪升级 (25 分)

7-3 打怪升级 (25 分) 题意&#xff1a; 很多游戏都有打怪升级的环节&#xff0c;玩家需要打败一系列怪兽去赢取成就和徽章。这里我们考虑一种简单的打怪升级游戏&#xff0c;游戏规则是&#xff0c;给定有 N 个堡垒的地图&#xff0c;堡垒之间有道路相连&#xff0c;每条道路…

Skywalking部署常见问题以及注意事项

Skywalking部署常见问题以及注意事项IntroSkyWalking 创建与2015年&#xff0c;提供分布式追踪功能。从5.x开始&#xff0c;项目进化为一个完成功能的Application Performance Management系统。他被用于追踪、监控和诊断分布式系统&#xff0c;特别是使用微服务架构&#xff0c…

CF1486B Eastern Exhibition

CF1486B Eastern Exhibition 题意&#xff1a; 二维平面上有 n 个点&#xff0c;要找一个点&#xff0c;使得所有点到它的曼哈顿距离&#xff08; x 和 y 的坐标差距之和&#xff09;之和最小。请问有几个满足该要求的点&#xff1f; 题解&#xff1a; 我们先考虑一维的情况…

GitHub推出包管理服务,npm与Nuget全支持

GitHub 今天推出了一项名为 GitHub Package Registry 的新产品&#xff0c;它提供了软件包管理服务&#xff0c;开发者通过它可发布公共或私有软件包。官方介绍&#xff0c;GitHub Package Registry 完全集成在 GitHub 中&#xff0c;因此和 repo 一样&#xff0c;用户可以使用…

Codeforces Round #708 (Div. 2)

Codeforces Round #708 (Div. 2) 题号题目知识点AMeximization思维BM-arrays思维C1k-LCM (easy version)构造C2k-LCM (hard version)构造DGenius思维构造E1Square-free division (easy version)质因子分解E2Square-free division (hard version)线性dp

CF1497E1 Square-free division (easy version)

CF1497E1 Square-free division (easy version) 题意&#xff1a; 这是简单版&#xff0c;此题中 k0 给出一串长为 n 的序列 a1,a2,a3...ana_1,a_2,a_3...a_na1​,a2​,a3​...an​ 把它分成尽量少的块使每一块中任意两数的乘积不是一个完全平方数。 输出最少的块数。 题解&…

发布dotNetCore程序到Kubernetes

上一篇《Mac中搭建Kubernetes》介绍了怎样在Mac中搭建单节点的Kubernetes&#xff0c;本文将编写一个dotNetCore的示例程序&#xff0c;并发布到Kubernetes中。环境基本步骤创建dotnetCore示例项目&#xff1b;本地搭建私有registry&#xff0c;或者使用DockerHub&#xff0c;本…

CF1497E2 Square-free division (hard version)

CF1497E2 Square-free division (hard version) 题意&#xff1a; 数组 a 由 n 个正整数构成。你需要将它们分割成最小数量的连续子段&#xff0c;使得每一个子段中的任意两个数&#xff08;不同位置&#xff09;的乘积不为完全平方数。 除此之外&#xff0c;你被允许在分割之…

[Cake] 2. dotnet 全局工具 cake

在上篇博客[Cake] 1. CI中的Cake中介绍了如何在CI中利用Cake来保持与CI/CD环境的解耦。当时dotnet 2.1还未正式发布&#xff0c;dotnet 还没有工具的支持&#xff0c;使得安装cake非常麻烦。不过随着 dotnet tool 的加入&#xff0c;这一问题得到了很好的解决。目前安装cake&am…

CF1497C k-LCM

CF1497C1 k-LCM (easy version) CF1497C2 k-LCM (hard version) 题意&#xff1a; 给定一个整数 n&#xff0c;请找到 k 个和为 n 的正整数a1,a2,....,aka_1,a_2,....,a_ka1​,a2​,....,ak​&#xff0c;使得lcma1,a2,....,ak<n2lcm{a_1,a_2,....,a_k}<\frac{n}{2}lcm…

官博翻译 | .NET Core 即 .NET 的未来

点击上方蓝字关注“汪宇杰博客”文 / Scott Hunter译 / 汪宇杰我们在2014年11月推出了.NET Core 1.0。.NET Core 的目标是借鉴我们过去12年构建、发布和服务.NET Framework的经验去构建更好的产品。这些改进如&#xff1a;并行安装&#xff08;您可以安装新版本&#xff0c;而…

CF1497D Genius

CF1497D Genius 题意&#xff1a; n个问题从i到n编号&#xff0c;第i个问题给出的ci2i,tagi,sic_i2^i,tag_i,s_ici​2i,tagi​,si​ 解决问题i后解决问题j条件是&#xff1a;IQ<|ci−cjc_i-c_jci​−cj​|,同时获得|si−sjs_i-s_jsi​−sj​|分 问题解决得次数和顺序不受限…

微软发布ML.NET 1.0,可一键添加机器学习模型

今天&#xff0c;我们很高兴宣布发布 ML.NET 1.0。ML.NET 是一个免费的、跨平台的开源机器学习框架&#xff0c;旨在将机器学习&#xff08;ML&#xff09;的强大功能引入.NET 应用程序。ML.NET GitHub&#xff1a;https://github.com/dotnet/machinelearning入门 http://dot.…

Codeforces Round #703 (Div. 2)

Codeforces Round #703 (Div. 2) 题号题目知识点AShifting Stacks思维BEastern Exhibition思维C1Guessing the Greatest (easy version)二分C2Guessing the Greatest (hard version)二分DMax Median思维二分EPaired PaymentFPairs of Paths

目前下载VS2017你可能会遇到这个坑

可能现在大伙都已经开始使用VS2019进行开发了。VS2019的下载使用也都很简单。由于工作需要&#xff0c;今天要在笔记本上安装VS2017,结果发现&#xff0c;VS2017的下载变得不是那么容易了&#xff0c;官方的下载方式也隐藏的很深&#xff0c;来来回回折腾了好一会才下载下来&am…

CF1486D Max Median

CF1486D Max Median 题意&#xff1a; 给定一个长度为 n 的序列 a&#xff0c;求所有长度 ≥k 的连续子序列中&#xff0c;中位数的最大值。定义中位数是一个长度为 x 的序列升序排序后的第 ⌊x12⌋\left\lfloor\frac{x1}{2}\right\rfloor⌊2x1​⌋位的值。 题解&#xff1a…