欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Dubbo之降級Mock源碼分析

 更新時間:2023年09月11日 16:01:59   作者:土豆肉絲蓋澆飯  
這篇文章主要為大家介紹了Dubbo降級Mock源碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

限流,熔斷,降級

通俗的講,降級可以形容為一些事件A的觸發(fā)導(dǎo)致發(fā)生取代原有事件B的事件C,而熔斷和限流就是這些事件A的其中之一。

熔斷是異常情況發(fā)生的保護(hù)措施,比如調(diào)用供應(yīng)商1的短信API失敗,改為調(diào)用供應(yīng)商2的短信API。

限流是防止異常情況發(fā)生的措施,比如搶票時候拋出的提示。

一句口訣,消費(fèi)者(使用)熔斷,提供者(使用)限流。

上面也講了熔斷和限流是觸發(fā)降級其中兩個事件,那么在dubbo中我們?nèi)绾斡|發(fā)降級(mock)?

  • 通過服務(wù)治理觸發(fā)(dubbo-admin)
  • 配置接口mock參數(shù)觸發(fā)

Dubbo中降級的使用

這邊的講解從通過配置接口mock參數(shù)觸發(fā)降級的方式入手,其實服務(wù)治理也是相當(dāng)于通過Override機(jī)制給接口增加了mock參數(shù)配置。

mock表達(dá)式格式

[force/fail] (return xxx |throw xxx | fail | true |default | {mockClassName} ) | false 

  • 如果表達(dá)式為false,表示不啟用降級功能
  • 如果表達(dá)式以force為前綴,表示強(qiáng)制降級
  • 如果表達(dá)以fail為前綴或者沒有前綴,表示失敗降級

去除force或fail前綴后,剩下的表達(dá)式表示具體的降級邏輯,邏輯如下

mock表達(dá)式作用
true/default/fail調(diào)用接口+Mock類對應(yīng)的方法
{mockClass}調(diào)用${mockClass}對應(yīng)方法
return xxx解析xxx為方法返回類型的對象返回
throw namespace...xxxException拋出對應(yīng)xxxException異常

其中對于xxx,支持基本類型,Collection,Map以及自定義對象

具體示例如下

示例類型
empty返回各種類型的空實現(xiàn)
null直接返回null
true/false返回對應(yīng)布爾類型
`abc`,abc根據(jù)方法返回類型解析,可能是String或枚舉
{...}根據(jù)方法返回類型解析,可能是map或自定義對象
[...]根據(jù)方法返回類型解析,可以是Collection中的任意一種

如何配置mock表達(dá)式

mock表達(dá)式的配置方式一共有3種

  • 通過xml或@Reference的mock字段
  • Dubbo Admin控制臺
    -在zk設(shè)置Override動態(tài)配置

后兩種的原理都是對zk節(jié)點(diǎn)進(jìn)行操作。

首先講下第一種配置方式

<dubbo:reference id="demoService" check="false" interface="com.alibaba.dubbo.demo.DemoService" mock="return scj">
            <dubbo:parameter key="sayHello.mock" value="return scj" />
 </dubbo:reference>

因為spring對xml的配置是不支持引號,方括號,尖括號等符號的,滿足的格式如以下正則

private static final Pattern PATTERN_NAME_HAS_SYMBOL = Pattern.compile("[:*,\\s/\\-._0-9a-zA-Z]+");

所以通過項目中配置的mock行為是有限的,因此在項目中推薦配置失敗降級,通過一個Mock類來實現(xiàn)我們的降級邏輯。

而通過Dubbo-admin控制臺去配置mock,更適合強(qiáng)制降級的場景。

通過zk觸發(fā)強(qiáng)制降級示例代碼如下

create -e /dubbo/com.alibaba.dubbo.demo.DemoService/configurators/override%3a%2f%2f10.111.27.41%3a20880%2fcom.alibaba.dubbo.demo.DemoService%3fsayHello.mock%3dforce%3areturn+%60scj+mock+hhh%60%26category%3dconfigurators 1

具體這個配置是如何生效的,可以看下我寫的ZookeeperRegistry這篇文章。

源碼分析

首先是找入口,入口類為MockClusterWrapper

public class MockClusterWrapper implements Cluster {
    private Cluster cluster;
    public MockClusterWrapper(Cluster cluster) {
        this.cluster = cluster;
    }
    @Override
    public <T> Invoker<T> join(Directory<T> directory) throws RpcException {
        return new MockClusterInvoker<T>(directory,
                this.cluster.join(directory));
    }
}

這個類是一個SPI自動包裝類,包裝(AOP)的mock邏輯會強(qiáng)制觸發(fā)。

