springboot通用分支處理超級(jí)管理員權(quán)限邏輯
前言
當(dāng)引入登錄模塊后我們需要做菜單。而菜單自然需要權(quán)限的參與,我們?cè)趕pringboot中設(shè)計(jì)的權(quán)限細(xì)粒度還算是比較細(xì)的。當(dāng)我們查詢菜單是需要根據(jù)權(quán)限查找對(duì)應(yīng)的菜單。但是在springboot中我設(shè)計(jì)了一個(gè)底層超級(jí)管理員
- 先來(lái)看看我一開始實(shí)現(xiàn)這個(gè)超級(jí)管理員菜單獲取的部分代碼
if (SecurityUtils.getSubject().hasRole(RoleList.SUPERADMIN)) { listemp = customMapper.selectRootMenusByRoleIdList(null,null, null, null); } else { listemp = customMapper.selectRootMenusByRoleIdList(roleList, oauthClientId,null, moduleCodes); }
這樣實(shí)現(xiàn)是很正常的思路,通過(guò)判斷角色是否是超級(jí)管理員來(lái)做分支執(zhí)行思路,但是超級(jí)管理員可能涉及到多個(gè)地方如果在每個(gè)地方都這樣if else執(zhí)行的,我覺(jué)得有點(diǎn)low, 所以我決定改造一下。不夠最終執(zhí)行的思路依然是if else判斷 。 只不過(guò)讓我們?cè)诖a層面上功能間不在那么雜糅在一起
自定義注解
首先我需要兩個(gè)注解,SuperDirection
和SuperDirectionHandler
分別表示需要判斷超級(jí)管理員分支和具體管理員分支的目標(biāo)函數(shù) 。 這句話說(shuō)的還是有點(diǎn)抽象的,容我慢慢道來(lái)!
SuperDirection
@Target({ElementType.TYPE, ElementType.METHOD}) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface SuperDirection { String value() default StringUtils.EMPTY; }
SuperDirectionHandler
@Target({ElementType.TYPE, ElementType.METHOD}) @Retention(RetentionPolicy.RUNTIME) @Documented @Component public @interface SuperDirectionHandler { }
作用
SuperDirection
是用于表明該方法需要進(jìn)行判斷超級(jí)管理員,而value值存儲(chǔ)的就是判斷的表達(dá)式。關(guān)于這個(gè)表達(dá)式我們后面介紹
SuperDirectionHandler
我們不難發(fā)現(xiàn)他沒(méi)有實(shí)際屬性但是多了一個(gè)@Component
注解。目的是方便Spring管理該注解;這樣我們就可以通過(guò)Spring來(lái)獲取被該注解標(biāo)注的類了。
位置
該注解釋放給全局使用的,在maltcloud結(jié)構(gòu)介紹中我們知道org.framework.core模塊是所有模塊的基石,所以這兩個(gè)注解我選擇在org.framework.core
模塊中
切面
- 我們想在方法執(zhí)行前進(jìn)行條件判斷,可選方案有很多我們可以在filter中攔截方法進(jìn)行判斷選擇執(zhí)行哪一個(gè),但是過(guò)濾器中我們無(wú)法直接獲取到方法的相關(guān)信息,注意這里說(shuō)的是無(wú)法直接獲取,如果你想在filter中實(shí)現(xiàn)也不是不行,這種方案感興趣的可以試試
- spring的另外一個(gè)特性切面正好符合我們的需求,我們只需要在aroud環(huán)繞方法中實(shí)現(xiàn)我們的需要。
- 首先我們定義一個(gè)切點(diǎn),切點(diǎn)攔截所有被
SuperDirection
注解標(biāo)注的類或者方法。
/**定義一個(gè)切點(diǎn); 攔截所有帶有SuperDirection注解的類和方法*/ @Pointcut("@annotation(com.github.zxhtom.core.annotaion.SuperDirection) || @within(com.github.zxhtom.core.annotaion.SuperDirection)") public void direction() { }
- 這里稍作一下解釋
@annotation
用于標(biāo)識(shí)方法上的SuperDirection
,@within
用于標(biāo)識(shí)在類上的SuperDirection
。 - 正常我們的一個(gè)業(yè)務(wù)處理都是放在service層的,spring中的三層架構(gòu)的service正常是實(shí)現(xiàn)一個(gè)接口然后實(shí)現(xiàn)。所以我們這里在切面中先獲取被攔截對(duì)象實(shí)現(xiàn)的接口。獲取到接口信息我們通過(guò)接口信息獲取該接口在spring中的其它實(shí)現(xiàn)類
- 在spring容器中提供了獲取bean集合的方法,加上我們maltcloud中實(shí)現(xiàn)了獲取
ApplicationContext
的工具類,所以我們通過(guò)如下來(lái)獲取bean集合
Map<String, ?> beansOfType = ApplicationContextUtil.getApplicationContext().getBeansOfType(接口class);
- 獲取到集合了,此時(shí)我們還是無(wú)法確定哪一個(gè)實(shí)現(xiàn)類使我們替補(bǔ)執(zhí)行超級(jí)管理員的bean 。 這就需要我們
SuperDirectionHandler
注解了。 - 這個(gè)時(shí)候我們?cè)谕ㄟ^(guò)Spring獲取被該注解標(biāo)識(shí)的類。這個(gè)時(shí)候獲取到很多不想關(guān)類,我們?cè)诤蜕厦娴?code>beansOfType進(jìn)行比對(duì)。就可以確定哪一個(gè)實(shí)現(xiàn)bean是我們需要的。
@Around("direction()") public Object aroud(ProceedingJoinPoint pjp) throws Throwable { Class<?>[] inters = pjp.getTarget().getClass().getInterfaces(); for (Class<?> inter : inters) { Map<String, ?> beansOfType = ApplicationContextUtil.getApplicationContext().getBeansOfType(inter); Map<String, Object> beansWithAnnotation = ApplicationContextUtil.getApplicationContext().getBeansWithAnnotation(SuperDirectionHandler.class); for (Map.Entry<String, ?> entry : beansOfType.entrySet()) { if (beansWithAnnotation.containsKey(entry.getKey())) { try { return doOthersHandler(entry.getValue(), pjp); } catch (Exception e) { log.error("分支執(zhí)行失敗,系統(tǒng)判定執(zhí)行原有分支...."); } } } } return pjp.proceed(); }
- 當(dāng)確定執(zhí)行類之后,我們只需要攜帶著該bean ,在根據(jù)
SuperDirection
上的表達(dá)式進(jìn)行判斷是執(zhí)行超級(jí)管理員實(shí)現(xiàn)類還是原有實(shí)現(xiàn)類的方法了。
條件判斷
- 在Aspect執(zhí)行中原有方法的執(zhí)行很簡(jiǎn)單,只需要
pjp.proceed()
就可以了。所以這里我們先獲取一下上面獲取到的超級(jí)管理員實(shí)現(xiàn)bean的對(duì)應(yīng)方法吧。
MethodSignature msig = (MethodSignature) pjp.getSignature(); Method targetMethod = value.getClass().getDeclaredMethod(msig.getName(),msig.getParameterTypes());
- 然后我們?cè)讷@取
SuperDirection
注解信息,因?yàn)樵撟⒔饪赡茉陬惿?,也可能在方法上,所以這里我們需要處理下
SuperDirection superDirection = null; superDirection = targetMethod.getAnnotation(SuperDirection.class); if (superDirection == null) { superDirection = pjp.getTarget().getClass().getAnnotation(SuperDirection.class); }
- 最終我們通過(guò)注解表達(dá)式判斷執(zhí)行情況
if(selectAnnotationChoiceDo(superDirection)){ //如果表達(dá)式驗(yàn)證通過(guò),則執(zhí)行替補(bǔ)bean實(shí)現(xiàn)類 return targetMethod.invoke(value,pjp.getArgs()); } //否則執(zhí)行原有bean實(shí)現(xiàn)類 return pjp.proceed();
表達(dá)式解析
- 表達(dá)式解析涉及兩個(gè)模塊,一個(gè)是登錄模塊中獲取當(dāng)前登錄用戶的角色,另外一個(gè)是我們上面提到的表達(dá)式解析
- 獲取當(dāng)前登錄用戶信息類我在
org.framework.web
中提供了bean 。 具體的實(shí)現(xiàn)由各個(gè)登錄子模塊負(fù)責(zé)去實(shí)現(xiàn),這里我們只需要引入spirng bean使用就要可以了。這里充分體現(xiàn)了模塊拆分的好處了。 - 至于表達(dá)式解析,我選擇放在本模塊中
org.framework.commons
。 這里目前簡(jiǎn)單提供了幾個(gè)表達(dá)式解析。
public interface RootChoiceExpression { public boolean haslogined(); public boolean hasRole(String role); public boolean hasAnyRole(String... roles); }
- 他們分別是驗(yàn)證是否登錄、是否擁有角色和角色組。關(guān)于他的視線最終也還是依賴上面提到的
org.framework.web
模塊中的登錄用戶的信息類
@Service public class DefaultChoiceExpression implements RootChoiceExpression { @Autowired OnlineSecurity onlineSecurity; @Override public boolean haslogined() { return onlineSecurity.getOnlinePrincipal()!=null; } @Override public boolean hasRole(String role) { return onlineSecurity.hasAnyRole(role); } @Override public boolean hasAnyRole(String... roles) { return onlineSecurity.hasAnyRole(roles); } }
- 這里也算是流出擴(kuò)展吧,后面根據(jù)項(xiàng)目需求我們可以重寫該表達(dá)式解析,在根據(jù)自己的業(yè)務(wù)進(jìn)行表達(dá)式新增。這里僅作為基礎(chǔ)功能
private boolean selectAnnotationChoiceDo(SuperDirection superDirection) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException { String value = superDirection.value(); if (StringUtils.isEmpty(value)) { return onlineSecurity.getRoleNames().contains(MaltcloudConstant.SUPERADMIN); } MethodInfo info = selectInfoFromExpression(value); Method declaredMethod = expression.getClass().getDeclaredMethod(info.getMethodName(), String.class); Object invoke = declaredMethod.invoke(expression, info.getArgs()); if (invoke != null && invoke.toString().equals("true")) { return true; } return false; }
- 首先根據(jù)正則解析出方法名和參數(shù)。然后根據(jù)反射調(diào)用我們spring中表達(dá)式bean去執(zhí)行我們?cè)?code>SuperDirection配置的表達(dá)式。通過(guò)上面我們又能發(fā)現(xiàn)目前表達(dá)式僅支持String傳參。因?yàn)樵?code>SuperDireciton傳遞過(guò)來(lái)的已經(jīng)是String了,所以在這里目前我還沒(méi)想到如何支持更多類型的數(shù)據(jù)。先埋坑吧!
- 該方法最終決定執(zhí)行原生方法還是替補(bǔ)方法。
演示使用
- 上面說(shuō)的那么枯燥主要是因?yàn)槭俏业囊粋€(gè)設(shè)計(jì)思路,下面我們來(lái)實(shí)操感受一下吧。
controller
- 首先我在controller中開發(fā)一個(gè)接口 。這里需要注意下因?yàn)槲覀兩厦鏁?huì)出現(xiàn)多個(gè)實(shí)現(xiàn)bean在spring中,所以我們?cè)谑褂眠@些接口的時(shí)候就不能單純的使用
@Autowired
了, 而需要通過(guò)beanName來(lái)使用了。這里名叫commonTestServiceImpl
的CommonTestService
接口的普通實(shí)現(xiàn)類,用于實(shí)現(xiàn)我們正常的操作。
@RestController @RequestMapping(value = "/demo/common") public class CommonController { @Qualifier(value = "commonTestServiceImpl") @Autowired CommonTestService commonTestService; @RequestMapping(value = "/test",method = RequestMethod.GET) public void test() { commonTestService.test(); } }
service
- 這里有兩個(gè)實(shí)現(xiàn)類分別是
CommonTestServiceImpl
、CommonTest2ServiceImpl
@Service @SuperDirectionHandler public class CommonTest2ServiceImpl implements CommonTestService { @Override public void test() { System.out.println("hello test 2"); } }
@Service @SuperDirection(value = "") public class CommonTestServiceImpl implements CommonTestService { @Override public void test() { System.out.println("hello i am test ing ..."); } }
- 在controller層我們使用的是
CommonTestServiceImpl
用來(lái)實(shí)現(xiàn)正常的邏輯。而CommonTest2ServiceImpl
是針對(duì)超級(jí)管理員做的操作。我們就可以進(jìn)行如上的配置。在SuperDirection
中配置空值標(biāo)識(shí)判斷超級(jí)管理員進(jìn)行分支執(zhí)行。你也可以配置目前支持的表達(dá)式,我這里簡(jiǎn)單點(diǎn)了。 - 然后通過(guò)
SuperDirectionHandler
標(biāo)識(shí)ConmmonTest2ServiceImpl
是替補(bǔ)執(zhí)行。
測(cè)試
- 由于為了簡(jiǎn)單測(cè)試,我在org.framework.demo.common模塊中還沒(méi)有引入login模塊,所以此時(shí)登錄用戶獲取類還是我們默認(rèn)的
OnlineSecurityImpl
- 通過(guò)上面代碼我們可以看出來(lái)我們當(dāng)前是沒(méi)有角色的,所以我們可以理解成調(diào)用接口是沒(méi)有超級(jí)管理員接口。那么就會(huì)執(zhí)行我們
CommonTestServiceImpl
中的方法。然后我們?cè)趯⑦@里的hasAnyRole改成true , 會(huì)發(fā)現(xiàn)就會(huì)執(zhí)行CommonTest2ServiceImpl
里的方法。
總結(jié)
- 經(jīng)過(guò)上面這么折騰,我們就可以在涉及到超級(jí)管理員的地方重新實(shí)現(xiàn)一下,然后再原有的實(shí)現(xiàn)類中只需要專注我們權(quán)限架構(gòu)中的規(guī)則進(jìn)行數(shù)據(jù)庫(kù)查詢等操作了,而不需要向我一開始那樣為超級(jí)管理員進(jìn)行特殊操作。如果后續(xù)我們需要為特殊用戶進(jìn)行特殊開發(fā)。我們就可以擴(kuò)展我們的表達(dá)式解析然后再開發(fā)我們備用接口就可以了。
- 這個(gè)思路主要來(lái)自于Spring Cloud 中的OpenFeign 的容災(zāi)降級(jí)的思路。
以上就是springboot通用分支處理超級(jí)管理員權(quán)限邏輯的詳細(xì)內(nèi)容,更多關(guān)于springboot通用分支處理權(quán)限的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- SpringBoot如何整合Springsecurity實(shí)現(xiàn)數(shù)據(jù)庫(kù)登錄及權(quán)限控制
- Springboot+Spring Security實(shí)現(xiàn)前后端分離登錄認(rèn)證及權(quán)限控制的示例代碼
- SpringBoot整合Security實(shí)現(xiàn)權(quán)限控制框架(案例詳解)
- SpringBoot整合Shiro實(shí)現(xiàn)權(quán)限控制的代碼實(shí)現(xiàn)
- Springboot+mybatis-plus+注解實(shí)現(xiàn)數(shù)據(jù)權(quán)限隔離
- springboot+springsecurity如何實(shí)現(xiàn)動(dòng)態(tài)url細(xì)粒度權(quán)限認(rèn)證
相關(guān)文章
利用java反射機(jī)制調(diào)用類的私有方法(推薦)
下面小編就為大家?guī)?lái)一篇利用java反射機(jī)制調(diào)用類的私有方法(推薦)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-08-08關(guān)于在IDEA中SpringBoot項(xiàng)目中activiti工作流的使用詳解
這篇文章主要介紹了關(guān)于在IDEA中SpringBoot項(xiàng)目中activiti工作流的使用詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-088個(gè)Spring事務(wù)失效場(chǎng)景詳解
相信大家對(duì)Spring種事務(wù)的使用并不陌生,但是你可能只是停留在基礎(chǔ)的使用層面上。今天,我們就簡(jiǎn)單來(lái)說(shuō)下Spring事務(wù)的原理,然后總結(jié)一下spring事務(wù)失敗的場(chǎng)景,并提出對(duì)應(yīng)的解決方案,需要的可以參考一下2022-12-12詳解Java的Struts框架以及相關(guān)的MVC設(shè)計(jì)理念
這篇文章主要介紹了詳解Java的Struts框架以及相關(guān)的MVC設(shè)計(jì)理念,Struts是Java的SSH三大web開發(fā)框架之一,需要的朋友可以參考下2015-12-12Java SSM框架(Spring+SpringMVC+MyBatis)搭建過(guò)程
最近一段時(shí)間搭建了ssm環(huán)境,并測(cè)試了幾個(gè)小項(xiàng)目,下面小編通過(guò)圖文并茂的形式給大家分享Java SSM框架(Spring+SpringMVC+MyBatis)搭建過(guò)程,需要的朋友參考下吧2017-11-11IntelliJ IDEA 使用經(jīng)驗(yàn)總結(jié)(推薦)
這篇文章主要介紹了IntelliJ IDEA 使用經(jīng)驗(yàn)總結(jié),非常不錯(cuò),具有參考價(jià)值,需要的朋友可以參考下2018-02-02