优秀的编程知识分享平台

网站首页 > 技术文章 正文

开箱即用多租户插件druid-multi-tenant-starter

nanyue 2024-08-04 17:15:16 技术文章 8 ℃

druid-multi-tenant-starter

  1. 目标多租户插件,开箱即用,支持单个租户以及多个租户,使用阿里巴巴的druid解析sql
  2. 默认每个表都应该有tenantId字段

支持忽略tenantId策略

  1. 支持原SQL语句中已有tenantId字段条件跳过
  2. 支持忽略指定表名添加tenantId字段条件
  3. 支持忽略指定Mapper接口SQL语句添加tenantId字段条件

快速开始

  • 在已经集成`Mybatis`项目中引入以下依赖
<dependency>
    <groupId>io.github.osinn</groupId>
    <artifactId>druid-multi-tenant-starter</artifactId>
    <version>1.0</version>
</dependency>

配置

mybatis:
  tenant:
    config:
      # 是否使用MyBatis拦截器方式修改sql,需要禁用druidFilterEnable
      enable: true
      # 数据库方言
      db-type: mysql
      # 是否忽略表按租户ID过滤,默认所有表都按租户ID过滤,指定表名称
      ignore-table-name:
#        - user
        - user_role
      # 数据库中租户ID的列名
      tenant-id-column: tenant_id
      # 是否使用druid过滤器方式修改sql,依赖druid数据库连接池,需要禁用enable=false
      druid-filter-enable: false

实现提供获取多租户值接口

  • 需要实现ITenantService接口提供获取多租户ID值
/**
 * 演示:提供多租户ID服务接口
 *
 * @author wency_cai
 */
@Service
public class TenantServiceImpl implements ITenantService<Integer>{

    @Override
    public List<Integer> getTenantIds() {
        // 查询系统多租户id,如果有多个返回多个值即可
        int tenantId = 1;
        return Lists.newArrayList(tenantId);
    }
}
  • 到此整合完成

多租户忽略Mapper方法

public interface UserMapper {

    /**
     * 添加@IgnoreTenantIdField注解来忽略设置多租户字段
     */
    @IgnoreTenantIdField
    void deleteTestIgnoreTenantIdById(Long id);
}

解析后到SQL语句如下

delete语句

DELETE FROM user
WHERE id = 1
   AND tenant_id = 11
------------------------------------- 

DELETE FROM user
WHERE id IN (
      SELECT id
      FROM user s
      WHERE s.tenant_id = 11
   )
   AND tenant_id = 11
------------------------------------- 

DELETE FROM system_permission_data sp
   LEFT JOIN system_role_data_relation re
   ON sp.id = re.data_id
      AND re.tenant_id = 11
   LEFT JOIN system_role sr
   ON sr.id = re.role_id
      AND sr.tenant_id = 11
WHERE sr.id = 1
   AND sp.tenant_id = 11

insert语句

INSERT INTO `user` (`id`, `username`, `password`, tenant_id)
VALUES (?, ?, ?, 11),
   (?, ?, ?, 11),
   (?, ?, ?, 11)

update语句

UPDATE user u
SET ds = ?, u.name = ?, id = 'fdf', ddd = ?
WHERE id = ?
   AND u.tenant_id = 11
------------------------------------- 

UPDATE user u
SET ds = ?, u.name = ?, id = 'fdf', ddd = ?
WHERE id IN (1, 2, 3)
   AND u.tenant_id = 11
------------------------------------- 

UPDATE user u
SET ds = ?, u.name = ?, id = 'fdf', ddd = ?
WHERE id IN (
      SELECT ur.user_id
      FROM user_role ur
      WHERE ur.tenant_id = 11
   )
   AND u.tenant_id = 11
------------------------------------- 

UPDATE user u
SET ds = ?, u.name = ?, id = 'fdf', ddd = ?
WHERE id IN (
      SELECT ur.user_id
      FROM user_role ur
      WHERE ur.id = 1
         AND ur.tenant_id = 11
   )
   AND u.tenant_id = 11
------------------------------------- 

UPDATE `user` u
   JOIN user_role ur
   ON ur.user_id = u.id
      AND u.id = 111
      AND ur.tenant_id = 11
SET u.qr_code = '1212'
WHERE u.tenant_id = 11
------------------------------------- 

UPDATE `user` u
   JOIN user_role ur
   ON ur.user_id = u.id
      AND u.id IN (
         SELECT urr.user_id
         FROM user_role urr
         WHERE urr.tenant_id = 11
      )
      AND ur.tenant_id = 11
SET u.qr_code = '1212'
WHERE u.tenant_id = 11
------------------------------------- 

UPDATE user
SET name = CASE 
   WHEN id = 1 THEN 'name1'
   WHEN id = 2 THEN 'name2'
   WHEN id = 3 THEN 'name3'
END
WHERE id IN (11, 22, 33, 3)
   AND tenant_id = 11;

select语句

SELECT id, name, tenant_id
FROM role
WHERE tenant_id = 11
------------------------------------- 

SELECT *
FROM user s
WHERE s.name = '333'
   AND s.tenant_id = 11
------------------------------------- 

SELECT tenant_id
FROM people
WHERE id IN (
      SELECT id
      FROM user s
      WHERE s.tenant_id = 11
   )
   AND tenant_id = 11
------------------------------------- 

SELECT tenant_id
FROM people
WHERE id IN (
      SELECT id
      FROM user s
      WHERE s.tenant_id = 11
   )
   AND id IN (
      SELECT u.id
      FROM user u
      WHERE u.tenant_id = 11
   )
   AND tenant_id = 11
------------------------------------- 

