mybatis-plus 租户处理器

mybatis-plus 租户处理器配置 租户解析器处理程序 import com.baomidou.mybatisplus.extension.plugins.tenant.TenantSqlParser; import net.s…

mybatis-plus 租户处理器

配置

租户解析器处理程序

import com.baomidou.mybatisplus.extension.plugins.tenant.TenantSqlParser;
import net.sf.jsqlparser.expression.BinaryExpression;
import net.sf.jsqlparser.expression.Expression;
import net.sf.jsqlparser.expression.Parenthesis;
import net.sf.jsqlparser.expression.operators.conditional.AndExpression;
import net.sf.jsqlparser.expression.operators.conditional.OrExpression;
import net.sf.jsqlparser.expression.operators.relational.*;
import net.sf.jsqlparser.schema.Column;
import net.sf.jsqlparser.schema.Table;
import net.sf.jsqlparser.statement.select.*;

import java.util.List;

/**   
* Mp租户解析器处理程序
* @author xxm  
* @date 2020/4/15 11:40
*/
public class MpTenantParserHandler extends TenantSqlParser {

	/**
	 * 目前仅支持:in, between, >, <, =, !=等比较操作,处理多租户的字段加上表别名
	 */
	protected Expression processTableAlias(Expression expression, Table table) {
		String tableAliasName;
		if (table.getAlias() == null) {
			tableAliasName = table.getName();
		} else {
			tableAliasName = table.getAlias().getName();
		}
		if (expression instanceof InExpression) {
			InExpression in = (InExpression) expression;
			if (in.getLeftExpression() instanceof Column) {
				setTableAliasNameForColumn((Column) in.getLeftExpression(), tableAliasName);
			}
		} else if (expression instanceof BinaryExpression) {
			BinaryExpression compare = (BinaryExpression) expression;
			if (compare.getLeftExpression() instanceof Column) {
				setTableAliasNameForColumn((Column) compare.getLeftExpression(), tableAliasName);
			} else if (compare.getRightExpression() instanceof Column) {
				setTableAliasNameForColumn((Column) compare.getRightExpression(), tableAliasName);
			}
		} else if (expression instanceof Between) {
			Between between = (Between) expression;
			if (between.getLeftExpression() instanceof Column) {
				setTableAliasNameForColumn((Column) between.getLeftExpression(), tableAliasName);
			}
		}
		return expression;
	}

	private void setTableAliasNameForColumn(Column column, String tableAliasName) {
		column.setColumnName(tableAliasName + "." + column.getColumnName());
	}

	/**
	 * 默认是按 tenant_id=1 按等于条件追加
	 *
	 * @param currentExpression 现有的条件:比如你原来的sql查询条件
	 * @param table
	 * @return
	 */
	@Override
	protected Expression builderExpression(Expression currentExpression, Table table) {
		final Expression tenantExpression = this.getTenantHandler().getTenantId(true);
		Expression appendExpression;
		if (!(tenantExpression instanceof SupportsOldOracleJoinSyntax)) {
			appendExpression = new EqualsTo();
			((EqualsTo) appendExpression).setLeftExpression(this.getAliasColumn(table));
			((EqualsTo) appendExpression).setRightExpression(tenantExpression);
		} else {
			appendExpression = processTableAlias(tenantExpression, table);
		}
		if (currentExpression == null) {
			return appendExpression;
		}
		if (currentExpression instanceof BinaryExpression) {
			BinaryExpression binaryExpression = (BinaryExpression) currentExpression;
			if (binaryExpression.getLeftExpression() instanceof FromItem) {
				processFromItem((FromItem) binaryExpression.getLeftExpression());
			}
			if (binaryExpression.getRightExpression() instanceof FromItem) {
				processFromItem((FromItem) binaryExpression.getRightExpression());
			}
		} else if (currentExpression instanceof InExpression) {
			InExpression inExp = (InExpression) currentExpression;
			ItemsList rightItems = inExp.getRightItemsList();
			if (rightItems instanceof SubSelect) {
				processSelectBody(((SubSelect) rightItems).getSelectBody());
			}
		}
		if (currentExpression instanceof OrExpression) {
			return new AndExpression(new Parenthesis(currentExpression), appendExpression);
		} else {
			return new AndExpression(currentExpression, appendExpression);
		}
	}

