Catalog Service – 解析微软微服务架构实例代码

上一篇我们说了identity service,因为其基于identityserver4开发的,所以知识点不是很多,今天我们来看下catalog service,今后的讲解都会把不同的、重点的拿出来讲,希望大家明白。

源码分析

我们先看下它的目录结构,很标准的webapi目录:

image

首先看下Program,跟IdentityService类似,多了一个UseWebRoot(“Pics”),把pics这个目录设置成了webroot,其他都一样。

在Startup的构造方法中,我们也看到了使用了secret manager tool,但是多了一个参数,在这里我们看到的是Assembly类型,其实secret只需要其中的userSecretsId而已。

在ConfigureServices中,我们看到如下代码:

services.AddMvc(options =>{options.Filters.Add(typeof(HttpGlobalExceptionFilter));}).AddControllersAsServices();

添加了一个filter,这个HTtpGlobalExceptionFilter可以在项目中找到,大概的意思就是遇到抛出CatalogDomainException类型的错误时,返回特定的错误码。

AddControllersAsServices这个扩展方法是把项目中的Controller都注册到Services中,我们看下源码:

        public static IMvcCoreBuilder AddControllersAsServices(this IMvcCoreBuilder builder)        {            var feature = new ControllerFeature();            builder.PartManager.PopulateFeature(feature);foreach (var controller in feature.Controllers.Select(c => c.AsType()))            {                builder.Services.TryAddTransient(controller, controller);            }            builder.Services.Replace(ServiceDescriptor.Transient());return builder;        }

中间那段foreach就是,这样我们在项目中通过依赖注入方式都能方便的访问到各个controller了。