MockClusterWrapper的join方法會返回MockClusterInvoker,MockClusterInvoker會包裝實際cluster生成的ClusterInvoker,以此來實現(xiàn)對mock邏輯的注入。

我們看下MockClusterInvoker#invoke方法增加了什么邏輯。

public Result invoke(Invocation invocation) throws RpcException {
        Result result = null;
        //獲取directoryUrl,會包括客戶端和Configuration節(jié)點(diǎn)url的合并url
        //從method.xxx獲取參數(shù),如果沒有,直接從xxx參數(shù)獲取
        String value = directory.getUrl().getMethodParameter(invocation.getMethodName(), Constants.MOCK_KEY, Boolean.FALSE.toString()).trim();
        if (value.length() == 0 || value.equalsIgnoreCase("false")) {
            //如果沒有mock配置,不走mock邏輯
            result = this.invoker.invoke(invocation);
        } else if (value.startsWith("force")) {
            //force開頭,強(qiáng)制進(jìn)行mock //這個force只能通過override觸發(fā)
            if (logger.isWarnEnabled()) {
                logger.info("force-mock: " + invocation.getMethodName() + " force-mock enabled , url : " + directory.getUrl());
            }
            //force:direct mock
            result = doMockInvoke(invocation, null);
        } else {
            //不是force開頭,調(diào)用失敗走mock邏輯
            try {
                result = this.invoker.invoke(invocation);
            } catch (RpcException e) {
                if (e.isBiz()) {
                    throw e;
                } else {
                    if (logger.isWarnEnabled()) {
                        logger.warn("fail-mock: " + invocation.getMethodName() + " fail-mock enabled , url : " + directory.getUrl(), e);
                    }
                    //不是force開頭,會在調(diào)用失敗后,走mock邏輯
                    result = doMockInvoke(invocation, e);
                }
            }
        }
        return result;
    }

MockClusterInvoker#invoke方法首先會通過mock配置是否為force開頭或者是否為false來判斷走強(qiáng)制降級,失敗降級還是不走降級邏輯。

  • 強(qiáng)制Mock,直接調(diào)用mock邏輯,屏蔽對提供者調(diào)用
  • 失敗Mock,會先進(jìn)行集群調(diào)用,出現(xiàn)異常后,再走mock邏輯

mock邏輯入口為doMockInvoke方法,在該方法內(nèi)封裝了一個隱藏邏輯。之前我們都認(rèn)為觸發(fā)mock是通過dubbo的override機(jī)制,但是看了這個方法后發(fā)現(xiàn),我們可以在zk的下增加一個mock協(xié)議的提供者,也是能夠觸發(fā)mock的,并且后者的優(yōu)先級大于前者。

private Result doMockInvoke(Invocation invocation, RpcException e) {
        Result result = null;
        Invoker<T> minvoker;
        //如果有mock協(xié)議的provider 優(yōu)先使用mock provider
        List<Invoker<T>> mockInvokers = selectMockInvoker(invocation);
        if (CollectionUtils.isEmpty(mockInvokers)) {
            //使用override后的url生成mockinvoker
            minvoker = (Invoker<T>) new MockInvoker(directory.getUrl());
        } else {
            //使用mock provider
            minvoker = mockInvokers.get(0);
        }
        try {
            result = minvoker.invoke(invocation);
        } catch (RpcException me) {
            if (me.isBiz()) {
                result = new RpcResult(me.getCause());
            } else {
                throw new RpcException(me.getCode(), getMockExceptionMessage(e, me), me.getCause());
            }
        } catch (Throwable me) {
            throw new RpcException(getMockExceptionMessage(e, me), me.getCause());
        }
        return result;
    }

不管是通過何種方式設(shè)置的mock邏輯,最終在doMockInvoke我們都會拿到一個MockInvoker,關(guān)于Mock表達(dá)式的解析,mock對象的返回等邏輯都封裝在這個Invoker里。

MockInvoker的源碼解析如下,和我上面列的mock表達(dá)式配置基本一致。

