Spring Cloud Zuul的動態路由怎樣做?集成Nacos實現很簡單

来源:https://www.cnblogs.com/zlt2000/archive/2019/08/29/11428272.html
-Advertisement-
Play Games

一、說明 網關的核心概念就是路由配置和路由規則,而作為所有請求流量的入口,在實際生產環境中為了保證高可靠和高可用,是儘量要避免重啟的,所以實現動態路由是非常有必要的;本文主要介紹實現的思路,並且以 為數據源來講解   二、實現要點 要實現動態路由只需關註下麵4個點 1. 網關啟動時, 的數 ...


一、說明

網關的核心概念就是路由配置和路由規則,而作為所有請求流量的入口,在實際生產環境中為了保證高可靠和高可用,是儘量要避免重啟的,所以實現動態路由是非常有必要的;本文主要介紹實現的思路,並且以Nacos為數據源來講解

 

二、實現要點

要實現動態路由只需關註下麵4個點

  1. 網關啟動時,動態路由的數據怎樣載入進來
  2. 靜態路由動態路由以那個為準,ps:靜態路由指的是配置文件里寫死的路由配置
  3. 監聽動態路由的數據源變化
  4. 數據有變化時怎樣通知zuul刷新路由

 

三、具體實現

3.1. 實現動態路由的數據載入

  • 重寫SimpleRouteLocator類的locateRoutes方法,此方法是載入路由配置的,父類中是獲取properties中的路由配置,可以通過擴展此方法,達到動態獲取配置的目的
  • 這裡採用靜態路由動態路由共存,相同路由id以動態路由優先覆蓋的實現方式

AbstractDynRouteLocator類可查看:AbstractDynRouteLocator.java

public abstract class AbstractDynRouteLocator extends SimpleRouteLocator implements RefreshableRouteLocator {
    private ZuulProperties properties;

    public AbstractDynRouteLocator(String servletPath, ZuulProperties properties) {
        super(servletPath, properties);
        this.properties = properties;
    }

    /**
     * 刷新路由
     */
    @Override
    public void refresh() {
        doRefresh();
    }

    @Override
    protected Map<String, ZuulRoute> locateRoutes() {
        LinkedHashMap<String, ZuulRoute> routesMap = new LinkedHashMap<>();
        // 從application.properties中載入靜態路由信息
        routesMap.putAll(super.locateRoutes());
        // 從數據源中載入動態路由信息
        routesMap.putAll(loadDynamicRoute());
        // 優化一下配置
        LinkedHashMap<String, ZuulRoute> values = new LinkedHashMap<>();
        for (Map.Entry<String, ZuulRoute> entry : routesMap.entrySet()) {
            String path = entry.getKey();
            // Prepend with slash if not already present.
            if (!path.startsWith("/")) {
                path = "/" + path;
            }
            if (StringUtils.hasText(this.properties.getPrefix())) {
                path = this.properties.getPrefix() + path;
                if (!path.startsWith("/")) {
                    path = "/" + path;
                }
            }
            values.put(path, entry.getValue());
        }
        return values;
    }

    /**
     * 載入路由配置,由子類去實現
     */
    public abstract Map<String, ZuulRoute> loadDynamicRoute();
}

由於動態路由的數據可以有很多種途徑,如:Nacos、Redis、Zookeeper、DB等,所以這裡定義一個抽象類,由具體的實現類去定義loadDynamicRoute方法

 

3.2. Nacos路由實現類

NacosDynRouteLocator類完整的代碼實現可查看:NacosDynRouteLocator.java

3.2.1. 實現loadDynamicRoute方法獲取動態數據

    @Override
    public Map<String, ZuulProperties.ZuulRoute> loadDynamicRoute() {
        Map<String, ZuulRoute> routes = new LinkedHashMap<>();
        if (zuulRouteEntities == null) {
            zuulRouteEntities = getNacosConfig();
        }
        for (ZuulRouteEntity result : zuulRouteEntities) {
            if (StrUtil.isBlank(result.getPath()) || !result.isEnabled()) {
                continue;
            }
            ZuulRoute zuulRoute = new ZuulRoute();
            BeanUtil.copyProperties(result, zuulRoute);
            routes.put(zuulRoute.getPath(), zuulRoute);
        }
        return routes;
    }
        
    private List<ZuulRouteEntity> getNacosConfig() {
        try {
            String content = nacosConfigProperties.configServiceInstance().getConfig(ZUUL_DATA_ID, ZUUL_GROUP_ID,5000);
            return getListByStr(content);
        } catch (NacosException e) {
            log.error("listenerNacos-error", e);
        }
        return new ArrayList<>(0);
    }

3.2.2. 增加NacosListener監聽路由數據變化

    private void addListener() {
        try {
            nacosConfigProperties.configServiceInstance().addListener(ZUUL_DATA_ID, ZUUL_GROUP_ID, new Listener() {
                @Override
                public Executor getExecutor() {
                    return null;
                }

                @Override
                public void receiveConfigInfo(String configInfo) {
                    //賦值路由信息
                    locator.setZuulRouteEntities(getListByStr(configInfo));
                    RoutesRefreshedEvent routesRefreshedEvent = new RoutesRefreshedEvent(locator);
                    publisher.publishEvent(routesRefreshedEvent);
                }
            });
        } catch (NacosException e) {
            log.error("nacos-addListener-error", e);
        }
    }

