SpringCloud微服務(wù)實(shí)戰(zhàn)——搭建企業(yè)級(jí)開(kāi)發(fā)框架(二十二):基于MybatisPlus插件TenantLineInnerInterceptor實(shí)現(xiàn)多租戶功能

多租戶技術(shù)的基本概念:
??多租戶技術(shù)(英語(yǔ):multi-tenancy technology)或稱(chēng)多重租賃技術(shù),是一種軟件架構(gòu)技術(shù),它是在探討與實(shí)現(xiàn)如何于多用戶的環(huán)境下共用相同的系統(tǒng)或程序組件,并且仍可確保各用戶間數(shù)據(jù)的隔離性隆夯。
??在云計(jì)算的加持之下,多租戶技術(shù)被廣為運(yùn)用于開(kāi)發(fā)云各式服務(wù)榛搔,不論是IaaS,PaaS還是SaaS,都可以看到多租戶技術(shù)的影子暑刃。

??前面介紹過(guò)GitEgg框架與數(shù)據(jù)庫(kù)交互使用了Mybatis增強(qiáng)工具M(jìn)ybatis-Plus,Mybatis-Plus提供了TenantLineInnerInterceptor租戶處理器來(lái)實(shí)現(xiàn)多租戶功能膜眠,其原理就是Mybatis-Plus實(shí)現(xiàn)了自定義Mybatis攔截器(Interceptor)岩臣,在需要執(zhí)行的sql后面自動(dòng)添加租戶的查詢條件,實(shí)際和分頁(yè)插件宵膨,數(shù)據(jù)權(quán)限攔截器是同樣的實(shí)現(xiàn)方式架谎。

簡(jiǎn)而言之多租戶技術(shù)就是可以讓一套系統(tǒng)通過(guò)配置給不同的客戶提供服務(wù),每個(gè)客戶看到的數(shù)據(jù)都是屬于自己的辟躏,就好像每個(gè)客戶都擁有自己一套獨(dú)立完善的系統(tǒng)谷扣。

下面是在GitEgg系統(tǒng)的應(yīng)用配置:

1、在gitegg-platform-mybatis工程下新建多租戶組件配置文件TenantProperties.java和TenantConfig.java鸿脓,TenantProperties.java用于系統(tǒng)讀取配置文件抑钟,這里會(huì)在Nacos配置中心設(shè)置多組戶的具體配置信息,TenantConfig.java是插件需要讀取的配置有三個(gè)配置項(xiàng):
TenantId租戶ID野哭、TenantIdColumn多租戶的字段名、ignoreTable不需要多租戶隔離的表幻件。
TenantProperties.java:

package com.gitegg.platform.mybatis.props;

import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;

import java.util.List;

/**
 * 白名單配置
 */
@Data
@Configuration
@ConfigurationProperties(prefix = "tenant")
public class TenantProperties {

    /**
     * 是否開(kāi)啟租戶模式
     */
    private Boolean enable;

    /**
     * 多租戶字段名稱(chēng)
     */
    private String column;

    /**
     * 需要排除的多租戶的表
     */
    private List<String> exclusionTable;

}

TenantConfig.java:

package com.gitegg.platform.mybatis.config;

import com.baomidou.mybatisplus.extension.plugins.handler.TenantLineHandler;
import com.baomidou.mybatisplus.extension.plugins.inner.TenantLineInnerInterceptor;
import com.gitegg.platform.boot.util.GitEggAuthUtils;
import com.gitegg.platform.mybatis.props.TenantProperties;
import lombok.RequiredArgsConstructor;
import net.sf.jsqlparser.expression.Expression;
import net.sf.jsqlparser.expression.NullValue;
import net.sf.jsqlparser.expression.StringValue;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.autoconfigure.AutoConfigureBefore;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;


/**
 * 多租戶配置中心
 *
 * @author GitEgg
 */
@Configuration
@RequiredArgsConstructor(onConstructor_ = @Autowired)
@AutoConfigureBefore(MybatisPlusConfig.class)
public class TenantConfig {

    private final TenantProperties tenantProperties;

    /**
     * 新多租戶插件配置,一緩和二緩遵循mybatis的規(guī)則,
     * 需要設(shè)置 MybatisConfiguration#useDeprecatedExecutor = false
     * 避免緩存萬(wàn)一出現(xiàn)問(wèn)題
     *
     * @return TenantLineInnerInterceptor
     */
    @Bean
    public TenantLineInnerInterceptor tenantLineInnerInterceptor() {
        return new TenantLineInnerInterceptor(new TenantLineHandler() {
            /**
             * 獲取租戶ID
             * @return Expression
             */
            @Override
            public Expression getTenantId() {
                String tenant = GitEggAuthUtils.getTenantId();
                if (tenant != null) {
                    return new StringValue(GitEggAuthUtils.getTenantId());
                }
                return new NullValue();
            }

            /**
             * 獲取多租戶的字段名
             * @return String
             */
            @Override
            public String getTenantIdColumn() {
                return tenantProperties.getColumn();
            }

            /**
             * 過(guò)濾不需要根據(jù)租戶隔離的表
             * 這是 default 方法,默認(rèn)返回 false 表示所有表都需要拼多租戶條件
             * @param tableName 表名
             */
            @Override
            public boolean ignoreTable(String tableName) {
                return tenantProperties.getExclusionTable().stream().anyMatch(
                        (t) -> t.equalsIgnoreCase(tableName)
                );
            }
        });
    }
}