Going down:

            services.AddDbContext(options =>            {                options.UseSqlServer(Configuration["ConnectionString"],                                     sqlServerOptionsAction: sqlOptions =>                                     {                                         sqlOptions.MigrationsAssembly(typeof(Startup).GetTypeInfo().Assembly.GetName().Name);                                         //Configuring Connection Resiliency:   sqlOptions.EnableRetryOnFailure(maxRetryCount: 5, maxRetryDelay: TimeSpan.FromSeconds(30), errorNumbersToAdd: null);                                     });// Changing default behavior when client evaluation occurs to throw. // Default in EF Core would be to log a warning when client evaluation is performed.options.ConfigureWarnings(warnings => warnings.Throw(RelationalEventId.QueryClientEvaluationWarning));//Check Client vs. Server evaluation: });

对DBContext的配置的时候,这里使用了Connection Resiliency(弹回连接)的方式,其中可以看到使用migration的时候,它使用了MigrationsAssembly(AssemblyName),这种方式跟我之前讲的FluentNhibernate有点类似,EnableRetryOnFailure设置了这个Action的失败尝试机制,如果Migration的时候遇到Failure,就会自动重试,这种方式避免了app与database分离造成的连接偶尔失败造成的影响。为什么会有这个机制呢?因为当我们的database在云端的时候,比如Azure SQL,不可避免的会出现网络连接问题,即使我们把app和database放在一个数据中心中,我相信偶尔也会有这个问题,我们现在可以通过配置,使其如果遇到失败就会重新操作,一定程度避免了网络偶尔造成的问题。你也可以设置一些策略,使其能够在运行命令的时候能够进行重试EF默认情况下只是记录client evaluation中的warns,我们可以通过ConfigureWarnings使其抛出这个警告,你也可以配置成忽略。

接下来我们看到如下代码:

services.Configure(Configuration);

我们可以在eShop的各个项目中都能找到类似的语句,它会把一些项目相关的Settings注册到services中,使其成为环境变量,我们可通过setting.json进行配置。除了通过setting.json进行配置,我们还能通过Docker run –e 进行灵活化配置。

在这里我们的CatalogSetting含有一个ExternalCatalogBaseUrl属性,我们在docker run的时候可以输入如下命令:

docke run -e "ExternalCatalogBaseUrl=http://localhost:5011/" ....

这样就能灵活的通过docker命令进行配置了,非常方便,我们也可以通过-e对我们setting.json中的变量进行赋值,比如ConnectionString,你可以通过点击了解更多相关内容。

            // Add framework services.services.AddSwaggerGen();            services.ConfigureSwaggerGen(options =>            {                options.DescribeAllEnumsAsStrings();                options.SingleApiVersion(new Swashbuckle.Swagger.Model.Info()                {                    Title = "eShopOnContainers - Catalog HTTP API",                    Version = "v1",                    Description = "The Catalog Microservice HTTP API. This is a Data-Driven/CRUD microservice sample",                    TermsOfService = "Terms Of Service"                });            });            services.AddCors(options =>            {                options.AddPolicy("CorsPolicy",                    builder => builder.AllowAnyOrigin()                    .AllowAnyMethod()                    .AllowAnyHeader()                    .AllowCredentials());            });

上面两段代码,分别配置了SwaggerGen和Cors(跨域)策略,SwaggenGen是一个非常实用的框架,它能自动把我们的api转为web方式呈现在我们眼前,还能进行调试,非常好用。Cors的配置这里用的不好,它允许了所有请求,建议还是按照实际需求来吧,否则没有跨域设置的意义了。

接下来我们看到了一系列的add service的操作,都是关于EventBus的,稍微看了下,发现目前只做了log的动作,我们看下代码:

if (raiseProductPriceChangedEvent) // Save and publish integration event if price has changed{//Create Integration Event to be published through the Event Busvar priceChangedEvent = new ProductPriceChangedIntegrationEvent(catalogItem.Id, productToUpdate.Price, oldPrice);// Achieving atomicity between original Catalog database operation and the IntegrationEventLog thanks to a local transactionawait _catalogIntegrationEventService.SaveEventAndCatalogContextChangesAsync(priceChangedEvent);// Publish through the Event Bus and mark the saved event as publishedawait _catalogIntegrationEventService.PublishThroughEventBusAsync(priceChangedEvent);}

上面的代码意思是在价格有变动的时候,我们就调用EventService进行保存,同时对操作进行了记录。PublishThroughEventBusAsync方法则对这条记录的State更改为published。目前来说我不太清楚为何要用这种方式,也不知道为何取名为EventBus,不过我在项目的issue中已经提出了这个问题,希望项目的开发者们能给我一个答案。我有查看了Basket.Api,在这个项目中会有订阅行为,具体的等到下一章我们再仔细看看。

ok,我们再看下Configure方法,下面一段代码我们可以学习下:

var context = (CatalogContext)app            .ApplicationServices.GetService(typeof(CatalogContext));WaitForSqlAvailability(context, loggerFactory);

我们看到在这里它调用了之前注册的CatalogContext,它并没有通过new进行实例化,而是通过GetService的方式获取之前的注册,这样context所依赖的其他实例也一并带进来了,非常方便好用。

WaitForSqlAvailability方法是对数据库可用进行尝试,因为后面它需要进行数据迁移。

CatalogService包含了2个Controller,一个是PicController,一个是CatalogController,PicController仅仅是根据ID获取了图片,CatalogController展示了用webapi如何做CURD。

运行部署

如果你要运行Catalog.Api,你必须安装MSSQL和RabbitMQ,这次我把我的系统换成了Win10 Pro,并在电脑上使用Docker安装了MSSQL-Server-Linux和RabbitMQ。安装这2个非常简单,仅仅需要输入几条命令即可:

docker run --name mssql -e 'ACCEPT_EULA=Y' -e 'SA_PASSWORD=Pass@word' -p 5433:1433 -d microsoft/mssql-server-linuxdocker run -d --hostname my-rabbit --name rabbitmq -p 8080:15672 -p 5672:5672 rabbitmq:3-management

ok,我们使用docker创建了mssql和rabbitmq,这里注意一下,我把mssql的端口映射到了本机的5433上,还有rabbitmq的管理页面,我映射到了本机的8080端口,你可以通过http://localhost:8080 进行访问。

上一篇我们说过我们可以通过iisexpress/Kestrel或者docker的形式运行因为牵涉到配置,所以这两种方式的运行有些不同。

一、iisExpress或Kestrel方式下,因为刚刚我们把mssql和rabbitmq的端口都映射到了本机,所以我们只需要在setting.json中把数据库连接和rabbitmq的地址指向本机即可,如下:

{  "ConnectionString": "Server=tcp:127.0.0.1,5433;Initial Catalog=Microsoft.eShopOnContainers.Services.CatalogDb;User Id=sa;Password=Pass@word",  "ExternalCatalogBaseUrl": "http://localhost:5101",  "EventBusConnection": "localhost",  "Logging": {"IncludeScopes": false,"LogLevel": {      "Default": "Debug",      "System": "Information",      "Microsoft": "Information"}  }}

ok,Ctrl+F5,运行一下看看:

image

当看到上面这个页面,说明你的运行正常了,你还得测试下api是否运行正常,比如Pic,比如Items。

二、docker中运行,参照上一篇的方式,先publish再build image, 不过这里要注意一点,因为你之前的ConnectionString和EventBusConnection都是指向本机(127.0.0.1)的,所以这里必须改一下,改成主机的ip地址或者是对应容器的ip也可以,如果您不想更改的话,也可以通过docker -e进行设置,比如:

docker run -p 8899:80 --name catalog -e "EventBusConnection=172.17.0.2" -d catalog:01

我这里的172.17.0.2是我rabbitmq容器的ip地址,你可以通过docker inspect containerId 进行查看容器的ip。

如果一切配置都正确的话,你就可以通过浏览器http://localhost:8899 进行浏览了。

当然,除了正常浏览外,你还需测试下api是否正常。

catalog.api.demo

困惑

在这个项目中有一些疑惑,希望大家能够给我答案。

Connection Resiliency,我看了很久,字面意思是弹性连接,但我觉得用弹性好像不太适合,一般来讲我们说的弹性都是指架构或者系统的伸缩性,我一开始也是从这个角度去了解,但看了很多文章,觉得它只是让我们在启动的时候,设置一些重试策略,在后面调用中可使用此策略,策略会根据你设置的重试次数、延迟时间等去自动重试,避免因为偶尔的错误造成的影响,所以觉得用弹回比较恰当。

EventBus,我感觉很奇怪,为什么一定要取这个名字呢?在Android中,很明确的,它是进行订阅发布,消息传递,可以解耦发布者和订阅者,但在Catalog.Api里,变成了记录操作,没有看到解耦,也没有看到订阅。在我的理解中,应该在Startup进行订阅操作,发布者CatalogController在进行update操作的时候,订阅者进行add log动作,但在这个实例中,我看到的是同步进行了这些操作,所以很不解。

Mssql-server-linux,当你用Docker安装了以后,你却不能使用visual studio 2017的sql server data tools进行查询(只能进行连接),为了查看效果,还需要安装Microsoft Sql Server Management Studio(必须17版本以后)进行查看数据。

写在最后

这次的文章来的比较晚,一方面有点忙,另一方面就是上面提到的困惑,面对困惑我试着去解答,但有时候真的无法解答,所以提出来集思广益。

后面可能会比较慢,需要学习的东西真多,一边写一边学习成为这次系列的乐趣,现在每天坚持6公里快走,夜走能够是我保持头脑清晰,思考项目中的疑问,现在发觉生活越发有趣。

或许有很多人觉得只看了Startup就够了吗?其实真不够,我目前先把框架的源码过一遍,后面会分篇讲述,比如Connection Resiliency。

以上就是Catalog Service – 解析微软服务架构实例代码的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1432430.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 08:05:13
下一篇 2025年12月17日 08:05:29

相关推荐

  • 深入学习响应式布局框架:适合初学者到专家的详尽指南

    响应式布局框架解析:从初学者到专家的必备指南 随着移动设备的普及和多样化,响应式布局成为了现代Web设计的必备技能。响应式布局框架以其简单、灵活和可维护的特点,成为了开发者们的首选工具。然而,对于初学者来说,学习和理解响应式布局框架可能会感到有些困惑。本文将从初学者到专家,为您提供一个详细的指南,帮…

    2025年12月24日
    000
  • 揭秘静态定位的不同类型:揭示常见的静态定位类型详解

    静态定位类型大揭秘:解析常见的静态定位类型有哪些 引言: 随着移动互联网和定位技术的迅速发展,静态定位在现代生活中扮演了重要的角色。静态定位是指通过对目标物体或者人体的位置信息进行测量和计算,来确定其准确位置的技术。它在导航、地图应用、智能交通系统等方面具有广泛的应用。本文将为大家揭秘常见的静态定位…

    2025年12月24日
    000
  • 优化CSS解析过程中的回流和重绘技巧

    CSS回流和重绘解析及优化技巧 近年来,网页性能优化成为了前端开发中的重要环节,其中包括对CSS回流和重绘的解析及优化。在优化CSS的过程中,我们需要了解回流和重绘的定义,并学习一些具体的优化技巧。 什么是回流和重绘? 回流(reflow)和重绘(repaint)是浏览器渲染引擎对网页进行布局和绘制…

    2025年12月24日
    000
  • 探秘五种流行的CSS布局框架

    CSS布局框架大揭秘:五种流行框架解析 引言:在网页设计和开发过程中,如何实现良好的布局一直是一个重要的挑战。CSS布局框架通过提供强大的工具和组件,帮助我们更轻松地创建复杂的布局,提高开发效率。在本文中,我们将探讨五种流行的CSS布局框架,并提供具体的代码示例。 一、BootstrapBootst…

    2025年12月24日 好文分享
    000
  • 深入了解五种常见的CSS布局框架解析

    了解CSS布局框架:五种常见布局解析 在网页设计与开发中,CSS布局是一个非常重要的方面。一个好的布局可以使网页更加美观、功能更加完善。而了解CSS布局框架的知识,则能够帮助我们更好地掌握网页的布局技巧。本文将介绍五种常见的CSS布局,并提供具体的代码示例。 一、流式布局(流式定位) 流式布局是一种…

    2025年12月24日
    000
  • 揭秘CSS框架:常见框架解析及特点研究

    CSS框架大揭秘:解析常见的几种框架及其特点,需要具体代码示例 引言:在现代网页设计中,CSS框架扮演着重要的角色,它们能够极大地简化我们的开发工作,提高开发效率。本文将深入解析常见的几种CSS框架,并提供详细的代码示例,帮助读者更好地理解和应用这些框架。 一、Bootstrap:Bootstrap…

    2025年12月24日
    000
  • 解析CSS伪类和伪元素的常见用法和实例

    深入探讨CSS伪类和伪元素的常见用法和实例解析 在前端开发中,CSS是我们常用的样式设计语言之一。除了基本的选择器和属性,CSS还提供了一些特殊的选择器,称为伪类和伪元素。本文将深入探讨CSS伪类和伪元素的常见用法和实例解析,并附上具体的代码示例。 一、伪类的常见用法和实例解析 :hover伪类 :…

    2025年12月24日
    000
  • CSS3的学习轨迹和常见误区解析

    CSS3的学习轨迹和常见误区解析 引言:随着Web技术的不断发展,CSS3已经成为了前端工程师必备的技能之一。通过掌握CSS3的各种功能和特效,我们能够创建出更加丰富多彩的网页布局和交互效果。本文将介绍CSS3的学习轨迹,同时分析一些常见的误区,并提供一些代码示例。 一、学习轨迹: 1.掌握基本语法…

    2025年12月24日
    000
  • canvas线条的属性解析

    本文主要和大家介绍了canvas线条的属性详解的相关资料,小编觉得挺不错的,现在分享给大家,希望能帮助到大家。 一、线条的帽子lineCap 取值:butt(默认值),round圆头,square方头 var canvas=document.getElementById(“canvas”);canv…

    2025年12月24日 好文分享
    000
  • 对CSS继承的深度解析

      我酷爱模块化设计。长期以来我都热衷于将网站分离成组件,而不是页面,并且动态地将那些组件合并到界面上。这种做法灵活,高效并且易维护。   但是我不想我的设计看上去是由一些不相关的东西组成的。我是在创造一个界面,而不是一张超现实主义的照片。   很幸运的是,已经有一项叫做 CSS 的技术,就是特意设…

    2025年12月23日
    000
  • 解析CSS3中nth-child与nth-of-type的区别

    css3中nth-child与nth-of-type的区别其实很简单::nth-of-type为什么要叫:nth-of-type?因为它是以”type”来区分的。也就是说:ele:nth-of-type(n)是指父元素下第n个ele元素, 而ele:nth-child(n)是…

    2025年12月23日 好文分享
    000
  • 如何添加html运行框_在网页中添加HTML代码运行框【添加】

    可实时运行HTML代码的交互式编辑框可通过三种方案实现:一、CodeMirror+iframe动态执行,用document.write渲染;二、JSFiddle风格iframe沙箱,用srcdoc属性加载;三、Monaco Editor+Blob URL,提供专业编辑体验。 如果您希望在网页中嵌入一…

    2025年12月23日
    000
  • 优化@font-face配置:确保自定义字体在移动设备上的兼容性与显示

    本文深入探讨了自定义字体通过@font-face规则在移动设备上无法正确显示的问题,并提供了详细的解决方案。核心在于优化字体格式的声明顺序,强调优先使用ttf、woff2和woff等广泛支持的字体格式,以提升跨平台兼容性,确保自定义字体在包括各类移动设备在内的所有环境中均能稳定渲染。 理解@font…

    2025年12月23日
    000
  • 解决移动设备上 @font-face 字体不显示的兼容性指南

    移动设备上 @font-face 字体显示异常,常见原因是字体格式优先级配置不当。本文将深入探讨不同字体格式的兼容性,并提供一套优化后的 @font-face 声明最佳实践,确保自定义字体在各类设备和浏览器上都能稳定加载与显示。 在现代网页设计中,自定义字体通过 @font-face 规则提供了丰富…

    2025年12月23日
    000
  • vs中的html怎么运行环境_vs配置html运行环境步骤【指南】

    在 Visual Studio 中运行 HTML 需通过 Web 项目模板并借助 IIS Express 服务。1. 安装 VS 时勾选“ASP.NET 和 Web 开发”工作负载;2. 创建 ASP.NET 项目,添加 HTML 页面;3. 编写代码后设为起始页,点击 IIS Express 运行…

    2025年12月23日
    000
  • HTML文本在span中怎么设置样式_HTML文本在span中如何局部调整字体与颜色

    使用span标签结合style属性可对文本局部设置颜色、字体等样式;2. 推荐通过CSS类统一管理多处相同样式,提升可维护性;3. 中文建议设置Microsoft YaHei等常用字体并提供备选,颜色推荐十六进制值;4. 避免过多内联样式,保持结构与表现分离。 在HTML中,标签常用于对文本的局部进…

    2025年12月23日
    000
  • HTML文本与CSS怎么结合使用_HTML文本与CSS怎么结合实现丰富排版效果

    HTML提供结构,CSS负责样式,二者结合可实现美观布局。1. 内联样式通过style属性直接设置,适用于单元素调整但难复用;2. 内部样式表置于head中,用style标签定义,适合单页统一排版;3. 外部样式表将CSS独立为文件,通过link引入,利于多页共享与维护,推荐使用;4. 关键CSS属…

    2025年12月23日
    000
  • 网站根目录下的神秘HTML文件:识别与管理域名所有权验证文件

    网站根目录下发现的随机命名html文件,通常是用于验证域名或网站所有权的工具。本文将深入探讨这类文件的常见来源、作用及其在google search console等服务中的应用,并提供识别与管理建议,帮助网站管理员理解并妥善处理这些看似“多余”的文件。 1. 这类文件的神秘面纱 在维护或更新一个既…

    2025年12月23日
    000
  • Windows OneDrive同步HTML+CSS项目到多台电脑

    使用OneDrive同步HTML+CSS项目可行,需将项目放入OneDrive文件夹并登录同一账号实现多端同步。应采用英文命名、避免特殊字符,使用相对路径引用资源,如href=”css/style.css”。推荐结构:my-website/下分设css、js、images目录…

    2025年12月23日
    000
  • html代码怎么设置字体_html字体标签与CSS字体样式设置方法

    可通过HTML标签或CSS样式自定义网页文字效果:一、使用标签设置字体名称、大小、颜色;二、利用内联style属性设置font-family、font-size、color等;三、在中用定义内部样式表;四、通过link引入外部CSS文件实现样式分离;五、使用@font-face引入Web字体,确保跨…

    2025年12月23日
    000

发表回复

登录后才能评论
关注微信