微服務學習筆記(2)——使用Consul 實現 MagicOnion(GRpc) 服務註冊和發現

来源:https://www.cnblogs.com/hanfan/archive/2019/01/21/10299122.html
-Advertisement-
Play Games

1.下載打開Consul 筆者是windows下麵開發的(也可以使用Docker)。 官網下載windows的Consul https://www.consul.io/ 使用cmd視窗打開,輸入 訪問預設127.0.0.1:8500就可以看到界面化的Consul 2.在服務端註冊 接著上一篇 app ...


1.下載打開Consul

筆者是windows下麵開發的(也可以使用Docker)。
官網下載windows的Consul

https://www.consul.io/

使用cmd視窗打開,輸入consul agent -dev
訪問預設127.0.0.1:8500就可以看到界面化的Consul

2.在服務端註冊

接著上一篇

using Consul;
using Grpc.Core;
using GRPCServer.Entity;
using MagicOnion.Server;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using System;

namespace GRPCServer
{
    public class Startup
    {
        public Startup(IConfiguration configuration)
        {
            this.Configuration = configuration;
        }

        public IConfiguration Configuration { get; }

        // This method gets called by the runtime. Use this method to add services to the container.
        public void ConfigureServices(IServiceCollection services)
        {
            services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);

            MagicOnionServiceDefinition service = MagicOnionEngine.BuildServerServiceDefinition(new MagicOnionOptions(true)
            {
                MagicOnionLogger = new MagicOnionLogToGrpcLogger()
            });
            Server server = new Server
            {
                Services = { service },
                Ports = { new ServerPort(this.Configuration["Service:LocalIPAddress"], Convert.ToInt32(this.Configuration["Service:Port"]), ServerCredentials.Insecure) }
            };
            server.Start();

        }

        // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
        public void Configure(IApplicationBuilder app, IHostingEnvironment env, IApplicationLifetime lifetime)
        {
            if (env.IsDevelopment())
            {
                app.UseDeveloperExceptionPage();
            } 
            app.UseMvc(); 
             
            ServiceEntity serviceEntity = new ServiceEntity
            {
                IP = this.Configuration["Service:LocalIPAddress"],
                Port = Convert.ToInt32(this.Configuration["Service:Port"]),
                ServiceName = this.Configuration["Service:Name"],
                ConsulIP = this.Configuration["Consul:IP"],
                ConsulPort = Convert.ToInt32(this.Configuration["Consul:Port"])
            };
            var consulClient = new ConsulClient(x => x.Address = new Uri($"http://{serviceEntity.ConsulIP}:{serviceEntity.ConsulPort}"));//請求註冊的 Consul 地址
            var httpCheck = new AgentServiceCheck()
            {
                DeregisterCriticalServiceAfter = TimeSpan.FromSeconds(5),//服務啟動多久後註冊
                Interval = TimeSpan.FromSeconds(10),//健康檢查時間間隔,或者稱為心跳間隔
                HTTP = this.Configuration["Service:Examination"],//健康檢查地址
                Timeout = TimeSpan.FromSeconds(5)
            };
            var registration = new AgentServiceRegistration()
            {
                Checks = new[] { httpCheck },
                ID = Guid.NewGuid().ToString(),
                Name = serviceEntity.ServiceName,
                Address = serviceEntity.IP,
                Port = serviceEntity.Port,
                Tags = new[] { $"urlprefix-/{serviceEntity.ServiceName}" }//添加 urlprefix-/servicename 格式的 tag 標簽,以便 Fabio 識別
            }; 
            consulClient.Agent.ServiceRegister(registration).Wait();//服務啟動時註冊,內部實現其實就是使用 Consul API 進行註冊(HttpClient發起)
            lifetime.ApplicationStopping.Register(() =>
            {
                consulClient.Agent.ServiceDeregister(registration.ID).Wait();//服務停止時取消註冊
            }); 
        }
    }
}

appsettings.json

{
  "Service": {
    "Name": "Test3",
    "Port": "8083",
    "LocalIPAddress": "192.168.1.8",
    "Examination": "http://192.168.1.8:5000/api/Values"
  },
  "Consul": {
    "IP": "127.0.0.1",
    "Port": "8500"
  }
}
3.客戶端調用
using Consul;
using Grpc.Core;
using MagicOnion.Client;
using ServerDefinition;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;