final public class MockInvoker<T> implements Invoker<T> {
    private final static ProxyFactory proxyFactory = ExtensionLoader.getExtensionLoader(ProxyFactory.class).getAdaptiveExtension();
    private final static Map<String, Invoker<?>> mocks = new ConcurrentHashMap<String, Invoker<?>>();
    private final static Map<String, Throwable> throwables = new ConcurrentHashMap<String, Throwable>();
    private final URL url;
    public MockInvoker(URL url) {
        this.url = url;
    }
    public static Object parseMockValue(String mock) throws Exception {
        return parseMockValue(mock, null);
    }
    public static Object parseMockValue(String mock, Type[] returnTypes) throws Exception {
        Object value = null;
        if ("empty".equals(mock)) {
            //如果是empty value等于對應(yīng)類型空實現(xiàn) 具體邏輯見getEmptyObject
            value = ReflectUtils.getEmptyObject(returnTypes != null && returnTypes.length > 0 ? (Class<?>) returnTypes[0] : null);
        } else if ("null".equals(mock)) {
            value = null;
        } else if ("true".equals(mock)) {
            value = true;
        } else if ("false".equals(mock)) {
            value = false;
        } else if (mock.length() >= 2 && (mock.startsWith("\"") && mock.endsWith("\"")
                || mock.startsWith("\'") && mock.endsWith("\'"))) {
            //去除前后引號
            value = mock.subSequence(1, mock.length() - 1);
        } else if (returnTypes != null && returnTypes.length > 0 && returnTypes[0] == String.class) {
            //返回類型為String類型
            value = mock;
        } else if (StringUtils.isNumeric(mock, false)) {
            //如果是數(shù)字
            value = JSON.parse(mock);
        } else if (mock.startsWith("{")) {
            //{開頭 Json反序列化解析為Map
            value = JSON.parseObject(mock, Map.class);
        } else if (mock.startsWith("[")) {
            //[開頭 Json反序列化解析為List
            value = JSON.parseObject(mock, List.class);
        } else {
            //其他 不做處理
            value = mock;
        }
        //如果returnTypes不為空 進(jìn)一步做處理
        if (ArrayUtils.isNotEmpty(returnTypes)) {
            //returnTypes[0]為目標(biāo)類型  returnTypes[1]為目標(biāo)類型的泛型類型
            value = PojoUtils.realize(value, (Class<?>) returnTypes[0], returnTypes.length > 1 ? returnTypes[1] : null);
        }
        return value;
    }
    @Override
    public Result invoke(Invocation invocation) throws RpcException {
        //獲取針對方法的mock配置
        String mock = getUrl().getParameter(invocation.getMethodName() + "." + Constants.MOCK_KEY);
        if (invocation instanceof RpcInvocation) {
            ((RpcInvocation) invocation).setInvoker(this);
        }
        //如果針對方法沒有mock配置 取接口級別的mock配置
        if (StringUtils.isBlank(mock)) {
            mock = getUrl().getParameter(Constants.MOCK_KEY);
        }
        //如果沒有mock配置拋出異常
        if (StringUtils.isBlank(mock)) {
            throw new RpcException(new IllegalAccessException("mock can not be null. url :" + url));
        }
        //標(biāo)準(zhǔn)化mock配置 方便下面匹配
        mock = normalizeMock(URL.decode(mock));
        //return 開頭
        if (mock.startsWith(Constants.RETURN_PREFIX)) {
            //去除return
            mock = mock.substring(Constants.RETURN_PREFIX.length()).trim();
            try {
                //獲取這個接口的返回類型
                Type[] returnTypes = RpcUtils.getReturnTypes(invocation);
                //mock配置反序列化為對應(yīng)類型的value
                Object value = parseMockValue(mock, returnTypes);
                return new RpcResult(value);
            } catch (Exception ew) {
                throw new RpcException("mock return invoke error. method :" + invocation.getMethodName()
                        + ", mock:" + mock + ", url: " + url, ew);
            }
        } else if (mock.startsWith(Constants.THROW_PREFIX)) { //throw 開頭
            //去除throw
            mock = mock.substring(Constants.THROW_PREFIX.length()).trim();
            //如果mock為空 ,降級為拋出RpcException
            if (StringUtils.isBlank(mock)) {
                throw new RpcException("mocked exception for service degradation.");
            } else { // user customized class
                //反序列化為用戶自定義異常
                Throwable t = getThrowable(mock);
                //用RpcException包裝自定義異常拋出
                //注意異常類型設(shè)置為業(yè)務(wù)異常
                throw new RpcException(RpcException.BIZ_EXCEPTION, t);
            }
        } else { //impl mock
            try {
                //default或者具體接口實現(xiàn)類,默認(rèn)使用接口Mock這個類實現(xiàn)mock邏輯
                Invoker<T> invoker = getInvoker(mock);
                return invoker.invoke(invocation);
            } catch (Throwable t) {
                throw new RpcException("Failed to create mock implementation class " + mock, t);
            }
        }
    }
    /**
     * 生成mock異常對象
     * @param throwstr
     * @return
     */
    public static Throwable getThrowable(String throwstr) {
        //緩存邏輯
        Throwable throwable = throwables.get(throwstr);
        if (throwable != null) {
            return throwable;
        }
        try {
            Throwable t;
            //反射獲取異常class對象
            Class<?> bizException = ReflectUtils.forName(throwstr);
            Constructor<?> constructor;
            //反射獲取異常 參數(shù)為string的構(gòu)造函數(shù)
            constructor = ReflectUtils.findConstructor(bizException, String.class);
            //創(chuàng)建異常對象
            t = (Throwable) constructor.newInstance(new Object[]{"mocked exception for service degradation."});
            //緩存大小限制1000
            if (throwables.size() < 1000) {
                throwables.put(throwstr, t);
            }
            return t;
        } catch (Exception e) {
            //如果反射出現(xiàn)異常,降級為直接返回RpcException
            throw new RpcException("mock throw error :" + throwstr + " argument error.", e);
        }
    }
    @SuppressWarnings("unchecked")
    private Invoker<T> getInvoker(String mockService) {
        //緩存
        Invoker<T> invoker = (Invoker<T>) mocks.get(mockService);
        if (invoker != null) {
            return invoker;
        }
        //反射得到接口類class對象
        Class<T> serviceType = (Class<T>) ReflectUtils.forName(url.getServiceInterface());
        //得到接口mock類對象
        T mockObject = (T) getMockObject(mockService, serviceType);
        //生成代理
        invoker = proxyFactory.getInvoker(mockObject, serviceType, url);
        //緩存大小1000
        if (mocks.size() < 10000) {
            mocks.put(mockService, invoker);
        }
        return invoker;
    }
    /**
     * 獲取serviceType對應(yīng)mock實現(xiàn)對象
     * @param mockService
     * @param serviceType
     * @return
     */
    @SuppressWarnings("unchecked")
    public static Object getMockObject(String mockService, Class serviceType) {
        //如果mock配置為default 對應(yīng)mock實現(xiàn)類為 接口名+Mock
        //否則為指定了具體實現(xiàn)類
        if (ConfigUtils.isDefault(mockService)) {
            mockService = serviceType.getName() + "Mock";
        }
        //反射得到mock實現(xiàn)類class對象
        //注意這邊如果找不到 會拋出運(yùn)行時異常
        Class<?> mockClass = ReflectUtils.forName(mockService);
        //判斷是否實現(xiàn)了接口
        if (!serviceType.isAssignableFrom(mockClass)) {
            throw new IllegalStateException("The mock class " + mockClass.getName() +
                    " not implement interface " + serviceType.getName());
        }
        try {
            //創(chuàng)建實例
            return mockClass.newInstance();
        } catch (InstantiationException e) {
            throw new IllegalStateException("No default constructor from mock class " + mockClass.getName(), e);
        } catch (IllegalAccessException e) {
            throw new IllegalStateException(e);
        }
    }
    /**
     * Normalize mock string:
     *
     * <ol>
     * <li>return => return null</li>
     * <li>fail => default</li>
     * <li>force => default</li>
     * <li>fail:throw/return foo => throw/return foo</li>
     * <li>force:throw/return foo => throw/return foo</li>
     * </ol>
     *
     * @param mock mock string
     * @return normalized mock string
     */
    public static String normalizeMock(String mock) {
        if (mock == null) {
            return mock;
        }
        mock = mock.trim();
        if (mock.length() == 0) {
            return mock;
        }
        if (Constants.RETURN_KEY.equalsIgnoreCase(mock)) {
            return Constants.RETURN_PREFIX + "null";
        }
        if (ConfigUtils.isDefault(mock) || "fail".equalsIgnoreCase(mock) || "force".equalsIgnoreCase(mock)) {
            return "default";
        }
        if (mock.startsWith(Constants.FAIL_PREFIX)) {
            mock = mock.substring(Constants.FAIL_PREFIX.length()).trim();
        }
        if (mock.startsWith(Constants.FORCE_PREFIX)) {
            mock = mock.substring(Constants.FORCE_PREFIX.length()).trim();
        }
        if (mock.startsWith(Constants.RETURN_PREFIX) || mock.startsWith(Constants.THROW_PREFIX)) {
            mock = mock.replace('`', '"');
        }
        return mock;
    }
    @Override
    public URL getUrl() {
        return this.url;
    }
    @Override
    public boolean isAvailable() {
        return true;
    }
    @Override
    public void destroy() {
        //do nothing
    }
    @Override
    public Class<T> getInterface() {
        //FIXME
        return null;
    }
}

