电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

spring mvc url匹配禁用后綴訪問操作

瀏覽:8日期:2023-06-30 17:48:53
spring mvc url匹配禁用后綴訪問

在spring mvc中默認 訪問url 加任意后綴名都能訪問

比如:你想訪問 /login ,但是通過 /login.do /login.action /login.json 都能訪問

通常來說可能沒有影響,但對于權限控制,這就嚴重了。

權限控制通常有兩種思路:1)弱權限控制

允許所有url通過,僅對個別重要的url做權限控制。此種方式比較簡單,不需要對所有url資源進行配置,只配置重要的資源。

2)強權限控制

默認禁止所有url請求通過,僅開放授權的資源。此種方式對所有的url資源進行控制。在系統種需要整理所有的請求,或者某一目錄下所有的url資源。這種方式安全控制比較嚴格,操作麻煩,但相對安全。

如果用第二種方式,則上面spring mvc的訪問策略對安全沒有影響。

但如果用第一種安全策略,則會有很大的安全風險。

例如:我們控制了/login 的訪問,但是我們默認除/login的資源不受權限控制約束,那么攻擊者就可以用 /login.do /login.xxx 來訪問我們的資源。

在spring 3.1之后,url找對應方法的處理步驟,第一步,直接調用RequestMappingHandlerMapping查找到相應的處理方法,第二步,調用RequestMappingHandlerAdapter進行處理

我們在RequestMappingHandlerMapping中可以看到

/** * Whether to use suffix pattern match for registered file extensions only * when matching patterns to requests. * <p>If enabled, a controller method mapped to '/users' also matches to * '/users.json' assuming '.json' is a file extension registered with the * provided {@link #setContentNegotiationManager(ContentNegotiationManager) * contentNegotiationManager}. This can be useful for allowing only specific * URL extensions to be used as well as in cases where a '.' in the URL path * can lead to ambiguous interpretation of path variable content, (e.g. given * '/users/{user}' and incoming URLs such as '/users/john.j.joe' and * '/users/john.j.joe.json'). * <p>If enabled, this flag also enables * {@link #setUseSuffixPatternMatch(boolean) useSuffixPatternMatch}. The * default value is {@code false}. */public void setUseRegisteredSuffixPatternMatch(boolean useRegisteredSuffixPatternMatch) { this.useRegisteredSuffixPatternMatch = useRegisteredSuffixPatternMatch; this.useSuffixPatternMatch = (useRegisteredSuffixPatternMatch || this.useSuffixPatternMatch);}

那么如何來配置呢?

<mvc:annotation-driven> <mvc:path-matching suffix-pattern='false' /> </mvc:annotation-driven>

在匹配模式時是否使用后綴模式匹配,默認值為true。這樣你想訪問 /login ,通過 /login.* 就不能訪問了。

spring mvc 之 請求url 帶后綴的情況

RequestMappingInfoHandlerMapping 在處理http請求的時候, 如果 請求url 有后綴,如果找不到精確匹配的那個@RequestMapping方法。

那么,就把后綴去掉,然后.* 去匹配,這樣,一般都可以匹配。 比如有一個@RequestMapping('/rest'), 那么精確匹配的情況下, 只會匹配/rest請求。

但如果我前端發來一個 /rest.abcdef 這樣的請求, 又沒有配置 @RequestMapping('/rest.abcdef') 這樣映射的情況下, 那么@RequestMapping('/rest') 就會生效。

原理呢?處理鏈是這樣的:

at org.springframework.web.servlet.mvc.condition.PatternsRequestCondition.getMatchingPattern(PatternsRequestCondition.java:254)at org.springframework.web.servlet.mvc.condition.PatternsRequestCondition.getMatchingPatterns(PatternsRequestCondition.java:230)at org.springframework.web.servlet.mvc.condition.PatternsRequestCondition.getMatchingCondition(PatternsRequestCondition.java:210)at org.springframework.web.servlet.mvc.method.RequestMappingInfo.getMatchingCondition(RequestMappingInfo.java:214)at org.springframework.web.servlet.mvc.method.RequestMappingInfoHandlerMapping.getMatchingMapping(RequestMappingInfoHandlerMapping.java:79)at org.springframework.web.servlet.mvc.method.RequestMappingInfoHandlerMapping.getMatchingMapping(RequestMappingInfoHandlerMapping.java:56)at org.springframework.web.servlet.handler.AbstractHandlerMethodMapping.addMatchingMappings(AbstractHandlerMethodMapping.java:358)at org.springframework.web.servlet.handler.AbstractHandlerMethodMapping.lookupHandlerMethod(AbstractHandlerMethodMapping.java:328)at org.springframework.web.servlet.handler.AbstractHandlerMethodMapping.getHandlerInternal(AbstractHandlerMethodMapping.java:299)at org.springframework.web.servlet.handler.AbstractHandlerMethodMapping.getHandlerInternal(AbstractHandlerMethodMapping.java:57)at org.springframework.web.servlet.handler.AbstractHandlerMapping.getHandler(AbstractHandlerMapping.java:299)at org.springframework.web.servlet.DispatcherServlet.getHandler(DispatcherServlet.java:1104)at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:916)關鍵是PatternsRequestCondition, 具體來說是這個方法:

AbstractHandlerMethodMapping 的getHandlerInternal: protected HandlerMethod getHandlerInternal(HttpServletRequest request) throws Exception {String lookupPath = this.getUrlPathHelper().getLookupPathForRequest(request);if (this.logger.isDebugEnabled()) { this.logger.debug('Looking up handler method for path ' + lookupPath);}HandlerMethod handlerMethod = this.lookupHandlerMethod(lookupPath, request);// 這里是關鍵,它去尋找,找到了就找到了,找不到就不會再去尋找了if (this.logger.isDebugEnabled()) { if (handlerMethod != null) {this.logger.debug('Returning handler method [' + handlerMethod + ']'); } else {this.logger.debug('Did not find handler method for [' + lookupPath + ']'); }}return handlerMethod != null ? handlerMethod.createWithResolvedBean() : null; } protected HandlerMethod lookupHandlerMethod(String lookupPath, HttpServletRequest request) throws Exception {List<AbstractHandlerMethodMapping<T>.Match> matches = new ArrayList();List<T> directPathMatches = (List)this.urlMap.get(lookupPath); // directPathMatches, 直接匹配, 也可以說是 精確匹配if (directPathMatches != null) { this.addMatchingMappings(directPathMatches, matches, request);// 如果能夠精確匹配, 就會進來這里}if (matches.isEmpty()) { this.addMatchingMappings(this.handlerMethods.keySet(), matches, request);// 如果無法精確匹配, 就會進來這里}if (!matches.isEmpty()) { Comparator<AbstractHandlerMethodMapping<T>.Match> comparator = new AbstractHandlerMethodMapping.MatchComparator(this.getMappingComparator(request)); Collections.sort(matches, comparator); if (this.logger.isTraceEnabled()) {this.logger.trace('Found ' + matches.size() + ' matching mapping(s) for [' + lookupPath + '] : ' + matches); } AbstractHandlerMethodMapping<T>.Match bestMatch = (AbstractHandlerMethodMapping.Match)matches.get(0); if (matches.size() > 1) {AbstractHandlerMethodMapping<T>.Match secondBestMatch = (AbstractHandlerMethodMapping.Match)matches.get(1);if (comparator.compare(bestMatch, secondBestMatch) == 0) { Method m1 = bestMatch.handlerMethod.getMethod(); Method m2 = secondBestMatch.handlerMethod.getMethod(); throw new IllegalStateException('Ambiguous handler methods mapped for HTTP path ’' + request.getRequestURL() + '’: {' + m1 + ', ' + m2 + '}');} } this.handleMatch(bestMatch.mapping, lookupPath, request); return bestMatch.handlerMethod;} else { return this.handleNoMatch(this.handlerMethods.keySet(), lookupPath, request);} }public List<String> getMatchingPatterns(String lookupPath) {List<String> matches = new ArrayList();Iterator var3 = this.patterns.iterator();while(var3.hasNext()) { String pattern = (String)var3.next(); // pattern 是 @RequestMapping 提供的映射 String match = this.getMatchingPattern(pattern, lookupPath); // lookupPath + .* 后能夠匹配pattern, 那么就不為空 if (match != null) {matches.add(match);// 對于有后綴的情況, .* 后 }}Collections.sort(matches, this.pathMatcher.getPatternComparator(lookupPath));return matches; } 最關鍵是這里 getMatchingPatterns : private String getMatchingPattern(String pattern, String lookupPath) {if (pattern.equals(lookupPath)) { return pattern;} else { if (this.useSuffixPatternMatch) {if (!this.fileExtensions.isEmpty() && lookupPath.indexOf(46) != -1) { Iterator var5 = this.fileExtensions.iterator(); while(var5.hasNext()) {String extension = (String)var5.next();if (this.pathMatcher.match(pattern + extension, lookupPath)) { return pattern + extension;} }} else { boolean hasSuffix = pattern.indexOf(46) != -1; if (!hasSuffix && this.pathMatcher.match(pattern + '.*', lookupPath)) {return pattern + '.*'; // 關鍵是這里 }} } if (this.pathMatcher.match(pattern, lookupPath)) {return pattern; } else {return this.useTrailingSlashMatch && !pattern.endsWith('/') && this.pathMatcher.match(pattern + '/', lookupPath) ? pattern + '/' : null; }} }

而對于AbstractUrlHandlerMapping ,匹配不上就是匹配不上, 不會進行 +.* 后在匹配。

關鍵方法是這個:

protected Object lookupHandler(String urlPath, HttpServletRequest request) throws Exception {Object handler = this.handlerMap.get(urlPath);if (handler != null) { if (handler instanceof String) {String handlerName = (String)handler;handler = this.getApplicationContext().getBean(handlerName); } this.validateHandler(handler, request); return this.buildPathExposingHandler(handler, urlPath, urlPath, (Map)null);} else { List<String> matchingPatterns = new ArrayList(); Iterator var5 = this.handlerMap.keySet().iterator(); while(var5.hasNext()) {String registeredPattern = (String)var5.next();if (this.getPathMatcher().match(registeredPattern, urlPath)) { matchingPatterns.add(registeredPattern);} } String bestPatternMatch = null; Comparator<String> patternComparator = this.getPathMatcher().getPatternComparator(urlPath); if (!matchingPatterns.isEmpty()) {Collections.sort(matchingPatterns, patternComparator);if (this.logger.isDebugEnabled()) { this.logger.debug('Matching patterns for request [' + urlPath + '] are ' + matchingPatterns);}bestPatternMatch = (String)matchingPatterns.get(0); } if (bestPatternMatch != null) {handler = this.handlerMap.get(bestPatternMatch);String pathWithinMapping;if (handler instanceof String) { pathWithinMapping = (String)handler; handler = this.getApplicationContext().getBean(pathWithinMapping);}this.validateHandler(handler, request);pathWithinMapping = this.getPathMatcher().extractPathWithinPattern(bestPatternMatch, urlPath);Map<String, String> uriTemplateVariables = new LinkedHashMap();Iterator var9 = matchingPatterns.iterator();while(var9.hasNext()) { String matchingPattern = (String)var9.next(); if (patternComparator.compare(bestPatternMatch, matchingPattern) == 0) {Map<String, String> vars = this.getPathMatcher().extractUriTemplateVariables(matchingPattern, urlPath);Map<String, String> decodedVars = this.getUrlPathHelper().decodePathVariables(request, vars);uriTemplateVariables.putAll(decodedVars); }}if (this.logger.isDebugEnabled()) { this.logger.debug('URI Template variables for request [' + urlPath + '] are ' + uriTemplateVariables);}return this.buildPathExposingHandler(handler, bestPatternMatch, pathWithinMapping, uriTemplateVariables); } else {return null; }} }

當然, 或許我們可以設置自定義的PathMatcher ,從而到達目的。 默認的 是AntPathMatcher 。

以上為個人經驗,希望能給大家一個參考,也希望大家多多支持好吧啦網。

標簽: Spring
相關文章:
主站蜘蛛池模板: 板材品牌-中国胶合板行业十大品牌-环保板材-上海声达板材 | 南京泽朗生物科技有限公司-液体饮料代加工_果汁饮料代加工_固体饮料代加工 | 定量包装秤,吨袋包装称,伸缩溜管,全自动包装秤,码垛机器人,无锡市邦尧机械工程有限公司 | 步进驱动器「一体化」步进电机品牌厂家-一体式步进驱动 | TPE_TPE热塑性弹性体_TPE原料价格_TPE材料厂家-惠州市中塑王塑胶制品公司- 中塑王塑胶制品有限公司 | 全自动翻转振荡器-浸出式水平振荡器厂家-土壤干燥箱价格-常州普天仪器 | 在线钠离子分析仪-硅酸根离子浓度测定仪-油液水分测定仪价格-北京时代新维测控设备有限公司 | 成都APP开发-成都App定制-成都app开发公司-【未来久】 | 日本SMC气缸接头-速度控制阀-日本三菱伺服电机-苏州禾力自动化科技有限公司 | 电伴热系统施工_仪表电伴热保温箱厂家_沃安电伴热管缆工业技术(济南)有限公司 | 双相钢_双相不锈钢_双相钢圆钢棒_双相不锈钢报价「海新双相钢」 双能x射线骨密度检测仪_dxa骨密度仪_双能x线骨密度仪_品牌厂家【品源医疗】 | 电位器_轻触开关_USB连接器_广东精密龙电子科技有限公司 | 电磁辐射仪-电磁辐射检测仪-pm2.5检测仪-多功能射线检测仪-上海何亦仪器仪表有限公司 | 旅游规划_旅游策划_乡村旅游规划_景区规划设计_旅游规划设计公司-北京绿道联合旅游规划设计有限公司 | 电动高压冲洗车_价格-江苏速利达机车有限公司 | 河南砖机首页-全自动液压免烧砖机,小型砌块水泥砖机厂家[十年老厂] | 衬氟旋塞阀-卡套旋塞阀-中升阀门首页| 电缆接头_防水接头_电缆防水接头 - 乐清市新豪电气有限公司 | 防水套管-柔性防水套管-刚性防水套管-上海执品管件有限公司 | 杭州中央空调维修_冷却塔/新风机柜/热水器/锅炉除垢清洗_除垢剂_风机盘管_冷凝器清洗-杭州亿诺能源有限公司 | 不锈钢复合板厂家_钛钢复合板批发_铜铝复合板供应-威海泓方金属复合材料股份有限公司 | loft装修,上海嘉定酒店式公寓装修公司—曼城装饰| 广东健伦体育发展有限公司-体育工程配套及销售运动器材的体育用品服务商 | 整车VOC采样环境舱-甲醛VOC预处理舱-多舱法VOC检测环境仓-上海科绿特科技仪器有限公司 | 带锯机|木工带锯机圆木推台锯|跑车带锯机|河北茂业机械制造有限公司| | 耐酸碱胶管_耐腐蚀软管总成_化学品输送软管_漯河利通液压科技耐油耐磨喷砂软管|耐腐蚀化学软管 | 合同书格式和范文_合同书样本模板_电子版合同,找范文吧 | 昆山新莱洁净应用材料股份有限公司-卫生级蝶阀,无菌取样阀,不锈钢隔膜阀,换向阀,离心泵 | 上海律师咨询_上海法律在线咨询免费_找对口律师上策法网-策法网 广东高华家具-公寓床|学生宿舍双层铁床厂家【质保十年】 | 地图标注-手机导航电子地图如何标注-房地产商场地图标记【DiTuBiaoZhu.net】 | 房车价格_依维柯/大通/东风御风/福特全顺/江铃图片_云梯搬家车厂家-程力专用汽车股份有限公司 | 科研ELISA试剂盒,酶联免疫检测试剂盒,昆虫_植物ELISA酶免试剂盒-上海仁捷生物科技有限公司 | _网名词典_网名大全_qq网名_情侣网名_个性网名 | 安全,主动,被动,柔性,山体滑坡,sns,钢丝绳,边坡,防护网,护栏网,围栏,栏杆,栅栏,厂家 - 护栏网防护网生产厂家 | 上海噪音治理公司-专业隔音降噪公司-中广通环保 | 杭州公司变更法人-代理记账收费价格-公司注销代办_杭州福道财务管理咨询有限公司 | 桂林腻子粉_内墙外墙抗裂砂浆腻子粉推荐广西鑫达涂料厂家供应 | 连续密炼机_双转子连续密炼机_连续式密炼机-南京永睿机械制造有限公司 | 扬尘监测_扬尘监测系统_带证扬尘监测设备 - 郑州港迪科技有限公司 | 石英砂矿石色选机_履带辣椒色选机_X光异物检测机-合肥幼狮光电科技 | 小型气象站_车载气象站_便携气象站-山东风途物联网 |