註意路由數據變化後不需要自己手動刷新路由,只需要給zuul發送一個RoutesRefreshedEvent事件即可,zuul自己有個ZuulRefreshListener類會監聽事件幫我們刷新路由

 

3.3. 配置類創建NacosDynRouteLocator的Bean

DynamicZuulRouteConfig可查看:NacosDynRouteLocator.java

@Configuration
@ConditionalOnProperty(prefix = "zlt.gateway.dynamicRoute", name = "enabled", havingValue = "true")
public class DynamicZuulRouteConfig {
    @Autowired
    private ZuulProperties zuulProperties;

    @Autowired
    private DispatcherServletPath dispatcherServletPath;

    /**
     * Nacos實現方式
     */
    @Configuration
    @ConditionalOnProperty(prefix = "zlt.gateway.dynamicRoute", name = "dataType", havingValue = "nacos", matchIfMissing = true)
    public class NacosZuulRoute {
        @Autowired
        private NacosConfigProperties nacosConfigProperties;

        @Autowired
        private ApplicationEventPublisher publisher;

        @Bean
        public NacosDynRouteLocator nacosDynRouteLocator() {
            return new NacosDynRouteLocator(nacosConfigProperties, publisher, dispatcherServletPath.getPrefix(), zuulProperties);
        }
    }
}

這裡通過自定義配置來控制是否開啟動態路由功能

 

3.4. 添加Nacos路由配置

file
新增配置項:

  • Data Id:zuul-routes
  • Group:ZUUL_GATEWAY
  • 配置內容:
[
    {
        "enabled":true,
        "id":"csdn",
        "path":"/csdn/**",
        "retryable":false,
        "stripPrefix":true,
        "url":"https://www.csdn.net/"
    }, {
        "enabled":true,
        "id":"github",
        "path":"/github/**",
        "retryable":false,
        "stripPrefix":true,
        "url":"http://github.com/"
    }
]

添加兩條路由數據

 

四、測試

  • 啟動網關通過/actuator/routes端點查看當前路由信息
    file

    可以看到靜態路由和Nacos里配置的兩條路由信息並存顯示

  • 修改Nacos配置,關閉csdn路由
    file
  • 刷新查看網關的路由信息
    file

    csdn的路由已經看不到了,實現了動態改變路由配置

 
推薦閱讀

 
請掃碼關註我的公眾號
file


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

-Advertisement-
Play Games
更多相關文章
  • 偷偷和你們說,我搞了一份內部資料,該內部資料共有13張PPT,據作者透露,該PPT至少花了整整1周時間才編寫完成,其內容簡潔明瞭,內容深度足夠,易於初學者理解,也給深度開發人員分享了不一樣的消息隊列的玩法。特別重要的是,該架構目前已大面積的穩定應用於生產環境。 ...
  • Python 入門之 文件 1、文件操作 找到文件位置 雙擊打開 進行一些操作 r—read(讀) w—write(寫) a—追加 rd 讀位元組 wd 清空寫,寫位元組 ad 追加寫(位元組) r+ 讀寫 w+ 寫讀 a+ 追加讀 關閉文件 open() : 打開 ,open通過python控制操作系統 ...
  • 壓測的時候,經常需要使用動態參數,即每次請求的參數不一樣,這樣才能更好的模擬真實的用戶請求 我們首先能想到的也是最常見的就是用戶ID 1. 用戶參數 我們可以定義一個變數,然後指定多個值,這樣每次請求的時候都會順序迴圈去取這些值。通過${variable}取值。 具體參見官方文檔,上面寫的比較詳盡, ...
  • Java I/O文件 文件記憶體 》程式 》記憶體鍵盤 控制台 文件 什麼是文件? 相關記錄或放在一起的數據的集合 文件一般存儲在哪裡? 磁碟,u盤,移動硬碟 Java程式如何訪問文件屬性? JAVA API:java.io.File類 File類訪問文件屬性 File類 》創建文件對象 》操作文件或目 ...
  • SAP壓縮excel產生zip文件後作為附件發送mail案例. ...
  • 近期練習flask寫個blog, 安裝flask擴展時 報ERROR: Could not install packages due to an EnvironmentError: [Errno 13] Permission denied: '/Library/Python/2.7/site-pac ...
  • effective java 3th 序,java語言的簡單與複雜,以及本書出現的理由 ...
  • 如何解決驗證碼的問題,用什麼模塊,聽過哪些人工打碼平臺? PIL、pytesser、tesseract模塊 平臺的話有:(打碼平臺特殊,不保證時效性) 雲打碼 掙碼 斐斐打碼 若快打碼 超級鷹本文首發於Python黑洞網,博客園同步更新 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...