ASP.NET Core 集成 React SPA 应用

AgileConfig的UI使用react重写快完成了。上次搞定了基于jwt的登录模式(AntDesign Pro + .NET Core 实现基于JWT的登录认证),但是还有点问题。现在使用react重写后,agileconfig成了个确确实实的前后端分离项目。那么其实部署的话要分2个站点部署,把前端build完的静态内容部署在一个网站,把server端也部署在一个站点。然后修改前端的baseURL让spa的api请求都指向server的网站。
这样做也不是不行,但是这不符合AgileConfig的精神,那就是简单。asp.net core程序本身其实就是一个http服务器,所以完全可以把spa网站使用它来承载。这样只需要部署一个站点就可以同时跑spa跟后端server了。
其实最简单的办法就是把build完的文件全部丢wwwroot文件夹下面。然后访问:

http://localhost:5000/index.html

但是这样我们的入口是index.html,这样看起来比较别扭,不够友好。而且这些文件直接丢在wwwroot的根目录下,会跟网站其他js、css等内容混合在一起,也很混乱。
那么下面我们就要解决这两个文件,我们要达到的目的有2个:

  1. spa的入口path友好,比如http://localhost:5000/ui

  2. spa静态文件存放的目录独立,比如存放在wwwroot/ui文件夹下,或者别的什么目录下。

要实现以上内容只需要一个自定义中间件就可以了。

wwwroot\ui

wwwroot\ui

我们把build完的静态文件全部复制到wwwroot\ui文件夹内,以跟其他静态资源进行区分。当然你也可以放在任意目录下,只要是能读取到就可以。

ReactUIMiddleware