	@Override
	protected void processPlainSelect(PlainSelect plainSelect, boolean addColumn) {
		FromItem fromItem = plainSelect.getFromItem();
		if (fromItem instanceof Table) {
			Table fromTable = (Table) fromItem;
			if (!this.getTenantHandler().doTableFilter(fromTable.getName())) {
				plainSelect.setWhere(builderExpression(plainSelect.getWhere(), fromTable));
				if (addColumn) {
					plainSelect.getSelectItems().add(new SelectExpressionItem(new Column(this.getTenantHandler().getTenantIdColumn())));
				}
			}
		} else {
			processFromItem(fromItem);
		}
		List<Join> joins = plainSelect.getJoins();
		if (joins != null && joins.size() > 0) {
			joins.forEach(j -> {
				processJoin(j);
				processFromItem(j.getRightItem());
			});
		}
	}
}

代码100分

租户处理器

代码100分/**
 * sql租户解析器
 * @author xxm
 * @date 2020/4/14 15:28
 */
@Component
@AllArgsConstructor
public class MpTenantHandler implements TenantHandler {

	private final HeaderHolder headerHolder;

	/**
	 * 租户值
	 */
	@Override
	public Expression getTenantId(boolean where) {
		Long tid = headerHolder.findTid();
		return new LongValue(tid);
	}

	/**
	 * 租户字段名称
	 */
	@Override
	public String getTenantIdColumn() {
		return "tid";
	}

	@Override
	public boolean doTableFilter(String tableName) {
		//TODO 如果是超级租户 不进行过滤
		return Objects.equals(headerHolder.getTid(), 1L);
	}
}

MP配置

/**   
* mp配置
* @author xxm  
* @date 2020/4/8 14:05
*/
@Configuration
public class MybatisPlusConfig {

    /**
     * 多租户属于 SQL 解析部分,依赖 MP 分页插件
     */
    @Bean
    public PaginationInterceptor paginationInterceptor(MpTenantHandler mpTenantHandler) {
        PaginationInterceptor paginationInterceptor = new PaginationInterceptor();

        List<ISqlParser> sqlParserList = new ArrayList<>();
        TenantSqlParser tenantSqlParser = new MpTenantParserHandler();
        // 解析处理拦截器
        tenantSqlParser.setTenantHandler(mpTenantHandler);

        sqlParserList.add(tenantSqlParser);
        paginationInterceptor.setSqlParserList(sqlParserList);
        return paginationInterceptor;
    }
}

租户字段处理

代码100分public class BaseEntity {
    /** 租户id */
    @TableField(insertStrategy = FieldStrategy.NEVER,
            updateStrategy = FieldStrategy.NEVER)
    private Long tid;
}

插入和更新时租户字段由租户处理器接管,mp自带的增删改查不再有租户字段

查询

插入

/**
     * insert 语句处理
     */
    @Override
    public void processInsert(Insert insert) {
        if (tenantHandler.doTableFilter(insert.getTable().getName())) {
            // 过滤退出执行
            return;
        }
        insert.getColumns().add(new Column(tenantHandler.getTenantIdColumn()));
        if (insert.getSelect() != null) {
            processPlainSelect((PlainSelect) insert.getSelect().getSelectBody(), true);
        } else if (insert.getItemsList() != null) {
            // fixed github pull/295
            ItemsList itemsList = insert.getItemsList();
            if (itemsList instanceof MultiExpressionList) {
                ((MultiExpressionList) itemsList).getExprList().forEach(el -> el.getExpressions().add(tenantHandler.getTenantId(false)));
            } else {
                ((ExpressionList) insert.getItemsList()).getExpressions().add(tenantHandler.getTenantId(false));
            }
        } else {
            throw ExceptionUtils.mpe("Failed to process multiple-table update, please exclude the tableName or statementId");
        }
    }

