[TOC]html
今年4月份的时候,和平台组的同事一块儿调研了一下Nacos,也就在那个时候写了.net core版本的非官方版的SDK。mysql
虽然公司内部因为某些缘由最后没有真正的用起来,但不少人仍是挺看好的。在和镇汐大大沟通后,决定写一篇博客简单介绍一下。git
下面这个图,就是本文的重点了。github
Nacos是一个易于构建云原生应用的动态服务发现、配置管理和服务管理平台,它提供了一组简单易用的特性集,帮助咱们快速实现动态服务发现、服务配置、服务元数据及流量管理。算法
它有下面的关键特性sql
特性仍是挺多的,也有挺多值的挖掘的地方。有关Nacos的更多信息能够访问下面的地址:docker
下面就开始正题了,第一步确定是先把Nacos跑起来。数据库
因为是演示,因此直接用docker启动了Standalone Mysql
模式的。json
git clone --depth 1 https://github.com/nacos-group/nacos-docker.git cd nacos-docker docker-compose -f example/standalone-mysql.yaml up
运行docker-compose后,会先拉取几个镜像回来,而后就看到下面的输出,基本就是正常启动了。api
打开浏览器访问 http://localhost:8848/nacos
就能够看到Nacos控制台的登陆界面了。
初始的用户名和密码都是 nacos,登陆进来以后大概是这样的。
能够看到运行起来的Nacos,版本是1.1.3,还有清晰可见的几个大菜单,这些都是能够很方便咱们去进行管理的。
那咱们就先来看一下Nacos的配置管理吧。
在上面的特性大图中,已经很明确的告诉了咱们配置管理的几个重要功能。
在配置中有几个比较重要的概念须要先了解一下。
先添加下面这个nuget包,而后看一下这个配置要怎么玩。
dotnet add package nacos-sdk-csharp-unofficial
还有必不可少的就是在Startup
里面进行配置。
public void ConfigureServices(IServiceCollection services) { // configuration services.AddNacos(configure => { // default timeout configure.DefaultTimeOut = 8; // nacos's endpoint configure.ServerAddresses = new System.Collections.Generic.List<string> { "localhost:8848" }; // namespace configure.Namespace = ""; // listen interval configure.ListenInterval = 1000; }); services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2); }
这个也算是比较常见的配置了,就很少说了,还能够经过配置文件来加载配置。
这些配置里面,其实最主要的就是Nacos的地址。
先来看看最简单的获取配置信息。
SDK中提供了一个名为INacosConfigClient
的Client接口,这个接口里面的全部内容都是操做配置相关的。
[Route("api/[controller]")] [ApiController] public class ConfigController : ControllerBase { private readonly INacosConfigClient _configClient; public ConfigController(INacosConfigClient configClient) { _configClient = configClient; } // GET api/config?key=demo1 [HttpGet("")] public async Task<string> Get([FromQuery]string key) { var res = await _configClient.GetConfigAsync(new GetConfigRequest { DataId = key, Group = "DEFAULT_GROUP", //Tenant = "tenant" }) ; return string.IsNullOrWhiteSpace(res) ? "Not Found" : res; } }
上面获取配置的这个获取配置的方法,大意就是 读取默认命名空间(public)下面的DEFAULT_GROUP
这个配置分组下面的,名为key的配置Id的值。
若是咱们输入的key,在Nacos上面没有,那个这个方法就会返回 Not Found
给调用方,若是有,那就会返回具体的配置值。
因为咱们是刚运行起行,什么都没有操做,因此确定是没有任何配置信息的。
那咱们就先添加一个,看看效果如何。
一样在上面的控制器中加入下面的发布配置的方法,一样也是经过INacosConfigClient
来添加配置。
// GET api/config/add?key=demo1&value=123 [HttpGet("add")] public async Task<string> Add([FromQuery]string key, [FromQuery]string value) { var res = await _configClient.PublishConfigAsync(new PublishConfigRequest { DataId = key, Group = "DEFAULT_GROUP", //Tenant = "tenant" Content = value }); return res.ToString(); }
这个时候咱们已经添加成功了。
\回去控制台,也能够看到刚才加的配置已经出来了。
再一次访问获取配置信息的接口,就已经能够拿到对应的配置内容了。
下面经过控制台去修改一下配置的内容。
点发布按钮的时候,会有一个比较页面,让咱们对比先后修改了那些内容。
这个时候咱们经过INacosConfigClient
去访问的话,发现是获取不到咱们刚才更新的内容的。
这个是由于,从Nacos读取配置成功后,会写入配置信息到本地缓存中,后面访问的话会优先去读缓存的内容。
那么要怎么作到有人修改了配置内容后,它能实时生效呢?其实很简单,只须要添加一下对配置的监听就能够了。
这个得益于Nacos容许咱们监听配置,以便实时感知配置变动。若是配置变动,则用获取配置接口获取配置的最新值,动态刷新本地缓存。
下面是一个简单的示例,这里用的是BackgroundService
来处理的。
public class ListenConfigurationBgTask : BackgroundService { private readonly ILogger _logger; private readonly INacosConfigClient _configClient; public ListenConfigurationBgTask(ILoggerFactory loggerFactory, INacosConfigClient configClient) { _logger = loggerFactory.CreateLogger<ListenConfigurationBgTask>(); _configClient = configClient; } protected override async Task ExecuteAsync(CancellationToken stoppingToken) { // Add listener await _configClient.AddListenerAsync(new AddListenerRequest { DataId = "demo1", //Group = "DEFAULT_GROUP", //Tenant = "tenant", Callbacks = new List<Action<string>> { x => { _logger.LogInformation($" We found something changed!!! {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")} [{x}]"); }, } }); } public override async Task StopAsync(CancellationToken cancellationToken) { // Remove listener await _configClient.RemoveListenerAsync(new RemoveListenerRequest { DataId = "demo1", Callbacks = new List<Action> { () => { _logger.LogInformation($" Removed listerner "); }, } }); await base.StopAsync(cancellationToken); } }
这里其实没有什么内容,就是在程序启动的时候添加一下监听,而后在程序退出的时候,一样也退出监听。
不要忘记在Startup中加下面的代码,这样配置的监听才会生效!
services.AddHostedService<ListenConfigurationBgTask>();
当咱们添加监听以后,修改了配置文件的内容,它就能够动态的更新加载了。
一样的,控制台里面也有监听的记录,能够在监听查询里面找到。
下面是具体的程序日志输出
配置的每一次修改,都会有历史记录,能够从历史版本里面找到。
除了能看历史的记录,还能够回滚到指定的版本,这是个颇有用的功能。
在数据库中,配置信息的保存是这样的
还有一个删除配置的方法,这里就不介绍了,都是差很少的用法,不过正常状况下是不该该删除配置的,除非是多余的。
关于Nacos配置管理的介绍就先到这里了,有兴趣的朋友能够继续去深究。
下面咱们就来看看Nacos的服务发现。
关于服务注册和发现,听的比较多的大概就是,consul, eureka, etcd , k8s 等等。
思路其实都差很少,在服务启动的时候,把当前服务的相关信息注册上去,而后要调用某个服务的时候,就获取这个服务下面的列表,而后选一个可用的进行访问。最后就是当服务中止的时候,咱们要注销当前的服务。
目前这个SDK提供了两种形式,一种是原始的API,一种是对原始API进行了封装,能够直接注册和发现相应的下游服务。
原始的API在一个名为INacosNamingClient
的Client接口中提供,这个接口里面的全部内容都是服务发现相关的。
不过在这里只介绍封装事后的使用方法,固然也能够本身根据原始的API进行封装处理。
首先要添加下面这个nuget包。
dotnet add package nacos-sdk-csharp-unofficial.AspNetCore
先起来一个服务。
先在配置文件appsettings.json
中添加下面的内容
{ "nacos": { "ServerAddresses": [ "localhost:8848" ], "DefaultTimeOut": 15, "Namespace": "", "ListenInterval": 1000, "ServiceName": "BaseService", "Weight": 10 } }
这个配置主要表达了,这个实例的服务名是 BaseService
, 权重是10
, Nacos的地址是 localhost:8848
。
而后在Startup中把当前实例注册到Nacos。
namespace BaseService { using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.Hosting; using Microsoft.AspNetCore.Mvc; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using Nacos.AspNetCore; public class Startup { public Startup(IConfiguration configuration) { Configuration = configuration; } public IConfiguration Configuration { get; } public void ConfigureServices(IServiceCollection services) { // important step services.AddNacosAspNetCore(Configuration); services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2); } public void Configure(IApplicationBuilder app, IHostingEnvironment env) { if (env.IsDevelopment()) { app.UseDeveloperExceptionPage(); } app.UseMvc(); // important step app.UseNacosAspNetCore(); } } }
这里只须要简单的配置这两个地方就能够完成服务的注册功能了!!
下面就启动这个程序。
能够看到在启动程序的时候,当前实例就会向Nacos发送心跳,心跳的里面包含了IP和端口等信息。
回到控制台,咱们能够看到这个服务如今已经有一个实例了。
再启动一个同服务名的实例,这里只对接口返回的内容作了一下调整,其余都是同样的!
这个时候点进服务的详情里面,能够看到更加具体的信息。
服务如今是已经注册上来了,下面咱们就再来一个服务去调用上面这个注册好的服务。
Startup中的内容都是差很少的,不一样的是,若是肯定服务不被内部其它应用调用的话,能够不注册到Nacos上面。
public class Startup { public Startup(IConfiguration configuration) { Configuration = configuration; } public IConfiguration Configuration { get; } public void ConfigureServices(IServiceCollection services) { services.AddHttpClient(); services.AddNacosAspNetCore(Configuration); services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2); } public void Configure(IApplicationBuilder app, IHostingEnvironment env) { if (env.IsDevelopment()) { app.UseDeveloperExceptionPage(); } app.UseMvc(); //app.UseNacosAspNetCore(); } }
而后就是发现服务了。
INacosServerManager
里面提供了一个只根据服务名来获取健康的实例的地址信息。不足的地方就是忽略了命名空间和集群这些参数,会考虑在后面的版本中加上吧。
这里获取到的地址信息是随机取出来的,最简单的轮训算法。。获取到一次全部的实例地址信息后会缓存10秒钟,这10秒钟里面就会直接从缓存中的地址信息取一个。
[Route("api/[controller]")] [ApiController] public class ValuesController : ControllerBase { private readonly INacosServerManager _serverManager; private readonly IHttpClientFactory _clientFactory; public ValuesController(INacosServerManager serverManager, IHttpClientFactory clientFactory) { _serverManager = serverManager; _clientFactory = clientFactory; } // GET api/values [HttpGet] public async Task<string> GetAsync() { var result = await GetResultAsync(); if (string.IsNullOrWhiteSpace(result)) { result = "ERROR!!!"; } return result; } private async Task<string> GetResultAsync() { var baseUrl = await _serverManager.GetServerAsync("BaseService"); if (string.IsNullOrWhiteSpace(baseUrl)) { return ""; } var url = $"{baseUrl}/api/values"; var client = _clientFactory.CreateClient(); var result = await client.GetAsync(url); return await result.Content.ReadAsStringAsync(); } }
效果就来看动图了。
在两个实例的健康状态都是true的时候,会随机调用一个实例。
当把其中一个实例停掉的时候,这个实例的健康状态就会被标识为false,这个时候就不会调用到这个false的实例。
当把这个实例从新运行以后,又恢复到随机调用的状况。
Nacos的服务发现除了上面介绍的,还有系统开关,数据指标,集群信息等功能,有待去深刻挖掘。
Nacos使用起来不算复杂,算是比较容易上手的,用的公司也挺多的了。
还有个把 steeltoe 和Nacos结合起来的项目 skynet-cloud 也能够看看。
文中的示例代码能够戳这里 NacosDemo
SDK的地址 nacos-sdk-csharp
但愿感兴趣的大佬给个星星,也十分但愿有大佬来一块儿维护这个项目,和提些建议。
由于是第一次写SDK类的东西,参考了其余平台提供.NET的SDK,而后结合Nacos的Open API写的,有可能会有很多遗漏和bug,还请各位大佬多多包涵。
原文出处:https://www.cnblogs.com/catcher1994/p/11489052.html