2拨黔、可在工程下新建application.yml,配置將來(lái)需要在Nacos上配置的信息:

tenant:
  # 是否開(kāi)啟租戶模式
  enable: true
  # 需要排除的多租戶的表
  exclusionTable:
    - "t_sys_district"
    - "oauth_client_details"
  # 租戶字段名稱(chēng)
  column: tenant_id

3绰沥、修改MybatisPlusConfig.java篱蝇,把多租戶過(guò)濾器加載進(jìn)來(lái)使其生效:

package com.gitegg.platform.mybatis.config;

import com.baomidou.mybatisplus.annotation.DbType;
import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.BlockAttackInnerInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.OptimisticLockerInnerInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.TenantLineInnerInterceptor;
import com.gitegg.platform.mybatis.props.TenantProperties;
import lombok.RequiredArgsConstructor;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
@RequiredArgsConstructor(onConstructor_ = @Autowired)
@MapperScan("com.gitegg.**.mapper.**")
public class MybatisPlusConfig {

    private final TenantLineInnerInterceptor tenantLineInnerInterceptor;

    private final TenantProperties tenantProperties;

    /**
     * 新的分頁(yè)插件,一緩和二緩遵循mybatis的規(guī)則,需要設(shè)置 MybatisConfiguration#useDeprecatedExecutor = false
     * 避免緩存出現(xiàn)問(wèn)題(該屬性會(huì)在舊插件移除后一同移除)
     */
    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {

        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();

        //多租戶插件
        if (tenantProperties.getEnable()) {
            interceptor.addInnerInterceptor(tenantLineInnerInterceptor);
        }

        //分頁(yè)插件
        interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));

        //防止全表更新與刪除插件: BlockAttackInnerInterceptor
        BlockAttackInnerInterceptor blockAttackInnerInterceptor = new BlockAttackInnerInterceptor();
        interceptor.addInnerInterceptor(blockAttackInnerInterceptor);

        return interceptor;
    }

    /**
     * 樂(lè)觀鎖插件 當(dāng)要更新一條記錄的時(shí)候贺待,希望這條記錄沒(méi)有被別人更新
     * https://mybatis.plus/guide/interceptor-optimistic-locker.html#optimisticlockerinnerinterceptor
     */
    @Bean
    public OptimisticLockerInnerInterceptor optimisticLockerInterceptor() {
        return new OptimisticLockerInnerInterceptor();
    }

}

4、在GitEggAuthUtils方法中新增獲取租戶信息的公共方法零截,租戶信息在Gateway進(jìn)行轉(zhuǎn)發(fā)時(shí)進(jìn)行設(shè)置麸塞,后面會(huì)說(shuō)明如何講租戶信息設(shè)置到Header中:

package com.gitegg.platform.boot.util;

import cn.hutool.json.JSONUtil;
import com.gitegg.platform.base.constant.AuthConstant;
import com.gitegg.platform.base.domain.GitEggUser;
import org.springframework.util.StringUtils;

import javax.servlet.http.HttpServletRequest;
import java.io.UnsupportedEncodingException;
import java.net.URLDecoder;

public class GitEggAuthUtils {

    /**
     * 獲取用戶信息
     *
     * @return GitEggUser
     */
    public static GitEggUser getCurrentUser() {
        HttpServletRequest request = GitEggWebUtils.getRequest();
        if (request == null) {
            return null;
        }
        try {
            String user = request.getHeader(AuthConstant.HEADER_USER);
            if (StringUtils.isEmpty(user))
            {
                return null;
            }
            String userStr = URLDecoder.decode(user,"UTF-8");
            GitEggUser gitEggUser = JSONUtil.toBean(userStr, GitEggUser.class);
            return gitEggUser;
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
            return null;
        }

    }

    /**
     * 獲取租戶Id
     *
     * @return tenantId
     */
    public static String getTenantId() {
        HttpServletRequest request = GitEggWebUtils.getRequest();
        if (request == null) {
            return null;
        }
        try {
            String tenantId = request.getHeader(AuthConstant.TENANT_ID);
            String user = request.getHeader(AuthConstant.HEADER_USER);
            //如果請(qǐng)求頭中的tenantId為空,那么嘗試是否能夠從登陸用戶中去獲取租戶id
            if (StringUtils.isEmpty(tenantId) && !StringUtils.isEmpty(user))
            {
                String userStr = URLDecoder.decode(user,"UTF-8");
                GitEggUser gitEggUser = JSONUtil.toBean(userStr, GitEggUser.class);
                if (null != gitEggUser)
                {
                    tenantId = gitEggUser.getTenantId();
                }
            }
            return tenantId;
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
            return null;
        }

    }
}