SELECT tenant_id
FROM people
WHERE id IN (
      SELECT id
      FROM user s
      WHERE s.tenant_id = 11
   )
   AND id IN (
      SELECT u.id
      FROM user u
      WHERE u.tenant_id = 11
   )
   AND tenant_id IN (
      SELECT u.tenant_id
      FROM user u
      WHERE u.tenant_id = 11
   )
   AND tenant_id = 11
------------------------------------- 

SELECT p.tenant_id
   , (
      SELECT u.name
      FROM user u
      WHERE u.id = p.user_id
         AND u.tenant_id = 11
   )
FROM people p
WHERE p.tenant_id = 11
------------------------------------- 

SELECT p.tenant_id
   , (
      SELECT u.name
      FROM user u
      WHERE u.id = p.user_id
         AND u.tenant_id = 11
   )
   , (
      SELECT s.name
      FROM user s
      WHERE s.id = p.user_id
         AND s.tenant_id = 11
   )
FROM people p
WHERE p.tenant_id = 11
------------------------------------- 

SELECT p.tenant_id
   , (
      SELECT u.name
         , (
            SELECT s.name
            FROM user s
            WHERE s.id = p.user_id
               AND s.tenant_id = 11
         )
      FROM user u
      WHERE u.id = p.user_id
         AND u.tenant_id = 11
   )
FROM people p
WHERE p.tenant_id = 11
------------------------------------- 

SELECT u.*
FROM `user` u
   JOIN user_role ur
   ON ur.user_id = u.id
      AND ur.tenant_id = 11
WHERE u.tenant_id = 11
------------------------------------- 

SELECT u.*
FROM `user` u
   JOIN user_role ur
   ON ur.user_id = u.id
      AND u.id = 22
      AND ur.tenant_id = 11
WHERE u.tenant_id = 11
------------------------------------- 

SELECT u.*
   , (
      SELECT ur.name
      FROM role r
      WHERE r.id = ur.role_id
         AND r.tenant_id = 11
   ) AS r_name
FROM `user` u
   JOIN user_role ur
   ON ur.user_id = u.id
      AND u.id = 22
      AND ur.tenant_id = 11
WHERE u.tenant_id = 11
------------------------------------- 

SELECT u.*
FROM `user` u
   JOIN user_role ur
   ON ur.user_id = u.id
      AND ur.role_id IN (
         SELECT r.id
         FROM role r
         WHERE r.tenant_id = 11
      )
      AND ur.tenant_id = 11
WHERE u.tenant_id = 11
------------------------------------- 

SELECT u.*
FROM `user` u
   JOIN user_role ur
   ON ur.user_id = u.id
      AND ur.role_id IN (
         SELECT r.id
         FROM role r
         WHERE r.tenant_id = 11
      )
      AND ur.test_id IN (
         SELECT r.test_id
         FROM role r
         WHERE r.tenant_id = 11
      )
      AND ur.tenant_id = 11
WHERE u.tenant_id = 11
------------------------------------- 

SELECT u.*
FROM `user` u
   LEFT JOIN user_role ur
   ON ur.user_id = u.id
      AND ur.tenant_id = 11
WHERE u.tenant_id = 11
------------------------------------- 

SELECT u.*
FROM `user` u
   RIGHT JOIN user_role ur
   ON ur.user_id = u.id
      AND ur.tenant_id = 11
WHERE u.tenant_id = 11
------------------------------------- 

SELECT COUNT(*), id
FROM `user`
WHERE tenant_id = 11
GROUP BY id
HAVING COUNT(*) >= 1
------------------------------------- 

SELECT EXISTS (
      SELECT *
      FROM `user`
      WHERE username = ?
         AND tenant_id = 11
   ) AS d
------------------------------------- 

SELECT EXISTS (
      SELECT tenant_id
      FROM people
      WHERE id IN (
            SELECT id
            FROM user s
            WHERE s.tenant_id = 11
         )
         AND tenant_id = 11
   ) AS d
------------------------------------- 

SELECT username
FROM `user`
WHERE tenant_id = 11
UNION
SELECT username
FROM account
WHERE tenant_id = 11
------------------------------------- 

SELECT u.*, r.id AS r_id, r.NAME AS r_name
FROM `user` u
   LEFT JOIN user_role ur
   ON ur.user_id = u.id
      AND ur.tenant_id = 11
   LEFT JOIN role r
   ON r.id = ur.role_id
      AND u.id = 22
      AND r.tenant_id = 11
WHERE u.tenant_id = 11
------------------------------------- 

SELECT u.*, r.id AS r_id, r.NAME AS r_name
FROM `user` u
   LEFT JOIN user_role ur
   ON ur.user_id = u.id
      AND ur.tenant_id = 11
   LEFT JOIN role r
   ON r.id = ur.role_id
      AND u.id = 22
      AND r.tenant_id = 11
   LEFT JOIN menu m
   ON m.role_id = r.id
      AND m.tenant_id = 11
WHERE u.tenant_id = 11
------------------------------------- 

SELECT a, b
FROM (
   SELECT *
   FROM table_a
   WHERE tenant_id = 11
) temp
WHERE temp.a = 'a';
------------------------------------- 

SELECT temp.*
FROM (
   SELECT *
   FROM `user`
   WHERE tenant_id = 11
) temp
   JOIN user_role ur
   ON ur.user_id = temp.id
      AND ur.tenant_id = 11
WHERE temp.a = 'a'
   AND temp.tenant_id = 11;
------------------------------------- 

SELECT id, name, tenant_id
FROM role
WHERE tenant_id = 11

Tags:

最近发表
标签列表