需求描述
最近公司要求用springSecurity搞一套权限管理(RBAC),单独就tb_permission表数据来说,我需要读取到Controller中所有添加了自定义注解的函数并将其路由存入数据库,需要达到效果如下:
自定义注解
import java.lang.annotation.*;
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface CheckPermission {
/**
* 描述
*
* @return
*/
String descrption() default "" ;
}
测试Controller
import com.dar.road.core.annotation.CheckPermission;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
/**
* @Author gjw
* @Date 2020/5/20 上午9:40
*/
@RestController
@RequestMapping("/test")
public class TestController {
@PostMapping("/justTest")
@CheckPermission(descrption = "测试-只是个测试")
public void justTest() {
System.out.println("我只是个测试方法");
}
}
AnnotationUtil工具类
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.io.Resource;
import org.springframework.core.io.ResourceLoader;
import org.springframework.core.io.support.ResourcePatternResolver;
import org.springframework.core.io.support.ResourcePatternUtils;
import org.springframework.core.type.AnnotationMetadata;
import org.springframework.core.type.MethodMetadata;
import org.springframework.core.type.classreading.CachingMetadataReaderFactory;
import org.springframework.core.type.classreading.MetadataReader;
import org.springframework.core.type.classreading.MetadataReaderFactory;
import org.springframework.stereotype.Component;
import org.springframework.web.bind.annotation.*;
import java.util.*;
/**
* @Author weiwenbin
* @Date 2020/5/14 下午5:31
*/
@Component
public class AnnotationUtil {
@Autowired
private ResourceLoader resourceLoader;
private static final String VALUE = "value";
/**
* 获取指定包下所有添加了执行注解的方法信息
* @param classPath 包名
* @param tagAnnotationClass 指定注解类型
* @param <T>
* @return
* @throws Exception
*/
public <T> Map<String, Map<String, Object>> getAllAddTagAnnotationUrl(String classPath, Class<T> tagAnnotationClass) throws Exception {
Map<String, Map<String, Object>> resMap = new HashMap<>();
ResourcePatternResolver resolver = ResourcePatternUtils.getResourcePatternResolver(resourceLoader);
MetadataReaderFactory metaReader = new CachingMetadataReaderFactory(resourceLoader);
Resource[] resources = resolver.getResources(classPath);
for (org.springframework.core.io.Resource r : resources) {
MetadataReader reader = metaReader.getMetadataReader(r);
resMap = resolveClass(reader, resMap, tagAnnotationClass);
}
return resMap;
}
private <T> Map<String, Map<String, Object>> resolveClass(
MetadataReader reader, Map<String, Map<String, Object>> resMap, Class<T> tagAnnotationClass)
throws Exception {
String tagAnnotationClassCanonicalName = tagAnnotationClass.getCanonicalName();
//获取注解元数据
AnnotationMetadata annotationMetadata = reader.getAnnotationMetadata();
//获取类中RequestMapping注解的属性
Map<String, Object> annotationAttributes =
annotationMetadata.getAnnotationAttributes(RequestMapping.class.getCanonicalName());
//若类无RequestMapping注解
if (annotationAttributes == null) return resMap;
//获取RequestMapping注解的value
String[] pathParents = (String[]) annotationAttributes.get(VALUE);
if (0 == pathParents.length) return resMap;
//获取RequestMapping注解的value
String pathParent = pathParents[0];
//获取当前类中已添加要扫描注解的方法
Set<MethodMetadata> annotatedMethods = annotationMetadata.getAnnotatedMethods(tagAnnotationClassCanonicalName);
for (MethodMetadata annotatedMethod : annotatedMethods) {
//获取当前方法中要扫描注解的属性
Map<String, Object> targetAttr = annotatedMethod.getAnnotationAttributes(tagAnnotationClassCanonicalName);
//获取当前方法中要xxxMapping注解的属性
Map<String, Object> mappingAttr = getPathByMethod(annotatedMethod);
if (mappingAttr == null){
continue;
}
String[] childPath = (String[]) mappingAttr.get(VALUE);
if (targetAttr == null || childPath == null || childPath.length == 0) {
continue;
}
String path = pathParent + childPath[0];
boolean isHas = resMap.containsKey(path);
if (isHas){
throw new Exception("重复定义了相同的映射关系");
}
resMap.put(path, targetAttr);
}
return resMap;
}
private Map<String, Object> getPathByMethod(MethodMetadata annotatedMethod) {
Map<String, Object> annotationAttributes = annotatedMethod.getAnnotationAttributes(GetMapping.class.getCanonicalName());
if (annotationAttributes != null && annotationAttributes.get(VALUE) != null) {
return annotationAttributes;
}
annotationAttributes = annotatedMethod.getAnnotationAttributes(PostMapping.class.getCanonicalName());
if (annotationAttributes != null && annotationAttributes.get(VALUE) != null) {
return annotationAttributes;
}
annotationAttributes = annotatedMethod.getAnnotationAttributes(DeleteMapping.class.getCanonicalName());
if (annotationAttributes != null && annotationAttributes.get(VALUE) != null) {
return annotationAttributes;
}
annotationAttributes = annotatedMethod.getAnnotationAttributes(PutMapping.class.getCanonicalName());
if (annotationAttributes != null && annotationAttributes.get(VALUE) != null) {
return annotationAttributes;
}
annotationAttributes = annotatedMethod.getAnnotationAttributes(RequestMapping.class.getCanonicalName());
return annotationAttributes;
}
}
调用
上面做完之后直接调用,获取到map如图, 我这边直接将map做一次处理后直接存库(tb_permission)。