總結(jié)

  • Dubbo中的mock是用來做降級功能,通過SPI包裝類來強(qiáng)制植入mock邏輯
  • mock分為強(qiáng)制mock和失敗mock
  • 推薦在客戶端配置失敗mock邏輯,通過服務(wù)治理設(shè)置強(qiáng)制mock
  • 服務(wù)治理一般通過override讓客戶端mock邏輯生效,實際上也支持配置mock協(xié)議的provider讓mock生效,并且后者優(yōu)先級更高

以上就是Dubbo之降級Mock源碼分析的詳細(xì)內(nèi)容,更多關(guān)于Dubbo降級Mock源碼分析的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • java開發(fā)之SQL語句中DATE_FORMAT函數(shù)舉例詳解

    java開發(fā)之SQL語句中DATE_FORMAT函數(shù)舉例詳解

    要將日期值格式化為特定格式,請使用DATE_FORMAT函數(shù),下面這篇文章主要給大家介紹了關(guān)于java開發(fā)之SQL語句中DATE_FORMAT函數(shù)的相關(guān)資料,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-05-05
  • Java時間輪算法的實現(xiàn)代碼示例

    Java時間輪算法的實現(xiàn)代碼示例

    本篇文章主要介紹了Java時間輪算法的實現(xiàn)代碼示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-08-08
  • Java程序運(yùn)行時出現(xiàn)亂碼問題的排查與解決方法

    Java程序運(yùn)行時出現(xiàn)亂碼問題的排查與解決方法

    本文主要介紹了Java程序運(yùn)行時出現(xiàn)亂碼問題的排查與解決方法,包括檢查Java源文件編碼、檢查編譯時的編碼設(shè)置、檢查運(yùn)行時的編碼設(shè)置、檢查命令提示符的代碼頁、檢查命令提示符的字體、檢查 Java 程序的輸出代碼以及檢查環(huán)境變量,需要的朋友可以參考下
    2025-03-03
  • springboot整合kaptcha驗證碼的示例代碼

    springboot整合kaptcha驗證碼的示例代碼

    kaptcha是一個很有用的驗證碼生成工具,本篇文章主要介紹了springboot整合kaptcha驗證碼的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-06-06
  • Java操作Excel的示例詳解

    Java操作Excel的示例詳解

    在平時可以使用IO流對Excle進(jìn)行操作,但是現(xiàn)在使用更加方便的第三方組件來實現(xiàn)。本文就來和大家聊聊Java如何通過第三方組件實現(xiàn)操作Excel,需要的可以參考一下
    2023-01-01
  • Java Stream中自定義Collector實現(xiàn)復(fù)雜數(shù)據(jù)收集的方法

    Java Stream中自定義Collector實現(xiàn)復(fù)雜數(shù)據(jù)收集的方法

    Java Stream API中的Collector接口是一個強(qiáng)大的工具,它允許我們自定義數(shù)據(jù)收集、轉(zhuǎn)換和聚合的過程,,本文介紹了Java Stream中自定義Collector實現(xiàn)復(fù)雜數(shù)據(jù)收集方法,需要的朋友可以參考下
    2024-08-08
  • Mybatis-Plus使用p6spy對SQL性能進(jìn)行監(jiān)控的方法

    Mybatis-Plus使用p6spy對SQL性能進(jìn)行監(jiān)控的方法

    這篇文章主要介紹了Mybatis-Plus使用p6spy對SQL性能進(jìn)行監(jiān)控的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • mybatis-flex實現(xiàn)多數(shù)據(jù)源操作

    mybatis-flex實現(xiàn)多數(shù)據(jù)源操作

    MyBaits-Flex內(nèi)置了功能完善的多數(shù)據(jù)源支持,本文主要介紹了mybatis-flex實現(xiàn)多數(shù)據(jù)源操作,具有一定的參考價值,感興趣的可以了解一下
    2024-06-06
  • FuncGPT慧函數(shù)保護(hù)數(shù)據(jù)安全提高代碼質(zhì)量減少軟件故障(java示例)

    FuncGPT慧函數(shù)保護(hù)數(shù)據(jù)安全提高代碼質(zhì)量減少軟件故障(java示例)

    這篇文章主要為大家介紹了FuncGPT慧函數(shù)保護(hù)數(shù)據(jù)安全提高代碼質(zhì)量減少軟件故障(java示例),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-10-10
  • Java編程實現(xiàn)軌跡壓縮算法開放窗口實例代碼

    Java編程實現(xiàn)軌跡壓縮算法開放窗口實例代碼

    這篇文章主要介紹了Java編程實現(xiàn)軌跡壓縮算法開放窗口實例代碼,具有一定借鑒價值,需要的朋友可以參考下。
    2017-11-11

最新評論