Penetration_Testing_POC/books/Ognl小trick.html

450 lines
1.6 MiB
HTML
Raw Permalink Normal View History

<!DOCTYPE html> <html class style><!--
Page saved with SingleFile
url: https://forum.butian.net/share/2916
--><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>奇安信攻防社区-Ognl小trick</title>
<meta name=keywords content=奇安信,天眼,补天,漏洞,情报,攻防,安全>
<meta name=description content=奇安信攻防社区-Ognl小trick>
<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}h1,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){}ul{margin-top:0;margin-bottom:10px}@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}.col-md-9{width:75%}}@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:767px){}@media(max-width:767px){}@media(max-width:767px){}@media(min-width:768px) and (max-width:991px){}@media(min-width:768px) and (max
<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)
*/@-webkit-keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}@keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}</style>
<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}.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-keyword{color:#00f}.hljs-string,.hljs-title{color:#a31515}.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 a{background-color:transparent;color:var(--color-accent-fg);text-decoration:none}.markdown-body a:active,.markdown-body a:hover{outline-width:0}.markdown-body h1{margin:.67em 0;padding-bottom:.3em;font-size:2em;border-bottom:1px solid var(--color-border-muted)}.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 a:hover{text-decoration:underline}.markdown-body h1,.markdown-body h2,.markdown-body h3{margin-top:24px;margin-bottom:16px;font-weight:600;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 ul{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 !impor
<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
* Font Awesome Free 5.14.0 by @fontawesome - https://fontawesome.com
* License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License)
*/</style><style>/*!
* Font Awesome Free 5.14.0 by @fontawesome - https://fontawesome.com
* License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License)
*/</style><style>/*!
* Font Awesome Free 5.14.0 by @fontawesome - https://fontawesome.com
* License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License)
*/</style><style>/*!
* Font Awesome Free 5.14.0 by @fontawesome - https://fontawesome.com
* License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License)
*/@-webkit-keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}to{-webkit-transform:rotate(1turn);transform:rotate(1turn)}}@keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}to{-webkit-transform:rotate(1turn);transform:rotate(1turn)}}</style><meta name=referrer content=no-referrer><link rel=icon href=""><link rel=canonical href=https://forum.butian.net/share/2916><meta http-equiv=content-security-policy content="default-src 'none'; font-src 'self' data:; img-src 'self' data:; style-src 'unsafe-inline'; media-src 'self' data:; script-src 'unsafe-inline' data:; object-src 'self' data:; frame-src 'self' data:;"><style>img[src="data:,"],source[src="data:,"]{display:none!important}</style></head>
<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">Ognl小trick</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">
小trick一则
</div>
<textarea id=md_view_content style=display:none>OGNL小trick
==========
什么是OGNL表达式
----------
OGNL 是 Object-Graph Navigation Language对象图导航语言OGNL 最初是作为 WebWork 框架的一部分开发的,现在已成为 Apache Struts2 的一个关键组件这也就是为什么Struts2中的OGNL表达式漏洞这么多并被用于其他各种 Java 框架。
对象图导航
-----
OGNL的核心就在于`对象图导航`这个概念其实和数据结构中的图和很相似再OGNL或者面向对象编程语言中以引用作为边对象作为节点对象可以包含其他对象组合或与其他对象产生关联聚合从而在更大的对象图中形成子图。
以一段简单的代码作为解释:
```java
public class ObjectGraphDemo {
class People{
Car car;
}
class Car{
String carName = "BMW";
House house;
}
class House{
String houseName = "MyHouse";
}
public void printPeopleInfo(){
People people = new People();
System.out.println(people.car.carName);
System.out.println(people.car.house.houseName);
}
}
```
我们设定好了三个类,分别叫`People`、`Car`还有`House`,通过这三个类的引用,我们就可以体会到以引用作为边,对象作为节点的这么一种设计理念。
OGNL中都有什么
---------
想知道OGNL都有什么不如直接点进去看看我们都知道再`ognl.Ognl.getValue()`方法处会触发RCE漏洞那么也就是看在这里`getValue`方法接受了什么参数其实更简单的办法是去问问GPT
这里我将所有的`getValue`全都拷贝过来了,看起来很多其实全都是重载方法进行互相调用,通过看这些代码发现必不可少的三样东西是`expression`、`context`和`root`这也就是我们说的OGNL的三要素其实这些东西网上很多人都分析过了之所以这么啰嗦还是为了自己能够更好地理解。
```java
public static Object getValue(Object tree, Map context, Object root) throws OgnlException {
return getValue((Object)tree, (Map)context, root, (Class)null);
}
public static Object getValue(Object tree, Map context, Object root, Class resultType) throws OgnlException {
OgnlContext ognlContext = (OgnlContext)addDefaultContext(root, context);
Node node = (Node)tree;
Object result;
if (node.getAccessor() != null) {
result = node.getAccessor().get(ognlContext, root);
} else {
result = node.getValue(ognlContext, root);
}
if (resultType != null) {
result = getTypeConverter(context).convertValue(context, root, (Member)null, (String)null, result, resultType);
}
return result;
}
public static Object getValue(ExpressionAccessor expression, OgnlContext context, Object root) {
return expression.get(context, root);
}
public static Object getValue(ExpressionAccessor expression, OgnlContext context, Object root, Class resultType) {
return getTypeConverter(context).convertValue(context, root, (Member)null, (String)null, expression.get(context, root), resultType);
}
public static Object getValue(String expression, Map context, Object root) throws OgnlException {
return getValue((String)expression, (Map)context, root, (Class)null);
}
public static Object getValue(String expression, Map context, Object root, Class resultType) throws OgnlException {
return getValue(parseExpression(expression), context, root, resultType);
}
public static Object getValue(Object tree, Object root) throws OgnlException {
return getValue((Object)tree, (Object)root, (Class)null);
}
public static Object getValue(Object tree, Object root, Class resultType) throws OgnlException {
return getValue(tree, createDefaultContext(root), root, resultType);
}
public static Object getValue(String expression, Object root) throws OgnlException {
return getValue((String)expression, (Object)root, (Class)null);
}
public static Object getValue(String expression, Object root, Class resultType) throws OgnlException {
return getValue(parseExpression(expression), root, resultType);
}
```
那么接下来逐步了解OGNL三要素是什么。
### expression
表达式是OGNL的核心OGNL的内容都是从表达式出发的表达式规定了程序在解析后需要做什么操作。
### root
root对象在OGNL中可以看作是一个节点当表达式被解析后对谁进行操作这其中的“谁”就是root。
### context
context上下文是一个Map类型的数据结构包含OGNL表达式执行时候的上下文root也在其中
为了更好地了解我在demo中提供了一个setinfo接口可以方便观察root和非root节点在使用OGNL表达式时候的差异。
OGNL的使用
-------
### 基本使用
我在demo中留了一个例子来测试OGNL表达式那么接下来就来熟悉一下OGNL表达式的用法吧
OGNL的语法和Java很像基本上熟悉了Java就能简单使用OGNL表达式。
OGNL中可以使用的操作符+, -, \*, /, ++, --, ==, !=, =mod, in, not in
对于非`root`自定义对象,我们可以通过`.`来链接就和Java中一样比如我要访问`people1`中的`car`的`carName`字段,只需要使用`#people1.car.carName`就可以访问了。
对于`root`对象也一样,只不过因为`root`只有一个,所以不需要加`root`的名字就可以。
### 引用静态资源
要引用类的静态方法和字段,他们的表达方式是一样的`@class@member`或者`@class@method(args)`。
比如我们最喜欢的弹计算器操作,其实就可以写成`@java.lang.Runtime@getRuntime().exec("calc")`。
通过`@java.lang.Runtime`访问`Runtime`类,然后通过`@getRuntime().exec("calc")`拿到`Runtime`实例并执行命令。
![image.png](https://shs3.b.qianxin.com/attack_forum/2024/04/attach-4e97a96925fcb6543ab0b4f4b1ecd7934e9e625b.png)
### 数组、Map、容器
OGNL支持对数组、Map、容器进行操作如我在`People`类中新增了一个数组:
```java
public static class People {
public Car car = new Car();
public String[] stratt = new String[]{"1", "2", "3"};
}
```
同样的对于前文的`root`和`非root`节点,只需要分别用以下表达式去访问
rootstratt\[1\]读取
![image.png](https://shs3.b.qianxin.com/attack_forum/2024/04/attach-0876c0b065be2fa03215ad7d31d7fce833df5a87.png)
非root#people1.stratt\[2\]
![image.png](https://shs3.b.qianxin.com/attack_forum/2024/04/attach-c489b57d66aa8a5116c5cf3b099989df3df7fa9b.png)
也可以用Java中类似的方式新建数组并且访问
`new java.lang.String[]{"a","b","c"}[1]`
`new String[]{"a","b","c"}[2]`
这两种方法都是可以的。
![image.png](https://shs3.b.qianxin.com/attack_forum/2024/04/attach-98fc05f174b602369d1e97765ab5d7082deea6a2.png)
Map同理可以使用key的值来直接访问value
`#{"A":"a","B":"b","C":"c"}["B"]`
![image.png](https://shs3.b.qianxin.com/attack_forum/2024/04/attach-17d9d02c5207b3f78b07292f1a055e5ec7da0ce9.png)
### 选择和投影
这段就作为一个了解吧我直接复制了milktea师傅的文章中的一部分
OGNL支持类似数据库中的投影projection 和选择selection
投影就是选出集合中每个元素的相同属性组成新的集合,类似于关系数据库的字段操作。投影操作语法为 collection.{XXX}其中XXX是这个集合中每个元素的公共属性。
例如:`group.userList.{username}`将获得某个`group`中的所有`user`的`name`的列表。
选择就是过滤满足`selection`条件的集合元素,类似于关系数据库的纪录操作。选择操作的语法为:`collection.{X YYY}`,其中`X`是一个选择操作符,后面则是选择用的逻辑表达式。而选择操作符有三种:
- ?选择满足条件的所有元素
- ^选择满足条件的第一个元素
- $选择满足条件的最后一个元素
例如:`group.userList.{? #txxx.xxx != null}`将获得某个`group`中`user`的`name`不为空的`user`的列表。
Invocation.class
----------------
那天在公司审代码发现mybatis中频繁使用了`setAccessable()`,跟进去发现了这么一个类,正好也是闲的,就用它写了个弹计算器,心想也没啥用。
![image.png](https://shs3.b.qianxin.com/attack_forum/2024/04/attach-6784196504a8a3c27a38559241487203d4f58b5e.png)
![image.png](https://shs3.b.qianxin.com/attack_forum/2024/04/attach-155c805cbae2d86348bbe2f003633c1252d7b6ce.png)
配合ognl
------
后来闲的没事突然想这东西能不能写进ognl里面然后绕过反射的限制呢好巧不巧第二天Umbrella让我助他SSTI这个trick就有用了么这不。
在ognl中通过forname()可以判断有没有这个类也就是判断用没用mybatis。
图不放了,遇到有可能的环境大家自行查看就好。
有的那么就尝试构造个表达式获取一下runtime实例绕过防护试试。
![image.png](https://shs3.b.qianxin.com/attack_forum/2024/04/attach-ebec054ae40074195d37fdc2dd44757bfe8a622b.png)
线上同样不放图了被警告了使用的话是可以成功的那就试试直接r呢
![image.png](https://shs3.b.qianxin.com/attack_forum/2024/04/attach-e3694177f78cdc0ce084a3538db14e8a57d45a17.png)
具体描述和测试用例放在https://github.com/springkill/Ognl-Test 感兴趣自取。</textarea>
<div id=layer-photos-demo>
<div id=md_view><div class=markdown-body><h1 blockindex=0>OGNL小trick</h1>
<h2 blockindex=1>什么是OGNL表达式</h2>
<p blockindex=2>OGNL 是 Object-Graph Navigation Language对象图导航语言OGNL 最初是作为 WebWork 框架的一部分开发的,现在已成为 Apache Struts2 的一个关键组件这也就是为什么Struts2中的OGNL表达式漏洞这么多并被用于其他各种 Java 框架。</p>
<h2 blockindex=3>对象图导航</h2>
<p blockindex=4>OGNL的核心就在于<code>对象图导航</code>这个概念其实和数据结构中的图和很相似再OGNL或者面向对象编程语言中以引用作为边对象作为节点对象可以包含其他对象组合或与其他对象产生关联聚合从而在更大的对象图中形成子图。<br>
以一段简单的代码作为解释:</p>
<pre blockindex=5><code class="hljs language-java"><span class=hljs-keyword>public</span> <span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>ObjectGraphDemo</span> </span>{
<span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>People</span></span>{
Car car;
}
<span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>Car</span></span>{
String carName = <span class=hljs-string>"BMW"</span>;
House house;
}
<span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>House</span></span>{
String houseName = <span class=hljs-string>"MyHouse"</span>;
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>void</span> <span class=hljs-title>printPeopleInfo</span><span class=hljs-params>()</span></span>{
People people = <span class=hljs-keyword>new</span> People();
System.out.println(people.car.carName);
System.out.println(people.car.house.houseName);
}
}
</code></pre>
<p blockindex=6>我们设定好了三个类,分别叫<code>People</code><code>Car</code>还有<code>House</code>,通过这三个类的引用,我们就可以体会到以引用作为边,对象作为节点的这么一种设计理念。</p>
<h2 blockindex=7>OGNL中都有什么</h2>
<p blockindex=8>想知道OGNL都有什么不如直接点进去看看我们都知道再<code>ognl.Ognl.getValue()</code>方法处会触发RCE漏洞那么也就是看在这里<code>getValue</code>方法接受了什么参数其实更简单的办法是去问问GPT<br>
这里我将所有的<code>getValue</code>全都拷贝过来了,看起来很多其实全都是重载方法进行互相调用,通过看这些代码发现必不可少的三样东西是<code>expression</code><code>context</code><code>root</code>这也就是我们说的OGNL的三要素其实这些东西网上很多人都分析过了之所以这么啰嗦还是为了自己能够更好地理解。</p>
<pre blockindex=9><code class="hljs language-java"><span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(Object tree, Map context, Object root)</span> <span class=hljs-keyword>throws</span> OgnlException </span>{
<span class=hljs-keyword>return</span> getValue((Object)tree, (Map)context, root, (Class)<span class=hljs-keyword>null</span>);
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(Object tree, Map context, Object root, Class resultType)</span> <span class=hljs-keyword>throws</span> OgnlException </span>{
OgnlContext ognlContext = (OgnlContext)addDefaultContext(root, context);
Node node = (Node)tree;
Object result;
<span class=hljs-keyword>if</span> (node.getAccessor() != <span class=hljs-keyword>null</span>) {
result = node.getAccessor().get(ognlContext, root);
} <span class=hljs-keyword>else</span> {
result = node.getValue(ognlContext, root);
}
<span class=hljs-keyword>if</span> (resultType != <span class=hljs-keyword>null</span>) {
result = getTypeConverter(context).convertValue(context, root, (Member)<span class=hljs-keyword>null</span>, (String)<span class=hljs-keyword>null</span>, result, resultType);
}
<span class=hljs-keyword>return</span> result;
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(ExpressionAccessor expression, OgnlContext context, Object root)</span> </span>{
<span class=hljs-keyword>return</span> expression.get(context, root);
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(ExpressionAccessor expression, OgnlContext context, Object root, Class resultType)</span> </span>{
<span class=hljs-keyword>return</span> getTypeConverter(context).convertValue(context, root, (Member)<span class=hljs-keyword>null</span>, (String)<span class=hljs-keyword>null</span>, expression.get(context, root), resultType);
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(String expression, Map context, Object root)</span> <span class=hljs-keyword>throws</span> OgnlException </span>{
<span class=hljs-keyword>return</span> getValue((String)expression, (Map)context, root, (Class)<span class=hljs-keyword>null</span>);
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(String expression, Map context, Object root, Class resultType)</span> <span class=hljs-keyword>throws</span> OgnlException </span>{
<span class=hljs-keyword>return</span> getValue(parseExpression(expression), context, root, resultType);
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(Object tree, Object root)</span> <span class=hljs-keyword>throws</span> OgnlException </span>{
<span class=hljs-keyword>return</span> getValue((Object)tree, (Object)root, (Class)<span class=hljs-keyword>null</span>);
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(Object tree, Object root, Class resultType)</span> <span class=hljs-keyword>throws</span> OgnlException </span>{
<span class=hljs-keyword>return</span> getValue(tree, createDefaultContext(root), root, resultType);
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(String expression, Object root)</span> <span class=hljs-keyword>throws</span> OgnlException </span>{
<span class=hljs-keyword>return</span> getValue((String)expression, (Object)root, (Class)<span class=hljs-keyword>null</span>);
}
<span class=hljs-function><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> Object <span class=hljs-title>getValue</span><span class=hljs-params>(String expression, Object root, Class resultType)</span> <span class=hljs-keyword>throws</span> OgnlException </span>{
<span class=hljs-keyword>return</span> getValue(parseExpression(expression), root, resultType);
}
</code></pre>
<p blockindex=10>那么接下来逐步了解OGNL三要素是什么。</p>
<h3 blockindex=11>expression</h3>
<p blockindex=12>表达式是OGNL的核心OGNL的内容都是从表达式出发的表达式规定了程序在解析后需要做什么操作。</p>
<h3 blockindex=13>root</h3>
<p blockindex=14>root对象在OGNL中可以看作是一个节点当表达式被解析后对谁进行操作这其中的“谁”就是root。</p>
<h3 blockindex=15>context</h3>
<p blockindex=16>context上下文是一个Map类型的数据结构包含OGNL表达式执行时候的上下文root也在其中<br>
为了更好地了解我在demo中提供了一个setinfo接口可以方便观察root和非root节点在使用OGNL表达式时候的差异。</p>
<h2 blockindex=17>OGNL的使用</h2>
<h3 blockindex=18>基本使用</h3>
<p blockindex=19>我在demo中留了一个例子来测试OGNL表达式那么接下来就来熟悉一下OGNL表达式的用法吧<br>
OGNL的语法和Java很像基本上熟悉了Java就能简单使用OGNL表达式。<br>
OGNL中可以使用的操作符+, -, *, /, ++, --, ==, !=, =mod, in, not in<br>
对于非<code>root</code>自定义对象,我们可以通过<code>.</code>来链接就和Java中一样比如我要访问<code>people1</code>中的<code>car</code><code>carName</code>字段,只需要使用<code>#people1.car.carName</code>就可以访问了。<br>
对于<code>root</code>对象也一样,只不过因为<code>root</code>只有一个,所以不需要加<code>root</code>的名字就可以。</p>
<h3 blockindex=20>引用静态资源</h3>
<p blockindex=21>要引用类的静态方法和字段,他们的表达方式是一样的<code>@class@member</code>或者<code>@class@method(args)</code><br>
比如我们最喜欢的弹计算器操作,其实就可以写成<code>@java.lang.Runtime@getRuntime().exec("calc")</code><br>
通过<code>@java.lang.Runtime</code>访问<code>Runtime</code>类,然后通过<code>@getRuntime().exec("calc")</code>拿到<code>Runtime</code>实例并执行命令。<br>
<img src="
<h3 blockindex=22>数组、Map、容器</h3>
<p blockindex=23>OGNL支持对数组、Map、容器进行操作如我在<code>People</code>类中新增了一个数组:</p>
<pre blockindex=24><code class="hljs language-java"><span class=hljs-keyword>public</span> <span class=hljs-keyword>static</span> <span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>People</span> </span>{
<span class=hljs-keyword>public</span> Car car = <span class=hljs-keyword>new</span> Car();
<span class=hljs-keyword>public</span> String[] stratt = <span class=hljs-keyword>new</span> String[]{<span class=hljs-string>"1"</span>, <span class=hljs-string>"2"</span>, <span class=hljs-string>"3"</span>};
}
</code></pre>
<p blockindex=25>同样的对于前文的<code>root</code><code>非root</code>节点,只需要分别用以下表达式去访问<br>
rootstratt[1]读取<br>
<img src="
非root#people1.stratt[2]<br>
<img src="
也可以用Java中类似的方式新建数组并且访问<br>
<code>new java.lang.String[]{"a","b","c"}[1]</code><br>
<code>new String[]{"a","b","c"}[2]</code><br>
这两种方法都是可以的。<br>
<img src=
Map同理可以使用key的值来直接访问value<br>
<code>#{"A":"a","B":"b","C":"c"}["B"]</code><br>
<img src=
<h3 blockindex=26>选择和投影</h3>
<p blockindex=27>这段就作为一个了解吧我直接复制了milktea师傅的文章中的一部分<br>
OGNL支持类似数据库中的投影projection 和选择selection<br>
投影就是选出集合中每个元素的相同属性组成新的集合,类似于关系数据库的字段操作。投影操作语法为 collection.{XXX}其中XXX是这个集合中每个元素的公共属性。<br>
例如:<code>group.userList.{username}</code>将获得某个<code>group</code>中的所有<code>user</code><code>name</code>的列表。<br>
选择就是过滤满足<code>selection</code>条件的集合元素,类似于关系数据库的纪录操作。选择操作的语法为:<code>collection.{X YYY}</code>,其中<code>X</code>是一个选择操作符,后面则是选择用的逻辑表达式。而选择操作符有三种:</p>
<ul blockindex=28>
<li>?选择满足条件的所有元素</li>
<li>^选择满足条件的第一个元素</li>
<li>$选择满足条件的最后一个元素</li>
</ul>
<p blockindex=29>例如:<code>group.userList.{? #txxx.xxx != null}</code>将获得某个<code>group</code><code>user</code><code>name</code>不为空的<code>user</code>的列表。</p>
<h2 blockindex=30>Invocation.class</h2>
<p blockindex=31>那天在公司审代码发现mybatis中频繁使用了<code>setAccessable()</code>,跟进去发现了这么一个类,正好也是闲的,就用它写了个弹计算器,心想也没啥用。</p>
<p blockindex=32><img src="
<p blockindex=33><img src="
<h2 blockindex=34>配合ognl</h2>
<p blockindex=35>后来闲的没事突然想这东西能不能写进ognl里面然后绕过反射的限制呢好巧不巧第二天Umbrella让我助他SSTI这个trick就有用了么这不。</p>
<p blockindex=36>在ognl中通过forname()可以判断有没有这个类也就是判断用没用mybatis。<br>
图不放了,遇到有可能的环境大家自行查看就好。</p>
<p blockindex=37>有的那么就尝试构造个表达式获取一下runtime实例绕过防护试试。</p>
<p blockindex=38><img src="
<p blockindex=39>线上同样不放图了被警告了使用的话是可以成功的那就试试直接r呢</p>
<p blockindex=40><img src="
<p blockindex=41>具体描述和测试用例放在<a href=https://github.com/springkill/Ognl-Test>https://github.com/springkill/Ognl-Test</a> 感兴趣自取。</p></div></div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>