更新

/**
 * update 语句处理
 */
@Override
public void processUpdate(Update update) {
    final Table table = update.getTable();
    if (tenantHandler.doTableFilter(table.getName())) {
        // 过滤退出执行
        return;
    }
    update.setWhere(this.andExpression(table, update.getWhere()));
}

删除

/**
 * delete 语句处理
 */
@Override
public void processDelete(Delete delete) {
    if (tenantHandler.doTableFilter(delete.getTable().getName())) {
        // 过滤退出执行
        return;
    }
    delete.setWhere(this.andExpression(delete.getTable(), delete.getWhere()));
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/8982.html

(0)
上一篇 2023-02-18
下一篇 2023-02-18

相关推荐

  • 深入理解Python xrange

    深入理解Python xrangePython语言拥有众多的数据结构,其中一个广受欢迎的数据结构是range。它可以生成一个从起始数字开始,到结束数字结束的整数序列,步长为1。在Python 2.x版本中,xrange被引入,是range的替代品。虽然它们有许多相同之处,但也有一些重要的区别。本文将深入介绍Python xrange。

    2024-04-17
    19
  • 数据库死锁分析(行锁、间隙锁)[亲测有效]

    数据库死锁分析(行锁、间隙锁)[亲测有效]分享遇到过的一种间隙锁导致的死锁案例。文后有总结知识供参考 日志出现:Deadlock found when trying to get lock; try restarting transactio

    2023-05-09
    79
  • oracle存储过程迁移达梦心得「建议收藏」

    oracle存储过程迁移达梦心得「建议收藏」这几天把项目的存储过程从oracle迁移到了达梦8,记录一下心得。国产数据库做到这样,已经算很了不起了,跟oracle兼容性确实很高。 但还是有一些细节没做好,主要是出错提示不友好,另外一个网上的资料

    2023-03-04
    99
  • 数据采集实战(四)-[通俗易懂]

    数据采集实战(四)-[通俗易懂]1. 概述 前段时间在看一本很多人推荐的线性代数教材《线性代数应该这样学》第三版,这一版每个章节都有大量的习题。 官方网站上虽然按照章节提供了习题的答案,一来因为网站是国外的,访问不流畅,二来答案中还

    2023-04-24
    99
  • python权重排名(python怎么排名次)

    python权重排名(python怎么排名次)1、首先创建一个手机的列表,用print函数将列表打印到屏幕上。

    2023-12-01
    68
  • MySQL到底能有多少个字段[亲测有效]

    MySQL到底能有多少个字段[亲测有效]今天技术讨论群里 “一切随遇而安”同学看书时出现一个疑问,一个MySQL的表中到底可以有多少个字段?带着这个疑问,我们展开了探讨,也接着讨论了一个单字段长度的问题。 1. 官方文档说明 官方文档的内容

    2023-02-12
    95
  • [20221020]奇怪的增量备份.txt

    [20221020]奇怪的增量备份.txt[20221020]奇怪的增量备份.txt//生产系统做增量备份遇到的怪异问题,给奇葩的运维人员狠狠地涮了一把,做一个记录:1.环境:SYS@192.168.100.235:1521/orcl&g

    2023-06-12
    80
  • 今日讨论荣耀5G手机有哪些比较不错的推荐「终于解决」

    今日讨论荣耀5G手机有哪些比较不错的推荐「终于解决」  最近这一段时间,各大手机厂商一直发布争相发布新款5G手机,再度掀起了5G手机的热潮,看到很多网友在网上问5G手机有哪些不错的选择,我作为一个资深科技宅,现在为大家来推荐几款好用的5G手机。   …

    2023-02-20
    91

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注