|
|
|
@ -2,7 +2,6 @@ package org.dromara.common.mybatis.handler;
|
|
|
|
|
|
|
|
|
|
import cn.hutool.core.annotation.AnnotationUtil;
|
|
|
|
|
import cn.hutool.core.collection.CollUtil;
|
|
|
|
|
import cn.hutool.core.util.ClassUtil;
|
|
|
|
|
import cn.hutool.core.util.ObjectUtil;
|
|
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
|
|
import net.sf.jsqlparser.JSQLParserException;
|
|
|
|
@ -10,6 +9,7 @@ import net.sf.jsqlparser.expression.Expression;
|
|
|
|
|
import net.sf.jsqlparser.expression.Parenthesis;
|
|
|
|
|
import net.sf.jsqlparser.expression.operators.conditional.AndExpression;
|
|
|
|
|
import net.sf.jsqlparser.parser.CCJSqlParserUtil;
|
|
|
|
|
import org.apache.ibatis.io.Resources;
|
|
|
|
|
import org.dromara.common.core.exception.ServiceException;
|
|
|
|
|
import org.dromara.common.core.utils.SpringUtils;
|
|
|
|
|
import org.dromara.common.core.utils.StreamUtils;
|
|
|
|
@ -21,16 +21,26 @@ import org.dromara.common.mybatis.helper.DataPermissionHelper;
|
|
|
|
|
import org.dromara.common.satoken.utils.LoginHelper;
|
|
|
|
|
import org.dromara.system.api.model.LoginUser;
|
|
|
|
|
import org.dromara.system.api.model.RoleDTO;
|
|
|
|
|
import org.springframework.context.ConfigurableApplicationContext;
|
|
|
|
|
import org.springframework.context.expression.BeanFactoryResolver;
|
|
|
|
|
import org.springframework.core.io.Resource;
|
|
|
|
|
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
|
|
|
|
|
import org.springframework.core.io.support.ResourcePatternResolver;
|
|
|
|
|
import org.springframework.core.type.ClassMetadata;
|
|
|
|
|
import org.springframework.core.type.classreading.CachingMetadataReaderFactory;
|
|
|
|
|
import org.springframework.expression.BeanResolver;
|
|
|
|
|
import org.springframework.expression.ExpressionParser;
|
|
|
|
|
import org.springframework.expression.ParserContext;
|
|
|
|
|
import org.springframework.expression.common.TemplateParserContext;
|
|
|
|
|
import org.springframework.expression.spel.standard.SpelExpressionParser;
|
|
|
|
|
import org.springframework.expression.spel.support.StandardEvaluationContext;
|
|
|
|
|
import org.springframework.util.ClassUtils;
|
|
|
|
|
|
|
|
|
|
import java.lang.reflect.Method;
|
|
|
|
|
import java.util.*;
|
|
|
|
|
import java.util.Arrays;
|
|
|
|
|
import java.util.HashSet;
|
|
|
|
|
import java.util.Map;
|
|
|
|
|
import java.util.Set;
|
|
|
|
|
import java.util.concurrent.ConcurrentHashMap;
|
|
|
|
|
import java.util.function.Function;
|
|
|
|
|
|
|
|
|
@ -58,9 +68,13 @@ public class PlusDataPermissionHandler {
|
|
|
|
|
*/
|
|
|
|
|
private final BeanResolver beanResolver = new BeanFactoryResolver(SpringUtils.getBeanFactory());
|
|
|
|
|
|
|
|
|
|
public PlusDataPermissionHandler(String mapperPackage) {
|
|
|
|
|
scanMapperClasses(mapperPackage);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public Expression getSqlSegment(Expression where, String mappedStatementId, boolean isSelect) {
|
|
|
|
|
DataColumn[] dataColumns = findAnnotation(mappedStatementId);
|
|
|
|
|
DataPermission dataPermission = getDataPermission(mappedStatementId);
|
|
|
|
|
LoginUser currentUser = DataPermissionHelper.getVariable("user");
|
|
|
|
|
if (ObjectUtil.isNull(currentUser)) {
|
|
|
|
|
currentUser = LoginHelper.getLoginUser();
|
|
|
|
@ -70,7 +84,7 @@ public class PlusDataPermissionHandler {
|
|
|
|
|
if (LoginHelper.isSuperAdmin() || LoginHelper.isTenantAdmin()) {
|
|
|
|
|
return where;
|
|
|
|
|
}
|
|
|
|
|
String dataFilterSql = buildDataFilter(dataColumns, isSelect);
|
|
|
|
|
String dataFilterSql = buildDataFilter(dataPermission.value(), isSelect);
|
|
|
|
|
if (StringUtils.isBlank(dataFilterSql)) {
|
|
|
|
|
return where;
|
|
|
|
|
}
|
|
|
|
@ -144,43 +158,64 @@ public class PlusDataPermissionHandler {
|
|
|
|
|
return "";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public DataColumn[] findAnnotation(String mappedStatementId) {
|
|
|
|
|
StringBuilder sb = new StringBuilder(mappedStatementId);
|
|
|
|
|
int index = sb.lastIndexOf(".");
|
|
|
|
|
String clazzName = sb.substring(0, index);
|
|
|
|
|
String methodName = sb.substring(index + 1, sb.length());
|
|
|
|
|
Class<?> clazz;
|
|
|
|
|
/**
|
|
|
|
|
* 通过 mapperPackage 设置的扫描包 扫描缓存有注解的方法与类
|
|
|
|
|
*/
|
|
|
|
|
private void scanMapperClasses(String mapperPackage) {
|
|
|
|
|
PathMatchingResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();
|
|
|
|
|
CachingMetadataReaderFactory factory = new CachingMetadataReaderFactory();
|
|
|
|
|
String[] packagePatternArray = StringUtils.splitPreserveAllTokens(mapperPackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS);
|
|
|
|
|
String classpath = ResourcePatternResolver.CLASSPATH_ALL_URL_PREFIX;
|
|
|
|
|
try {
|
|
|
|
|
clazz = ClassUtil.loadClass(clazzName);
|
|
|
|
|
for (String packagePattern : packagePatternArray) {
|
|
|
|
|
String path = ClassUtils.convertClassNameToResourcePath(packagePattern);
|
|
|
|
|
Resource[] resources = resolver.getResources(classpath + path + "/*.class");
|
|
|
|
|
for (Resource resource : resources) {
|
|
|
|
|
ClassMetadata classMetadata = factory.getMetadataReader(resource).getClassMetadata();
|
|
|
|
|
Class<?> clazz = Resources.classForName(classMetadata.getClassName());
|
|
|
|
|
findAnnotation(clazz);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} catch (Exception e) {
|
|
|
|
|
return null;
|
|
|
|
|
log.error("初始化数据安全缓存时出错:{}", e.getMessage());
|
|
|
|
|
}
|
|
|
|
|
List<Method> methods = Arrays.stream(ClassUtil.getDeclaredMethods(clazz))
|
|
|
|
|
.filter(method -> method.getName().equals(methodName)).toList();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void findAnnotation(Class<?> clazz) {
|
|
|
|
|
DataPermission dataPermission;
|
|
|
|
|
// 获取方法注解
|
|
|
|
|
for (Method method : methods) {
|
|
|
|
|
dataPermission = dataPermissionCacheMap.get(mappedStatementId);
|
|
|
|
|
if (ObjectUtil.isNotNull(dataPermission)) {
|
|
|
|
|
return dataPermission.value();
|
|
|
|
|
for (Method method : clazz.getMethods()) {
|
|
|
|
|
if (method.isDefault() || method.isVarArgs()) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
String mappedStatementId = clazz.getName() + "." + method.getName();
|
|
|
|
|
if (AnnotationUtil.hasAnnotation(method, DataPermission.class)) {
|
|
|
|
|
dataPermission = AnnotationUtil.getAnnotation(method, DataPermission.class);
|
|
|
|
|
dataPermissionCacheMap.put(mappedStatementId, dataPermission);
|
|
|
|
|
return dataPermission.value();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
dataPermission = dataPermissionCacheMap.get(clazz.getName());
|
|
|
|
|
if (ObjectUtil.isNotNull(dataPermission)) {
|
|
|
|
|
return dataPermission.value();
|
|
|
|
|
}
|
|
|
|
|
// 获取类注解
|
|
|
|
|
if (AnnotationUtil.hasAnnotation(clazz, DataPermission.class)) {
|
|
|
|
|
dataPermission = AnnotationUtil.getAnnotation(clazz, DataPermission.class);
|
|
|
|
|
dataPermissionCacheMap.put(clazz.getName(), dataPermission);
|
|
|
|
|
return dataPermission.value();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public DataPermission getDataPermission(String mapperId) {
|
|
|
|
|
if (dataPermissionCacheMap.containsKey(mapperId)) {
|
|
|
|
|
return dataPermissionCacheMap.get(mapperId);
|
|
|
|
|
}
|
|
|
|
|
String clazzName = mapperId.substring(0, mapperId.lastIndexOf("."));
|
|
|
|
|
if (dataPermissionCacheMap.containsKey(clazzName)) {
|
|
|
|
|
return dataPermissionCacheMap.get(clazzName);
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 是否无效
|
|
|
|
|
*/
|
|
|
|
|
public boolean invalid(String mapperId) {
|
|
|
|
|
return getDataPermission(mapperId) == null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|