namespace AgileConfig.Server.Apisite.UIExtension
{public class ReactUIMiddleware{private static Dictionary<string, string> _contentTypes = new Dictionary<string, string>{{".html", "text/html; charset=utf-8"},{".css", "text/css; charset=utf-8"},{".js", "application/javascript"},{".png", "image/png"},{".svg", "image/svg+xml"},{ ".json","application/json;charset=utf-8"},{ ".ico","image/x-icon"}};private static ConcurrentDictionary<string, byte[]> _staticFilesCache = new ConcurrentDictionary<string, byte[]>();private readonly RequestDelegate _next;private readonly ILogger _logger;public ReactUIMiddleware(RequestDelegate next,ILoggerFactory loggerFactory){_next = next;_logger = loggerFactory.CreateLogger<ReactUIMiddleware>();}private bool ShouldHandleUIRequest(HttpContext context){return context.Request.Path.HasValue && context.Request.Path.Value.Equals("/ui", StringComparison.OrdinalIgnoreCase);}private bool ShouldHandleUIStaticFilesRequest(HttpContext context){//请求的的Referer为 0.0.0.0/ui ,以此为依据判断是否是reactui需要的静态文件if (context.Request.Path.HasValue && context.Request.Path.Value.Contains(".")){context.Request.Headers.TryGetValue("Referer", out StringValues refererValues);if (refererValues.Any()){var refererValue = refererValues.First();if (refererValue.EndsWith("/ui", StringComparison.OrdinalIgnoreCase)){return true;}}}return false;}public async Task Invoke(HttpContext context){const string uiDirectory = "wwwroot/ui";//handle /ui requestvar filePath = "";if (ShouldHandleUIRequest(context)){filePath = uiDirectory + "/index.html";}//handle static files that Referer = xxx/uiif (ShouldHandleUIStaticFilesRequest(context)){filePath = uiDirectory + context.Request.Path;}if (string.IsNullOrEmpty(filePath)){await _next(context);}else{//output the file bytesif (!File.Exists(filePath)){context.Response.StatusCode = 404;return;}context.Response.OnStarting(() =>{var extType = Path.GetExtension(filePath);if (_contentTypes.TryGetValue(extType, out string contentType)){context.Response.ContentType = contentType;}return Task.CompletedTask;});await context.Response.StartAsync();byte[] fileData = null;if (_staticFilesCache.TryGetValue(filePath, out byte[] outfileData)){fileData = outfileData;}else{fileData = await File.ReadAllBytesAsync(filePath);_staticFilesCache.TryAdd(filePath, fileData);}await context.Response.BodyWriter.WriteAsync(fileData);return;}}}
}

大概解释下这个中间件的思路。这个中间件的逻辑大概是分量部分。
1.拦截请求的路径为/ui的请求,直接从ui文件夹读取index.html静态文件的内容然后输出出去,这就相当于直接访问/index.html。但是这样的路径形式看起来更加友好。
2.拦截react spa需要的静态资源文件,比如css文件,js文件等。这里比较麻烦,因为spa拉静态文件的时候path是直接从网站root开始的,比如http://localhost:5000/xxx.js,那么怎么区分出来这个文件是react spa需要的呢?我们判断一下请求的Referer头部,如果Referer的path是/ui,那么就说明是react spa需要的静态资源,同样从ui文件夹去读取。
这里还需要给每个response设置指定的contentType不然浏览器无法准确识别资源。

   public void Configure(IApplicationBuilder app, IWebHostEnvironment env, IServiceProvider serviceProvider){if (env.IsDevelopment()){app.UseDeveloperExceptionPage();}else{app.UseMiddleware<ExceptionHandlerMiddleware>();}app.UseMiddleware<ReactUIMiddleware>();......}

在Startup类的Configure方法内使用这个中间件。这样我们的改造就差不多了。

运行一下

访问下http://localhost:5000/ui 可以看到spa成功加载进来了。

总结

为了能让asp.net core承载react spa应用,我们使用一个中间件进行拦截。当访问对应path的时候从本地文件夹内读取静态资源返回给浏览器,从而完成spa所需要资源的加载。这次使用react spa来演示,其实换成任何spa应用都是一样的操作。
代码在这:ReactUIMiddleware

关注我的公众号一起玩转技术

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

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

相关文章

IBM在欧洲启动“智能云”研发计划

据国外媒体报道&#xff0c;IBM今日宣布将与15个欧洲合作伙伴启动一项联合研究项目&#xff0c;其中包括标准组织以及电信、医疗和媒体行业的企业一起开发一种基于目标的、标准化的“智能云”存储架构。 IBM表示&#xff0c;此次合作的目的是改善富媒体数据和存储服务跨国界跨厂…

etcd php,etcd集群备份和数据恢复

etcd是一个分布式k-v数据库&#xff0c;在kubernetes中使用其管理集群的元数据。这里介绍etcd集群数据的备份和数据恢复的方法和步骤。快照定期备份crontab定期执行备份脚本&#xff0c;每半小时备份一次&#xff0c;本地、异地都备份(暂定&#xff1a;本地备份保留最近10个备份…

Ingress-nginx工作原理和实践

本文记录/分享 目前项目的 K8s 部署结构和请求追踪改造方案这个图算是一个通用的前后端分离的 k8s 部署结构:Nginx Ingress 负责暴露服务(nginx前端静态资源服务)&#xff0c; 根据十二要素应用的原 则&#xff0c;将后端 api 作为 nginx 服务的附加动态资源。Ingress vs Ingre…

oracle 查看用户状态,Oracle数据库查看用户状态

一、当前ORACLE用户的状态可查看视图DBA_USERS;一般情况下在使用的正常用户均处于OPEN状态。select username,account_status from dba_users;二、ORACLE数据库用户有多种状态&#xff0c;可查看视图USER_ASTATUS_MAP。select * from user_astatus_map;九种状态可分为两类&…

Dotnet洋葱架构实践

一个很清晰的架构实践&#xff0c;同时刨刨MySQL的坑。一、洋葱架构简介洋葱架构出来的其实有一点年头了。大约在2017年下半年&#xff0c;就有相关的说法了。不过&#xff0c;大量的文章在于理论性的讨论&#xff0c;而我们今天会用一个项目来完成这个架构。洋葱架构&#xff…

显示桌面,原来这么简单的

显示桌面的按钮其实很简单,我也是无意中发现的,代码如下,复制到记事本中,另存为.scf文件,文件名随便去,双击即可看看效果,^_^[Shell]Command2IconFileexplorer.exe,3[Taskbar]CommandToggleDesktop转载于:https://www.cnblogs.com/crid/archive/2010/11/28/5172732.html

php 面向对象 教程,PHP学习笔记之面向对象设计_PHP教程

面向对象设计是php程序开发中一个很重要的内容块&#xff0c;如果你想成为高级php程序员我们必须知道面向对象设计具体用法与写法。维护简单 模块化是面向对象编程中的一个特征。实体被表示为类和同一名字空间中具有相同功能的类&#xff0c;我们可以在名字空间中添加一个类而不…

EF Core3.0+ 通过拦截器实现读写分离与SQL日志记录

前言本文主要是讲解EF Core3.0 通过拦截器实现读写分离与SQL日志记录注意拦截器只有EF Core3.0 支持,2.1请考虑上下文工厂的形式实现.说点题外话..一晃又大半年没更新技术博客..唉,去年一年发生了太多事情..博主真的 一言难尽..有兴趣的可以去看看:记录一下,也许是转折,也许是结…

oracle clob 查询换行,sqoop clob从Oracle导入到hive   回车换行导致记录增多

sqoop clob从Oracle导入到hive 回车换行导致记录增多发布时间&#xff1a;2020-06-27 02:45:01来源&#xff1a;51CTO阅读&#xff1a;3159作者&#xff1a;jackwxhsqoop import --hive-import --hive-overwrite --connect jdbc:oracle:thin:192.168.92.136:1521:cyporcl --…

对于scanf的使用一点体会心得

今天非常的突发气象的在acm上面做了一下题目&#xff0c;悲剧的是多年不用c的人忘记了怎么样的使用scanf了&#xff0c;今天还学到了一点东西。 题目里面提示了输入两个数&#xff0c;规定第1&#xff5e;6列是第一个数的范围&#xff0c;第8&#xff5e;9列是第二个数的范围。…

毕业二十年,为什么人和人之间的差距那么大?

这是头哥侃码的第237篇原创最近天气逐渐转暖&#xff0c;身边的各种聚会也开始多了起来。找个周末&#xff0c;朋友之间喝点小酒&#xff0c;或者跟高中同学来一场久违的重逢&#xff0c;重温着曾经的回忆&#xff0c;加深着彼此之间的感情&#xff0c;想必都是不错的选择。什么…

oracle查询案例,2道经典的oracle查询案例

第一题&#xff1a;第一题&#xff1a;直接贴代码&#xff1a;select Id,Name,Money,(select Money from test1 a where a.Id decode(b.Id - 1,0,null,b.Id-1)) Money1 from test1 b;经典的子查询&#xff0c;注意的就是null值的处理问题&#xff0c;decode或者case是oracle很…

在一起的时候很好&#xff0c;离开后就感觉像是路上的陌生人&#xff0c;没有交点的平行线&#xff0c;也许它们在很遥远的地方交汇了&#xff0c;也许是下辈子吧&#xff01; 你对我的好&#xff0c;我不是很清楚&#xff0c;你是伪装的&#xff0c;其实用的着那么辛苦吗&…

oracle 12c 多线程,Oracle 12c(12.1)中性能优化功能增强之通过参数THREADED_EXECTION使用多线程模型...

1. 后台UNIX/Linux系统上&#xff0c;Oracle用多进程模型。例如&#xff1a;linux上一个常规安装的数据库会有如下进程列&#xff1a;$ ps -ef | grep [o]ra_oracle 15356 1 0 10:53 ? 00:00:00 ora_pmon_db12coracle 15358 1 0 10:53 ? 00:00:00 o…

使用工作单元UnitOfWork实现事务

概述工作单元模式有如下几个特性&#xff1a;1、使用同一上下文2、跟踪实体的状态3、保障事务一致性。工作单元用来维护一个由已经被业务事物修改的业务对象组成的列表。Unit Of Work模式负责协调这些修改的持久化工作以及所有标记的并发问题。在数据访问层中采用Unit Of Work…

MyEclipse配置Tomcat(图解)

1&#xff09; 安装完MyEclipse后&#xff0c;在Eclpise的菜单栏可看到MyEclipse一项&#xff0c; 下面的工具栏中可看到MyEclipse的Tomcat图标2&#xff09; 进行MyEclipse的配置&#xff0c; 从菜单栏中进入“Windows --> Preferences”3) 先要进行JDK的配置&#x…

Python难懂?买一次西瓜就懂了!

什么是code?code就就是一种语言&#xff0c;一种计算机能读懂的语言。计算机是一个傻*&#xff0c;他理解不了默认两可的任何东西。比如&#xff0c;你让你老公去买个西瓜&#xff0c;你老公会自己决定去哪里买&#xff0c;买几个&#xff0c;找个搞活动打折的买&#xff0c;总…

制作oracle11g yum源,利用安装盘简单制作yum源

第一步&#xff1a;创建一个新文件夹,并挂载光驱.[rootlocalhost ~]# mkdir /yum[rootlocalhost ~]# mount /dev/cdrom /mnt/cdrom第二步&#xff1a;安装工具包&#xff1a;[rootlocalhost ~]# rpm -ivh /mnt/cdrom/CentOS/createrepo-0.4.11-3.el5.noarch.rpm第三步&#xff…

安全删除U盘时,遇到提示“无法停止‘通用卷’设备时解决方法

安全删除U盘时&#xff0c;大家常常会遇到提示“无法停止‘通用卷’设备&#xff0c;请稍候再停止该设备。” 这种情况下可以强行拔下U盘吗&#xff1f;当然不可以&#xff01;这时候如果强行拔除的话&#xff0c;很容易损坏计算机U口或者你的U盘。如果你的U盘上有重要的资料&a…

宠粉老鱼皮带你 “入坑” GitHub!

昨天刚刚教大家 如何更快地访问 GitHub&#xff0c;按照惯例&#xff0c;今天不得来一个 GitHub 教程&#xff1f;最近&#xff0c;鱼皮在持续完善自己 GitHub 上的开源项目&#xff0c;也分享给了很多小伙伴&#xff0c;苦苦哀求大家可以给个 star。liyupi 的 GitHub但是&…