/* * Copyright (c) 2018-2028, Chill Zhuang All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * Neither the name of the dreamlu.net developer nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * Author: Chill 庄骞 (smallchill@163.com) */ package org.springblade.core.datascope.interceptor; import com.baomidou.mybatisplus.core.toolkit.PluginUtils; import com.baomidou.mybatisplus.core.toolkit.StringPool; import lombok.RequiredArgsConstructor; import lombok.extern.slf4j.Slf4j; import org.apache.ibatis.executor.Executor; import org.apache.ibatis.mapping.BoundSql; import org.apache.ibatis.mapping.MappedStatement; import org.apache.ibatis.mapping.SqlCommandType; import org.apache.ibatis.mapping.StatementType; import org.apache.ibatis.session.ResultHandler; import org.apache.ibatis.session.RowBounds; import org.springblade.core.datascope.annotation.DataAuth; import org.springblade.core.datascope.handler.DataScopeHandler; import org.springblade.core.datascope.model.DataScopeModel; import org.springblade.core.datascope.props.DataScopeProperties; import org.springblade.core.mp.intercept.QueryInterceptor; import org.springblade.core.secure.BladeUser; import org.springblade.core.secure.utils.AuthUtil; import org.springblade.core.tool.utils.ClassUtil; import org.springblade.core.tool.utils.SpringUtil; import org.springblade.core.tool.utils.StringUtil; import java.lang.reflect.Method; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; /** * mybatis 数据权限拦截器 * * @author L.cm, Chill */ @Slf4j @RequiredArgsConstructor @SuppressWarnings({"rawtypes"}) public class DataScopeInterceptor implements QueryInterceptor { private final ConcurrentMap dataAuthMap = new ConcurrentHashMap<>(8); private final DataScopeHandler dataScopeHandler; private final DataScopeProperties dataScopeProperties; @Override public void intercept(Executor executor, MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) { //未启用则放行 if (!dataScopeProperties.getEnabled()) { return; } //未取到用户则放行 BladeUser bladeUser = AuthUtil.getUser(); if (bladeUser == null) { return; } if (SqlCommandType.SELECT != ms.getSqlCommandType() || StatementType.CALLABLE == ms.getStatementType()) { return; } String originalSql = boundSql.getSql(); //查找注解中包含DataAuth类型的参数 DataAuth dataAuth = findDataAuthAnnotation(ms); //注解为空并且数据权限方法名未匹配到,则放行 String mapperId = ms.getId(); String className = mapperId.substring(0, mapperId.lastIndexOf(StringPool.DOT)); String mapperName = ClassUtil.getShortName(className); String methodName = mapperId.substring(mapperId.lastIndexOf(StringPool.DOT) + 1); boolean mapperSkip = dataScopeProperties.getMapperKey().stream().noneMatch(methodName::contains) || dataScopeProperties.getMapperExclude().stream().anyMatch(mapperName::contains); if (dataAuth == null && mapperSkip) { return; } //创建数据权限模型 DataScopeModel dataScope = new DataScopeModel(); //若注解不为空,则配置注解项 if (dataAuth != null) { dataScope.setResourceCode(dataAuth.code()); dataScope.setScopeColumn(dataAuth.column()); dataScope.setScopeType(dataAuth.type().getType()); dataScope.setScopeField(dataAuth.field()); dataScope.setScopeValue(dataAuth.value()); } //获取数据权限规则对应的筛选Sql String sqlCondition = dataScopeHandler.sqlCondition(mapperId, dataScope, bladeUser, originalSql); if (!StringUtil.isBlank(sqlCondition)) { PluginUtils.MPBoundSql mpBoundSql = PluginUtils.mpBoundSql(boundSql); mpBoundSql.sql(sqlCondition); } } /** * 获取数据权限注解信息 * * @param mappedStatement mappedStatement * @return DataAuth */ private DataAuth findDataAuthAnnotation(MappedStatement mappedStatement) { String id = mappedStatement.getId(); return dataAuthMap.computeIfAbsent(id, (key) -> { String className = key.substring(0, key.lastIndexOf(StringPool.DOT)); String mapperBean = StringUtil.firstCharToLower(ClassUtil.getShortName(className)); Object mapper = SpringUtil.getBean(mapperBean); String methodName = key.substring(key.lastIndexOf(StringPool.DOT) + 1); Class[] interfaces = ClassUtil.getAllInterfaces(mapper); for (Class mapperInterface : interfaces) { for (Method method : mapperInterface.getDeclaredMethods()) { if (methodName.equals(method.getName()) && method.isAnnotationPresent(DataAuth.class)) { return method.getAnnotation(DataAuth.class); } } } return null; }); } }