var aaa= AvaliableServices("Test3","").Result;


public static async Task<ServiceEntry[]> AvaliableServices(string name, string tags)
        {
            var services = new List<ServiceEntry>();
            using (var client = new ConsulClient())
            {
                foreach (var tag in tags.Split(','))
                {
                    var result = await client.Health.Service(name, !string.IsNullOrEmpty(tag) ? tag : null, true).ConfigureAwait(false);
                    foreach (var item in result.Response)
                    {
                        if (!services.Any(service => service.Node.Address == item.Node.Address
                            && service.Service.Port == item.Service.Port))
                        {
                            services.Add(item);
                        }
                    } 
                }
                //交集處理,僅取出完全匹配服務
                foreach (var tag in tags.Split(','))
                {
                    if (string.IsNullOrEmpty(tag))
                    {
                        continue;
                    }
                    var alsorans = services.Where(service => !service.Service.Tags.Contains(tag)).ToList();
                    foreach (var alsoran in alsorans)
                    {
                        services.Remove(alsoran);
                    }
                }
            }
            return services.ToArray();
        }
4.思考

這個時候我就能通過'Test3'來獲得Test3的服務和介面。

但是我是使用的MagicOnion,還是沒辦法拿到我定義的方法SumAsync

怎麼辦?

1.引用ITest (讓微服務之間有引用,不太好)

2.使用網關

5.預告

下一篇我會想法辦法使他們能相互通訊(其實我還不知道怎麼搞)


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • package Collection; import java.util.ArrayList; import java.util.Scanner; /*集合類的特點: * 大小可變 * * ArrayList實現: * 大小可變數組的實現 * * * 這是泛型 * 怎麼用 * 在出現的地方可以使用引... ...
  •    如果你寫了一個python庫,想讓別人快速使用你的庫,最簡單的方式就是使用python官方出品的庫托管網站 "pypi" 了。    pypi的全稱是Python Package Index,是python的一個軟體倉庫。pypi可以 ...
  • HashSet為無序不可重覆集合。底層幾乎全部藉助HashMap實現,比較簡單。本篇簡要分析一下HashSet源碼。 首先是成員變數: 1、真正保存數據的HashMap實例 2、map實例的值 常用方法: 1、add() 從這個HashSet的add()方法中,可以看出。HashSet的不可重覆,主 ...
  • '''文件名:小雙雙文件內容如下: 昨夜寒蟬不住鳴。驚回千里夢,已三更。起來獨自繞階行。人悄悄,簾外月朧明。白首為功名,舊山松飾老,阻歸程。欲將心事付瑤琴。知音少,弦斷有誰聽。''' 1. 昨夜寒蟬不住鳴。 驚回千里夢,已三更。 2. ['昨夜寒蟬不住鳴。\n', '驚回千里夢,已三更。\n', ' ...
  • 1. 在使用linq過程DefaultIfEmpty的過程中如果 O.RS 這個支段的值是null,在取這個數據 就會報錯 ,正確的寫法 2. 在使用Linq 用where條件判斷要好分辨大小寫 3. Linq寫完後要在Tolist()或FirstOrDefault()結尾 ...
  • 介紹 一款輕量級開源的代碼生成器,相對較動軟代碼生成器而言要輕量的多,支持多種資料庫,所用到dll組件也都在github有源碼,代碼非常的簡單有點基礎的看源碼可以把生成的項目改成自已的風格。 特色 該代碼生成器最大的特點就三個簡單 ,無需安裝,生成的代碼 簡單並且有教學用例,還有就是調試和修改模版簡 ...
  • 說實話,到底應該怎麼開始我也沒譜。自己平日里也就是小打小鬧,弄點輔助性的軟體,多少的可以提高自己的工作效率就行,再加上周邊基本沒人擼碼,也就自己傻呵呵的擼啊擼。 廢話不說,先說重要的。 ...
  • 委托聲明 委托是一種可以指向方法的數據類型。 聲明委托格式:delegate 返回值類型 委托類型名(參數); 比如delegate void MyDelgate(int n); 委托實例 創建委托類型對象格式: 也可以寫成簡化形式: //編譯器會自動搞成new Mydel (SayHello);註 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...