5涧衙、GitEgg-Cloud工程中g(shù)itegg-gateway子工程的AuthGlobalFilter增加設(shè)置TenantId的過(guò)濾方法

        String tenantId = exchange.getRequest().getHeaders().getFirst(AuthConstant.TENANT_ID);

        String token = exchange.getRequest().getHeaders().getFirst(AuthConstant.JWT_TOKEN_HEADER);

        if (StrUtil.isEmpty(tenantId) && StrUtil.isEmpty(token)) {
            return chain.filter(exchange);
        }

        Map<String, String> addHeaders = new HashMap<>();

        // 如果系統(tǒng)配置已開(kāi)啟租戶模式哪工,設(shè)置tenantId
        if (enable && StrUtil.isEmpty(tenantId)) {
            addHeaders.put(AuthConstant.TENANT_ID, tenantId);
        }

6、以上為后臺(tái)的多租戶功能集成步驟弧哎,在實(shí)際項(xiàng)目開(kāi)發(fā)過(guò)程中雁比,我們需要考慮到前端頁(yè)面在租戶信息上的配置,實(shí)現(xiàn)思路撤嫩,不用的租戶擁有不同的域名偎捎,前端頁(yè)面根據(jù)當(dāng)前域名獲取到對(duì)應(yīng)的租戶信息,并在公共請(qǐng)求方法設(shè)置TenantId參數(shù)序攘,保證每次請(qǐng)求能夠攜帶租戶信息茴她。

// request interceptor
request.interceptors.request.use(config => {
  const token = storage.get(ACCESS_TOKEN)
  // 如果 token 存在
  // 讓每個(gè)請(qǐng)求攜帶自定義 token 請(qǐng)根據(jù)實(shí)際情況自行修改
  if (token) {
    config.headers['Authorization'] = token
  }
  config.headers['TenantId'] = process.env.VUE_APP_TENANT_ID
  return config
}, errorHandler)
GitEgg-Cloud是一款基于SpringCloud整合搭建的企業(yè)級(jí)微服務(wù)應(yīng)用開(kāi)發(fā)框架,項(xiàng)目地址: https://gitee.com/wmz1930/GitEgg程奠,歡迎感興趣的小伙伴Star支持一下败京。
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市梦染,隨后出現(xiàn)的幾起案子赡麦,更是在濱河造成了極大的恐慌,老刑警劉巖帕识,帶你破解...
    沈念sama閱讀 216,651評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件泛粹,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡肮疗,警方通過(guò)查閱死者的電腦和手機(jī)晶姊,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,468評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)伪货,“玉大人们衙,你說(shuō)我怎么就攤上這事〖詈簦” “怎么了蒙挑?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,931評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)愚臀。 經(jīng)常有香客問(wèn)我忆蚀,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,218評(píng)論 1 292
  • 正文 為了忘掉前任馋袜,我火速辦了婚禮男旗,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘欣鳖。我一直安慰自己察皇,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,234評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布泽台。 她就那樣靜靜地躺著什荣,像睡著了一般。 火紅的嫁衣襯著肌膚如雪师痕。 梳的紋絲不亂的頭發(fā)上溃睹,一...
    開(kāi)封第一講書(shū)人閱讀 51,198評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音胰坟,去河邊找鬼因篇。 笑死,一個(gè)胖子當(dāng)著我的面吹牛笔横,可吹牛的內(nèi)容都是我干的竞滓。 我是一名探鬼主播,決...
    沈念sama閱讀 40,084評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼吹缔,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼商佑!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起厢塘,我...
    開(kāi)封第一講書(shū)人閱讀 38,926評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤茶没,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后晚碾,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體抓半,經(jīng)...
    沈念sama閱讀 45,341評(píng)論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,563評(píng)論 2 333
  • 正文 我和宋清朗相戀三年格嘁,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了笛求。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,731評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡糕簿,死狀恐怖探入,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情懂诗,我是刑警寧澤蜂嗽,帶...
    沈念sama閱讀 35,430評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站响禽,受9級(jí)特大地震影響徒爹,放射性物質(zhì)發(fā)生泄漏荚醒。R本人自食惡果不足惜芋类,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,036評(píng)論 3 326
  • 文/蒙蒙 一隆嗅、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧侯繁,春花似錦胖喳、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,676評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至咕别,卻和暖如春技健,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背惰拱。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,829評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工雌贱, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人偿短。 一個(gè)月前我還...
    沈念sama閱讀 47,743評(píng)論 2 368
  • 正文 我出身青樓欣孤,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親昔逗。 傳聞我的和親對(duì)象是個(gè)殘疾皇子降传,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,629評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容