blog/2024/10/20/jar-via-adb/index.html
2025-05-13 14:44:50 +00:00

405 lines
50 KiB
HTML
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<!DOCTYPE html><html lang="zh-CN" data-theme="dark"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0,viewport-fit=cover"><title>用 adb + app_process 执行 Java 代码 —— 一种无需安装 apk 的脱机代码执行方案 | 時痕</title><meta name="author" content="Linloir"><meta name="copyright" content="Linloir"><meta name="format-detection" content="telephone=no"><meta name="theme-color" content="#0d0d0d"><meta name="description" content="方案速览本方案本质上是使用了安卓提供的 app_process 命令,在将 Java 代码正确地打包为需要的 .jar 或是 .dex 文件后,通过 app_process 启动对应的入口函数来实现 adb 执行 Java 代码的能力。 对于目标 .jar 或是 .dex 文件,有两种不同的编译方案: .dex 文件方式: 创建一个普通的 Java 工程,正常地添加依赖,这里需要正确配置确保依赖">
<meta property="og:type" content="article">
<meta property="og:title" content="用 adb + app_process 执行 Java 代码 —— 一种无需安装 apk 的脱机代码执行方案">
<meta property="og:url" content="https://blog.linloir.cn/2024/10/20/jar-via-adb/">
<meta property="og:site_name" content="時痕">
<meta property="og:description" content="方案速览本方案本质上是使用了安卓提供的 app_process 命令,在将 Java 代码正确地打包为需要的 .jar 或是 .dex 文件后,通过 app_process 启动对应的入口函数来实现 adb 执行 Java 代码的能力。 对于目标 .jar 或是 .dex 文件,有两种不同的编译方案: .dex 文件方式: 创建一个普通的 Java 工程,正常地添加依赖,这里需要正确配置确保依赖">
<meta property="og:locale" content="zh_CN">
<meta property="og:image" content="https://blog.linloir.cn/img/cover.jpg">
<meta property="article:published_time" content="2024-10-20T13:22:40.000Z">
<meta property="article:modified_time" content="2025-05-13T14:43:59.801Z">
<meta property="article:author" content="Linloir">
<meta property="article:tag" content="工作">
<meta property="article:tag" content="安卓">
<meta property="article:tag" content="综合">
<meta name="twitter:card" content="summary">
<meta name="twitter:image" content="https://blog.linloir.cn/img/cover.jpg"><link rel="shortcut icon" href="/img/avatar.png"><link rel="canonical" href="https://blog.linloir.cn/2024/10/20/jar-via-adb/"><link rel="preconnect" href="//cdn.jsdelivr.net"/><link rel="preconnect" href="//busuanzi.ibruce.info"/><link rel="stylesheet" href="/css/index.css"><link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free/css/all.min.css"><link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/node-snackbar/dist/snackbar.min.css" media="print" onload="this.media='all'"><link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fancyapps/ui/dist/fancybox/fancybox.min.css" media="print" onload="this.media='all'"><script>
(() => {
const saveToLocal = {
set: (key, value, ttl) => {
if (!ttl) return
const expiry = Date.now() + ttl * 86400000
localStorage.setItem(key, JSON.stringify({ value, expiry }))
},
get: key => {
const itemStr = localStorage.getItem(key)
if (!itemStr) return undefined
const { value, expiry } = JSON.parse(itemStr)
if (Date.now() > expiry) {
localStorage.removeItem(key)
return undefined
}
return value
}
}
window.btf = {
saveToLocal,
getScript: (url, attr = {}) => new Promise((resolve, reject) => {
const script = document.createElement('script')
script.src = url
script.async = true
Object.entries(attr).forEach(([key, val]) => script.setAttribute(key, val))
script.onload = script.onreadystatechange = () => {
if (!script.readyState || /loaded|complete/.test(script.readyState)) resolve()
}
script.onerror = reject
document.head.appendChild(script)
}),
getCSS: (url, id) => new Promise((resolve, reject) => {
const link = document.createElement('link')
link.rel = 'stylesheet'
link.href = url
if (id) link.id = id
link.onload = link.onreadystatechange = () => {
if (!link.readyState || /loaded|complete/.test(link.readyState)) resolve()
}
link.onerror = reject
document.head.appendChild(link)
}),
addGlobalFn: (key, fn, name = false, parent = window) => {
if (!true && key.startsWith('pjax')) return
const globalFn = parent.globalFn || {}
globalFn[key] = globalFn[key] || {}
if (name && globalFn[key][name]) return
globalFn[key][name || Object.keys(globalFn[key]).length] = fn
parent.globalFn = globalFn
}
}
const activateDarkMode = () => {
document.documentElement.setAttribute('data-theme', 'dark')
if (document.querySelector('meta[name="theme-color"]') !== null) {
document.querySelector('meta[name="theme-color"]').setAttribute('content', 'undefined')
}
}
const activateLightMode = () => {
document.documentElement.setAttribute('data-theme', 'light')
if (document.querySelector('meta[name="theme-color"]') !== null) {
document.querySelector('meta[name="theme-color"]').setAttribute('content', 'undefined')
}
}
btf.activateDarkMode = activateDarkMode
btf.activateLightMode = activateLightMode
const theme = saveToLocal.get('theme')
const mediaQueryDark = window.matchMedia('(prefers-color-scheme: dark)')
const mediaQueryLight = window.matchMedia('(prefers-color-scheme: light)')
if (theme === undefined) {
if (mediaQueryLight.matches) activateLightMode()
else if (mediaQueryDark.matches) activateDarkMode()
else {
const hour = new Date().getHours()
const isNight = hour <= 6 || hour >= 18
isNight ? activateDarkMode() : activateLightMode()
}
mediaQueryDark.addEventListener('change', () => {
if (saveToLocal.get('theme') === undefined) {
e.matches ? activateDarkMode() : activateLightMode()
}
})
} else {
theme === 'light' ? activateLightMode() : activateDarkMode()
}
const asideStatus = saveToLocal.get('aside-status')
if (asideStatus !== undefined) {
document.documentElement.classList.toggle('hide-aside', asideStatus === 'hide')
}
const detectApple = () => {
if (/iPad|iPhone|iPod|Macintosh/.test(navigator.userAgent)) {
document.documentElement.classList.add('apple')
}
}
detectApple()
})()
</script><script>const GLOBAL_CONFIG = {
root: '/',
algolia: undefined,
localSearch: undefined,
translate: {"defaultEncoding":2,"translateDelay":0,"msgToTraditionalChinese":"繁","msgToSimplifiedChinese":"简"},
noticeOutdate: undefined,
highlight: {"plugin":"highlight.js","highlightCopy":true,"highlightLang":true,"highlightHeightLimit":false,"highlightFullpage":false,"highlightMacStyle":false},
copy: {
success: '复制成功',
error: '复制失败',
noSupport: '浏览器不支持'
},
relativeDate: {
homepage: false,
post: false
},
runtime: '',
dateSuffix: {
just: '刚刚',
min: '分钟前',
hour: '小时前',
day: '天前',
month: '个月前'
},
copyright: undefined,
lightbox: 'fancybox',
Snackbar: {"chs_to_cht":"已切换为繁体中文","cht_to_chs":"已切换为简体中文","day_to_night":"已切换为深色模式","night_to_day":"已切换为浅色模式","bgLight":"#49b1f5","bgDark":"#1f1f1f","position":"top-center"},
infinitegrid: {
js: 'https://cdn.jsdelivr.net/npm/@egjs/infinitegrid/dist/infinitegrid.min.js',
buttonText: '加载更多'
},
isPhotoFigcaption: false,
islazyload: false,
isAnchor: true,
percent: {
toc: false,
rightside: false,
},
autoDarkmode: true
}</script><script id="config-diff">var GLOBAL_CONFIG_SITE = {
title: '用 adb + app_process 执行 Java 代码 —— 一种无需安装 apk 的脱机代码执行方案',
isPost: true,
isHome: false,
isHighlightShrink: false,
isToc: true,
postUpdate: '2025-05-13 22:43:59'
}</script><meta name="generator" content="Hexo 7.3.0"></head><body><script>window.paceOptions = {
restartOnPushState: false
}
btf.addGlobalFn('pjaxSend', () => {
Pace.restart()
}, 'pace_restart')
</script><link rel="stylesheet" href="/css/minimal.css"/><script src="https://cdn.jsdelivr.net/npm/pace-js/pace.min.js"></script><div id="sidebar"><div id="menu-mask"></div><div id="sidebar-menus"><div class="avatar-img is-center"><img src="/img/avatar.png" onerror="onerror=null;src='/img/friend_404.gif'" alt="avatar"/></div><div class="site-data is-center"><a href="/archives/"><div class="headline">文章</div><div class="length-num">21</div></a><a href="/tags/"><div class="headline">标签</div><div class="length-num">20</div></a><a href="/categories/"><div class="headline">分类</div><div class="length-num">2</div></a></div><div class="menus_items"><div class="menus_item"><a class="site-page" href="/"><i class="fa-fw fas fa-home"></i><span> 主页</span></a></div><div class="menus_item"><a class="site-page" href="/tags/"><i class="fa-fw fas fa-tags"></i><span> 标签</span></a></div><div class="menus_item"><a class="site-page" href="/categories/"><i class="fa-fw fas fa-th"></i><span> 分类</span></a></div><div class="menus_item"><a class="site-page" href="/archives/"><i class="fa-fw fas fa-archive"></i><span> 归档</span></a></div></div></div></div><div class="post" id="body-wrap"><header class="post-bg" id="page-header" style="background-image: url(/img/cover.jpg);"><nav id="nav"><span id="blog-info"><a class="nav-site-title" href="/"><span class="site-name">時痕</span></a><a class="nav-page-title" href="/"><span class="site-name">用 adb + app_process 执行 Java 代码 —— 一种无需安装 apk 的脱机代码执行方案</span></a></span><div id="menus"><div class="menus_items"><div class="menus_item"><a class="site-page" href="/"><i class="fa-fw fas fa-home"></i><span> 主页</span></a></div><div class="menus_item"><a class="site-page" href="/tags/"><i class="fa-fw fas fa-tags"></i><span> 标签</span></a></div><div class="menus_item"><a class="site-page" href="/categories/"><i class="fa-fw fas fa-th"></i><span> 分类</span></a></div><div class="menus_item"><a class="site-page" href="/archives/"><i class="fa-fw fas fa-archive"></i><span> 归档</span></a></div></div><div id="toggle-menu"><span class="site-page"><i class="fas fa-bars fa-fw"></i></span></div></div></nav><div id="post-info"><h1 class="post-title">用 adb + app_process 执行 Java 代码 —— 一种无需安装 apk 的脱机代码执行方案</h1><div id="post-meta"><div class="meta-firstline"><span class="post-meta-date"><i class="far fa-calendar-alt fa-fw post-meta-icon"></i><span class="post-meta-label">发表于</span><time class="post-meta-date-created" datetime="2024-10-20T13:22:40.000Z" title="发表于 2024-10-20 21:22:40">2024-10-20</time><span class="post-meta-separator">|</span><i class="fas fa-history fa-fw post-meta-icon"></i><span class="post-meta-label">更新于</span><time class="post-meta-date-updated" datetime="2025-05-13T14:43:59.801Z" title="更新于 2025-05-13 22:43:59">2025-05-13</time></span><span class="post-meta-categories"><span class="post-meta-separator">|</span><i class="fas fa-inbox fa-fw post-meta-icon"></i><a class="post-meta-categories" href="/categories/%E6%8A%80%E6%9C%AF/">技术</a></span></div><div class="meta-secondline"><span class="post-meta-separator">|</span><span class="post-meta-wordcount"><i class="far fa-file-word fa-fw post-meta-icon"></i><span class="post-meta-label">总字数:</span><span class="word-count">4.1k</span><span class="post-meta-separator">|</span><i class="far fa-clock fa-fw post-meta-icon"></i><span class="post-meta-label">阅读时长:</span><span>14分钟</span></span><span class="post-meta-separator">|</span><span class="post-meta-pv-cv" id="" data-flag-title=""><i class="far fa-eye fa-fw post-meta-icon"></i><span class="post-meta-label">浏览量:</span><span id="busuanzi_value_page_pv"><i class="fa-solid fa-spinner fa-spin"></i></span></span></div></div></div></header><main class="layout" id="content-inner"><div id="post"><article class="post-content" id="article-container"><h2 id="方案速览"><a href="#方案速览" class="headerlink" title="方案速览"></a>方案速览</h2><p>本方案本质上是使用了安卓提供的 <code>app_process</code> 命令,在将 Java 代码正确地打包为需要的 <code>.jar</code> 或是 <code>.dex</code> 文件后,通过 <code>app_process</code> 启动对应的入口函数来实现 <code>adb</code> 执行 Java 代码的能力。</p>
<p>对于目标 <code>.jar</code> 或是 <code>.dex</code> 文件,有两种不同的编译方案:</p>
<ol>
<li><code>.dex</code> 文件方式:<ol>
<li>创建一个普通的 Java 工程,正常地添加依赖,<strong>这里需要正确配置确保依赖在构建的时候也会被打包进制品 jar 中</strong>,编写相关代码</li>
<li>构建 jar 包</li>
<li>在安卓 SDK 文件夹下 (一般为 <code>&lt;homedir&gt;\AppData\Local\Android\Sdk</code>),找到 <code>cmdline-tools\latest\bin\d8.bat</code> (这里可以将 <code>&lt;homedir&gt;\AppData\Local\Android\Sdkcmdline-tools\latest\bin</code> 添加到环境变量中方便后面调用 <code>d8</code> 命令),如果没有,可以在 Android Studio 更新 Commandline Tools 或是在 <code>build-tools\&lt;version&gt;</code> 下找到一个能用的</li>
<li>使用 <code>d8 [options] &lt;source jar&gt;</code> 命令编译出 <code>.dex</code> 文件,例如 <code>d8 server.jar</code>,会在当前工作目录下生成 <code>classes.dex</code></li>
</ol>
</li>
<li><code>.jar</code> 文件方式:<ol>
<li>创建一个 Empty Activity 的 Android Studio 安卓工程,正常编写 Java 代码 (在 <code>app/src/main/java</code> 下) 以及添加依赖</li>
<li>使用 Android Studio 的构建 apk 能力打包一个 apk (例如 <code>app-debug.apk</code>)</li>
<li><code>.apk</code> 后缀改为 <code>.jar</code></li>
</ol>
</li>
</ol>
<p>得到 <code>.jar</code> 或是 <code>.dex</code> 文件以后,使用 <code>adb [-s serial] push &lt;source file&gt; &lt;target&gt;</code> 命令推送到手机端</p>
<p>最后,在 <code>shell</code> 环境执行 <code>app_process</code> call 起 Java 程序:</p>
<p><code>app_process</code> 启动命令为 <code>app_process -Djava.class.path=&lt;path to jar&gt; [other java options] &lt;working directory (preferred /)&gt; com.your.package.Class [args] ...</code>,也即对于打包好的 <code>server.jar</code> 文件,执行命令可能为 <code>adb shell app_process -Djava.class.path=/data/local/tmp/server.jar / com.linloir.server.Main --port 23456</code></p>
<hr>
<h2 id="背景"><a href="#背景" class="headerlink" title="背景"></a>背景</h2><p>起因是想要做一个检查安卓设备网络连通性的能力,最初想的就是 <code>adb ping</code> 直接秒了,甚至还可以复杂点用 <code>adb ip route get 1.1.1.1</code> 拿到网关以后分别去 ping 网关和外网地址,从而判断设备到网关的连通性和网关到外网的连通性</p>
<p>但是很快遇到了一个很诡异的现象:某些设备连续 ping 同一个地址 (即 A 先 ping 网关,然后 B ping 网关,最后 C ping 网关,串行执行) 时,会有一些设备出现 80% 以上的丢包甚至直接完全无法 ping 通的情况,并且当某台设备 ping 不通的同时,再开一个 shell 让它 ping 另外一个地址 (例如外网地址),又是完全正常的,并且同一时间之前卡住的还是完全卡住的状态。反复研究发现有那么一些手机互相就像有羁绊一样,只要挨个 ping 相同地址就基本后一台铁挂,由于时间有限同时能力也有限,暂时没有深究了 (如果有后续就再补一篇文章)</p>
<p>于是,就因为这个 ping 这样谎报军情的现象,导致最后实在没办法再采用原本的方案,被迫不得不去找一个新的、能够兼容所有主流安卓设备的方案</p>
<p>思路很快确定了,就是做一个 HTTP 服务,给个 &#x2F;ping 接口,客户端这边去发 GET 请求然后确认回包 200 即可。但是,问题就在于,如何能够在所有主流安卓设备上发 HTTP 请求然后判断结果呢?</p>
<p>最初,我想到了 <code>curl</code>,然后发现不是所有的设备都有 <code>curl</code>;于是转而投奔 <code>nc</code>,结果发现 <code>nc</code> 也不是所有的设备都有;其他的命令其实基本就也不想再试了,因为就算当前手头上的设备都能支持,谁也说不好会不会新来的机器就不支持了…</p>
<p>这时候都感觉仿佛只能用最最丑陋的下策了 —— 装一个 apk 包然后用 adb 拉起来然后走 socket 通信去执行 HTTP 请求。但是这样的方案在自动化工程里面几乎是不可接受的比如光是自动化安装包这个环节就有各种坑可以踩就算手动装也有不小的工作量同时call 起 apk 的过程也可能影响前台的 app。总之这个方案一定是利大于弊的因此也就没有第一时间去做尝试而是继续去寻找可行的方案</p>
<p>经过同事点拨,突然想到,平常自动化测试用到的 <code>openatx/uiautomator2</code> 这个包,它本质上就是个 python 包装的反向代理,使得电脑端通过 python API 和手机侧的 server 通信,再把请求转换成 uiautomator 的指令加以执行。那么既然 <code>openatx/uiautomator2</code> 是这样的 server-client 架构,那它肯定在手机上也跑了个 server 服务吧,这个服务之前有注意到就是一个 <code>u2.jar</code> 文件,所以应该意味着,我应该也可以写一个 Java 程序然后想办法在安卓端跑起来</p>
<p>然后就开始了后文中试图让 Java 能在 adb shell 中跑起来的各种尝试…</p>
<h2 id="尝试"><a href="#尝试" class="headerlink" title="尝试"></a>尝试</h2><h3 id="通过-dalvikvm-执行-dex-文件"><a href="#通过-dalvikvm-执行-dex-文件" class="headerlink" title="通过 dalvikvm 执行 .dex 文件"></a>通过 dalvikvm 执行 .dex 文件</h3><p>最初的思路比较直接,安卓可以说是基于 Java 的,早期的运行时虚拟机 dalvikvm 即便到了 ART 时代仍然被系统所兼容,并且 <code>dalvikvm</code> 命令也仍然存在于所有安卓设备上,于是便试图通过 <code>dalvikvm -cp &lt;dex file&gt; com.your.package.Class</code> 来执行代码</p>
<p>在 demo 验证阶段,一个简单的 Hello World 程序是可以运行的,并且所有的设备都能够正确的执行,这基本证明了两点:</p>
<ol>
<li>使用 Java 来编写兼容大部分主流安卓设备的特殊逻辑代码并在设备上执行这一思路是可行的</li>
<li>dalvikvm 确实可以执行 Java 程序</li>
</ol>
<p>但是很快,在 HTTP 请求的代码验证中,部分设备就出现了各种各样的错误:</p>
<ol>
<li>触发 Segmentation fault</li>
<li>触发异常 <code>java.lang.UnsatisfiedLinkError</code> 并且直接无法执行</li>
<li>触发 <code>java.lang.ClassNotFoundException</code> 异常但是不影响执行</li>
</ol>
<p>其中,第二种异常提供了一些有用的信息:</p>
<figure class="highlight text"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br></pre></td><td class="code"><pre><span class="line">java.lang.reflect.InvocationTargetException</span><br><span class="line"> at java.lang.reflect.Method.invoke(Native Method)</span><br><span class="line"> at com.android.okhttp.OkUrlFactory.startBoost(OkUrlFactory.java:127)</span><br><span class="line"> at com.android.okhttp.OkUrlFactory.open(OkUrlFactory.java:79)</span><br><span class="line"> at com.android.okhttp.OkUrlFactory.open(OkUrlFactory.java:69)</span><br><span class="line"> at com.android.okhttp.HttpHandler.openConnection(HttpHandler.java:44)</span><br><span class="line"> at java.net.URL.openConnection(URL.java:993)</span><br><span class="line"> at HelloWorld.sendGetRequest(HelloWorld.java:64)</span><br><span class="line"> at HelloWorld.main(HelloWorld.java:26)</span><br><span class="line">Caused by: java.lang.UnsatisfiedLinkError: No implementation found for android.os.IBinder com.android.internal.os.BinderInternal.getContextObject() (tried Java_com_android_internal_os_BinderInternal_getContextObject and Java_com_android_internal_os_BinderInternal_getContextObject__)</span><br><span class="line"> at com.android.internal.os.BinderInternal.getContextObject(Native Method)</span><br><span class="line"> at android.os.ServiceManager.getIServiceManager(ServiceManager.java:40)</span><br><span class="line"> at android.os.ServiceManager.getService(ServiceManager.java:56)</span><br><span class="line"> at com.oppo.luckymoney.LMManager.init(LMManager.java:209)</span><br><span class="line"> at com.oppo.luckymoney.LMManager.&lt;init&gt;(LMManager.java:197)</span><br><span class="line"> at com.oppo.luckymoney.LMManager.getLMManager(LMManager.java:202)</span><br><span class="line"> ... 8 more</span><br></pre></td></tr></table></figure>
<p>它指出,即便我在上层使用的是 <code>java.net.URL.openConnection</code>,下层应用的实现还是变成了 <code>com.android.okhttp.OkUrlFactory</code> 这种设备相关的代码,这应该就是导致在不同设备上会有不通执行结果的原因</p>
<p>进一步猜测,<code>UnsatisfiedLinkError</code> 指出部分动态链接库没有被正确地加载,考虑到对于一个普通程序而言,在启动时安卓运行时肯定会进行一些初始化操作,其中肯定有一部分链接库会在这个阶段由运行时去完成链接,而 <code>dalvikvm</code> 作为直接与虚拟机交互的指令,很有可能就是缺少了这样一些准备操作导致一些相关的运行时库没有被正确链接上,进而导致了 <code>UnsatisfiedLinkError</code> 甚至是 Segmentation Fault</p>
<p>依据这样的猜测dalvikvm 应当是行不通了,势必要找到一个与安卓运行时挂钩的东西去执行 <code>.dex</code> 才能够避免同样的问题</p>
<h3 id="通过-uiautomator-1-0-执行-jar-文件"><a href="#通过-uiautomator-1-0-执行-jar-文件" class="headerlink" title="通过 uiautomator 1.0 执行 .jar 文件"></a>通过 uiautomator 1.0 执行 .jar 文件</h3><p>目光再次转向 <code>atx/uiautomator</code> 工程,在 readme 中读到了 uiautomator 1.0 时代能够通过 <code>uiautomator &lt;jar&gt; ... [-c class]</code> 的方式来执行 <code>.jar</code> 文件中自动化测试的代码,遂猜测其也是使用这样的方式启动的 <code>u2.jar</code> 文件。(其实这里犯了一个很大的错误让我与真相擦肩而过,进而浪费了许多时间在这一步上,这一点将在 <a href="#%E7%BB%8F%E9%AA%8C%E4%B8%8E%E6%95%99%E8%AE%AD">经验与教训</a> 一节详述)</p>
<p>网上能找到的文档大致描绘了一个这样的步骤:</p>
<ol>
<li>创建一个 Java 项目</li>
<li>添加 <code>android.jar</code><code>uiautomator.jar</code> 依赖</li>
<li>使用 <code>android create uitest-project -n &lt;name&gt; -t &lt;target version that match android.jar and uiautomator.jar&gt; -p &lt;path to project&gt;</code> 创建 xml</li>
<li>使用 <code>ant</code> 构建 jar 包</li>
<li>使用 <code>uiautomator &lt;jar&gt; ... [-c class]</code> 命令启动</li>
</ol>
<p>很遗憾的是,当前 Android SDK 中的 <code>android</code> 命令早已不包含 <code>create uitest-project</code> 这一条指令了,并且官方早就删除了与 uiautomator 1.0 相关的文档。同时现有的各种打包文档也都含糊不清IDE 也还是 Eclipse想要在当前的环境成功打包出来可以说是难上加难了。我尝试下载了旧版的 SDK、ant 以及 Java JDK8 来尝试完成这个工作,总是在各种环节出现不兼容的问题,遂最终也作罢。事后复盘想想估计就算真的成功编译了,估计是否兼容如今的安卓版本也是个大问题,并且要是别人想要复现一次这个过程也要遭受这样的苦难,实在不是一个值得尝试的思路</p>
<h2 id="最终方案"><a href="#最终方案" class="headerlink" title="最终方案"></a>最终方案</h2><p>最终,是在一篇关于 <a target="_blank" rel="noopener" href="https://codezjx.com/posts/scrcpy-source-code-analysis/">scrcpy 源码解读</a> 的博客中找到了一段关键的内容:</p>
<blockquote>
<p>上面我们提到 <code>scrcpy-server.jar</code> 是通过 <code>app_process</code> 运行起来的,那么 <code>app_process</code> 又是什么鬼?<code>app_process</code> 是启动 zygote 和其他 Java 程序的应用程序,它可以让虚拟机从 <code>main()</code> 方法开始执行一个 Java 程序。具体的用法可以参考 <code>app_process</code> 源码中的注释:</p>
<figure class="highlight text"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">&quot;Usage: app_process [java-options] cmd-dir start-class-name [options]\n&quot;</span><br></pre></td></tr></table></figure>
<p>与 APP 进程不同,通过 <code>app_process</code> 启动的进程可以在 root 权限和 shell 权限 (adb 默认) 下启动,也就分别拥有了调用不同 API 的能力。通常情况下 shell 权限启动的 <code>app_process</code> 只能够调用一些能够完成 <code>adb</code> 本身工作的 APIroot 权限启动的 app_process 进程则拥有更多权限,甚至能够调用系统 signature 保护级别的 API 及访问整个文件系统。</p>
<p>实际上不少 <code>adb</code> 命令都是对调用 <code>app_process</code> 进行了一些封装,这里举我们平时常用的 <code>am</code> 指令为例,我们在执行 <code>adb shell am</code> 的时候其实是执行了以下的脚本。通过 <code>app_process</code> 运行 <code>am.jar</code><code>com.android.commands.am.Am</code> 这个类的 <code>main()</code> 函数来完成具体操作。</p>
<figure class="highlight bash"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br></pre></td><td class="code"><pre><span class="line"><span class="meta">#!/system/bin/sh</span></span><br><span class="line"><span class="keyword">if</span> [ <span class="string">&quot;<span class="variable">$1</span>&quot;</span> != <span class="string">&quot;instrument&quot;</span> ] ; <span class="keyword">then</span></span><br><span class="line"> cmd activity <span class="string">&quot;<span class="variable">$@</span>&quot;</span></span><br><span class="line"><span class="keyword">else</span></span><br><span class="line"> base=/system</span><br><span class="line"> <span class="built_in">export</span> CLASSPATH=<span class="variable">$base</span>/framework/am.jar</span><br><span class="line"> <span class="built_in">exec</span> app_process <span class="variable">$base</span>/bin com.android.commands.am.Am <span class="string">&quot;<span class="variable">$@</span>&quot;</span></span><br><span class="line"><span class="keyword">fi</span></span><br></pre></td></tr></table></figure>
<p>这里需要注意的是 <code>app_process</code> 只能运行原始 dex 文件,也可以接收包含 <code>classes.dex</code><code>jar</code> 包,如 <code>am.jar</code>。这里 scrcpy 取巧直接用了编译 <code>apk</code> 的方式,再直接重命名为 <code>jar</code> 包,这样也可以被 <code>app_process</code> 运行。而且可以使用 gradle 方便的进行编译,直接在 Android Studio 中开发,调用原生 API像源码中的 <code>IRotationWatcher.aidl</code> 也可以直接生成相应的 <code>IPC Proxy</code> 类进行调用,省力又省心。</p>
<p><code>scrcpy-server.jar</code><code>adb</code> 下通过 <code>app_process</code> 运行起来后,默认就有了 shell 权限,像一些截屏、录屏、模拟按键点击等功能都可以直接使用,而且完全不需要任何权限的声明。</p>
</blockquote>
<p>也就是说,<code>app_process</code> 就是我最开始想要找的,带完整运行时版本的 <code>dalvikvm</code></p>
<p>在引用的文章中,也提到了 scrcpy 使用的一种打包方法:</p>
<ol>
<li>创建一个 Empty Activity 的 Android Studio 安卓工程,正常编写 Java 代码 (在 <code>app/src/main/java</code> 下) 以及添加依赖</li>
<li>使用 Android Studio 的构建 apk 能力打包一个 apk (例如 <code>app-debug.apk</code>)</li>
<li><code>.apk</code> 后缀改为 <code>.jar</code></li>
</ol>
<p>进一步分析,其实本质上就是借用了 Android Studio 将 Java 代码打包为 dex 文件的能力,真正使用的还是 apk 文件中的 <code>classes*.dex</code> 文件,通过以 <code>.zip</code> 打开构建的 apk 并删除其他无关文件再测试可以证实这一点。而由于 dex 本质上还是在给 Java 解释器提供 classpath只是可能不同于 jar 文件的组织方式,因此理论上对于其他方式打包的 dex 文件,<code>app_process</code> 应该也是支持的</p>
<p>这里想到前面用到的 <code>d8.bat</code> 工具,其就是将 <code>.jar</code> 文件转换为 <code>.dex</code> 文件,遂进行测试,证实了另一种打包的思路:</p>
<ol>
<li>创建一个普通的 Java 工程,正常地添加依赖,<strong>这里需要正确配置确保依赖在构建的时候也会被打包进制品 jar 中</strong>,编写相关代码</li>
<li>构建 jar 包</li>
<li>在安卓 SDK 文件夹下 (一般为 <code>&lt;homedir&gt;\AppData\Local\Android\Sdk</code>),找到 <code>cmdline-tools\latest\bin\d8.bat</code> (这里可以将 <code>&lt;homedir&gt;\AppData\Local\Android\Sdkcmdline-tools\latest\bin</code> 添加到环境变量中方便后面调用 <code>d8</code> 命令),如果没有,可以在 Android Studio 更新 Commandline Tools 或是在 <code>build-tools\&lt;version&gt;</code> 下找到一个能用的</li>
<li>使用 <code>d8 [options] &lt;source jar&gt;</code> 命令编译出 <code>.dex</code> 文件,例如 <code>d8 server.jar</code>,会在当前工作目录下生成 <code>classes.dex</code></li>
</ol>
<p>经过测试,其中第 4 点是必须的,原因在引用的文章中也有提到:<em>这里需要注意的是 <code>app_process</code> 只能运行原始 dex 文件,也可以接收包含 <code>classes.dex</code><code>jar</code></em>,也就是说对于 <code>app_process</code><code>.jar</code> 可能只是被作为一个文件夹看待而不是作为实际的 classpath 看待,实际作为 classpath 的是提供的 <code>.dex</code> 或是 <code>.jar</code> 中的 <code>.dex</code></p>
<p>由此可以看出,理论上只要能打包出带完整依赖的 <code>jar</code> 包,辅以 <code>d8</code> 的转换,就可以用来作为 <code>app_process</code> 的 classpath 了,只是使用 Android Studio 方案的明显优势就是对于 Android API 的天然支持和对于依赖管理和导出的便捷性吧,这里可以根据具体的需求和场景来选择</p>
<p>在有了 <code>.jar</code> 或是 <code>.dex</code> 后,参考 <code>app_process</code> 的启动参数:</p>
<figure class="highlight text"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br></pre></td><td class="code"><pre><span class="line">app_process [java-options] cmd-dir [--application | --zygote [--start-system-server]] start-class-name [options]</span><br><span class="line"></span><br><span class="line">app_process executes a interpreted runtime environment for dalvik bytecodes in an application-alike environment</span><br><span class="line"></span><br><span class="line">if together with --application option, System.out is redirected to logcat.</span><br><span class="line">otherwise, System.out remains untouched, i.e., the output is the stdout.</span><br><span class="line"></span><br><span class="line">app_process becomes zygote if starts with --zygote,</span><br><span class="line"></span><br><span class="line">if together with --start-system-server option, zygote also starts system server by forking itself.</span><br><span class="line">app_process can be used as a general program besides starting zygote.</span><br><span class="line"></span><br><span class="line">java-options: Options of dalvik/libart</span><br><span class="line">cmd-dir: Root path of the running process (e.g., root path of file operations)</span><br><span class="line">start-class-name: Class to run</span><br><span class="line">options: Options of the running class</span><br></pre></td></tr></table></figure>
<p>即对于打包好的 <code>server.jar</code> 文件,执行命令可能为 <code>adb shell app_process -Djava.class.path=/data/local/tmp/server.jar / com.linloir.server.Main --port 23456</code></p>
<h2 id="经验与教训"><a href="#经验与教训" class="headerlink" title="经验与教训"></a>经验与教训</h2><ol>
<li>在实现需要部署在多设备上的需求时,优先考虑平台或设备无关的实现方案,例如跨平台语言或是设备统一的解决方案 (如 apk)</li>
<li>验证不一定要在原先的工程里摘代码验证,写 bash 也可以,例如 curl 的验证应当直接写 bash 脚本去在每个 adb devices 获取到的设备上验证,而不是频繁地修改主体工程里与这一块逻辑相关的 go 代码再通过单测函数验证这样明显会浪费更多时间bash 脚本 GPT 很快就能实现然后进行快速验证</li>
<li>在研究 <code>atx/uiautomator2</code> 的实现的时候,先入为主地就认为作者使用了 <code>uiautomator &lt;jar&gt;</code> 这种方式调用,后面注意力都放在了查找作者针对 <code>u2.jar</code> 的打包方式和相关代码上了,以至于在仓库里搜索 <code>u2.jar</code> 的时候竟然没有注意到在搜索结果中 <code>core.py</code> 赫然有着 <code>command = &quot;CLASSPATH=/data/local/tmp/u2.jar app_process / com.wetest.uia2.Main&quot;</code> 这一启动方式,与答案擦肩而过并且进一步在寻找 uiautomator 1.0 的打包方式上浪费了大半天的时间</li>
<li>对于一个陌生的仓库,如果想要了解它的源码实现,不一定非要自己去读,除了进行仓库里的关键词搜索,也可以去查找别人的源码解析,走个捷径</li>
</ol>
</article><div class="post-copyright"><div class="post-copyright__author"><span class="post-copyright-meta"><i class="fas fa-circle-user fa-fw"></i>文章作者: </span><span class="post-copyright-info"><a href="https://blog.linloir.cn">Linloir</a></span></div><div class="post-copyright__type"><span class="post-copyright-meta"><i class="fas fa-square-arrow-up-right fa-fw"></i>文章链接: </span><span class="post-copyright-info"><a href="https://blog.linloir.cn/2024/10/20/jar-via-adb/">https://blog.linloir.cn/2024/10/20/jar-via-adb/</a></span></div><div class="post-copyright__notice"><span class="post-copyright-meta"><i class="fas fa-circle-exclamation fa-fw"></i>版权声明: </span><span class="post-copyright-info">本博客所有文章除特别声明外,均采用 <a href="https://creativecommons.org/licenses/by-nc-sa/4.0/" target="_blank">CC BY-NC-SA 4.0</a> 许可协议。转载请注明来源 <a href="https://blog.linloir.cn" target="_blank">時痕</a></span></div></div><div class="tag_share"><div class="post-meta__tag-list"><a class="post-meta__tags" href="/tags/%E5%B7%A5%E4%BD%9C/">工作</a><a class="post-meta__tags" href="/tags/%E5%AE%89%E5%8D%93/">安卓</a><a class="post-meta__tags" href="/tags/%E7%BB%BC%E5%90%88/">综合</a></div><div class="post-share"><div class="social-share" data-image="/img/cover.jpg" data-sites="facebook,twitter,wechat,weibo,qq"></div><link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/butterfly-extsrc/sharejs/dist/css/share.min.css" media="print" onload="this.media='all'"><script src="https://cdn.jsdelivr.net/npm/butterfly-extsrc/sharejs/dist/js/social-share.min.js" defer></script></div></div><nav class="pagination-post" id="pagination"><a class="prev-post pull-left" href="/2024/11/08/linux-commands/" title="长更 - Linux 常用指令"><img class="cover" src="/img/cover.jpg" onerror="onerror=null;src='/img/404.jpg'" alt="cover of previous post"><div class="pagination-info"><div class="label">上一篇</div><div class="prev_info">长更 - Linux 常用指令</div></div></a><a class="next-post pull-right" href="/2024/10/15/tencent-new-start/" title="短文 - 转正日"><img class="cover" src="/img/cover.jpg" onerror="onerror=null;src='/img/404.jpg'" alt="cover of next post"><div class="pagination-info"><div class="label">下一篇</div><div class="next_info">短文 - 转正日</div></div></a></nav><div class="relatedPosts"><div class="headline"><i class="fas fa-thumbs-up fa-fw"></i><span>相关推荐</span></div><div class="relatedPosts-list"><a href="/2024/11/22/debug-windows-socket-drain/" title="问题定位回顾 - Windows 上发起 tcp 连接时提示 Only one usage of each socket address (protocol&#x2F;network address&#x2F;port) is normally permitted"><img class="cover" src="/img/cover.jpg" alt="cover"><div class="content is-center"><div class="date"><i class="far fa-calendar-alt fa-fw"></i> 2024-11-22</div><div class="title">问题定位回顾 - Windows 上发起 tcp 连接时提示 Only one usage of each socket address (protocol&#x2F;network address&#x2F;port) is normally permitted</div></div></a><a href="/2024/10/15/tencent-new-start/" title="短文 - 转正日"><img class="cover" src="/img/cover.jpg" alt="cover"><div class="content is-center"><div class="date"><i class="far fa-calendar-alt fa-fw"></i> 2024-10-15</div><div class="title">短文 - 转正日</div></div></a></div></div></div><div class="aside-content" id="aside-content"><div class="card-widget card-info is-center"><div class="avatar-img"><img src="/img/avatar.png" onerror="this.onerror=null;this.src='/img/friend_404.gif'" alt="avatar"/></div><div class="author-info-name">Linloir</div><div class="author-info-description">我、技术、生活与值得分享的一切</div><div class="site-data"><a href="/archives/"><div class="headline">文章</div><div class="length-num">21</div></a><a href="/tags/"><div class="headline">标签</div><div class="length-num">20</div></a><a href="/categories/"><div class="headline">分类</div><div class="length-num">2</div></a></div><a id="card-info-btn" target="_blank" rel="noopener" href="https://github.com/Linloir"><i class="fab fa-github"></i><span>Follow Me</span></a><div class="card-info-social-icons"><a class="social-icon" href="https://github.com/Linloir" target="_blank" title="GitHub"><i class="fab fa-github"></i></a><a class="social-icon" href="mailto:jonathanzhang.st@gmail.com" target="_blank" title="Email"><i class="fas fa-envelope"></i></a></div></div><div class="sticky_layout"><div class="card-widget" id="card-toc"><div class="item-headline"><i class="fas fa-stream"></i><span>目录</span><span class="toc-percentage"></span></div><div class="toc-content"><ol class="toc"><li class="toc-item toc-level-2"><a class="toc-link" href="#%E6%96%B9%E6%A1%88%E9%80%9F%E8%A7%88"><span class="toc-number">1.</span> <span class="toc-text">方案速览</span></a></li><li class="toc-item toc-level-2"><a class="toc-link" href="#%E8%83%8C%E6%99%AF"><span class="toc-number">2.</span> <span class="toc-text">背景</span></a></li><li class="toc-item toc-level-2"><a class="toc-link" href="#%E5%B0%9D%E8%AF%95"><span class="toc-number">3.</span> <span class="toc-text">尝试</span></a><ol class="toc-child"><li class="toc-item toc-level-3"><a class="toc-link" href="#%E9%80%9A%E8%BF%87-dalvikvm-%E6%89%A7%E8%A1%8C-dex-%E6%96%87%E4%BB%B6"><span class="toc-number">3.1.</span> <span class="toc-text">通过 dalvikvm 执行 .dex 文件</span></a></li><li class="toc-item toc-level-3"><a class="toc-link" href="#%E9%80%9A%E8%BF%87-uiautomator-1-0-%E6%89%A7%E8%A1%8C-jar-%E6%96%87%E4%BB%B6"><span class="toc-number">3.2.</span> <span class="toc-text">通过 uiautomator 1.0 执行 .jar 文件</span></a></li></ol></li><li class="toc-item toc-level-2"><a class="toc-link" href="#%E6%9C%80%E7%BB%88%E6%96%B9%E6%A1%88"><span class="toc-number">4.</span> <span class="toc-text">最终方案</span></a></li><li class="toc-item toc-level-2"><a class="toc-link" href="#%E7%BB%8F%E9%AA%8C%E4%B8%8E%E6%95%99%E8%AE%AD"><span class="toc-number">5.</span> <span class="toc-text">经验与教训</span></a></li></ol></div></div><div class="card-widget card-recent-post"><div class="item-headline"><i class="fas fa-history"></i><span>最新文章</span></div><div class="aside-list"><div class="aside-list-item no-cover"><div class="content"><a class="title" href="/2025/05/12/swift-state-binding/" title="Swift 学习笔记 - 从 Property Wrapper 视角探索 State 与 Binding 如何工作">Swift 学习笔记 - 从 Property Wrapper 视角探索 State 与 Binding 如何工作</a><time datetime="2025-05-12T15:30:52.000Z" title="发表于 2025-05-12 23:30:52">2025-05-12</time></div></div><div class="aside-list-item no-cover"><div class="content"><a class="title" href="/2025/03/17/coroutine-llm-qa/" title="Coroutine 相关疑惑大模型问答记录">Coroutine 相关疑惑大模型问答记录</a><time datetime="2025-03-17T13:09:21.000Z" title="发表于 2025-03-17 21:09:21">2025-03-17</time></div></div><div class="aside-list-item no-cover"><div class="content"><a class="title" href="/2024/12/31/summary-2023-2024/" title="年终总结 - 2023 至 2024">年终总结 - 2023 至 2024</a><time datetime="2024-12-31T17:15:49.000Z" title="发表于 2025-01-01 01:15:49">2025-01-01</time></div></div><div class="aside-list-item no-cover"><div class="content"><a class="title" href="/2024/11/22/debug-windows-socket-drain/" title="问题定位回顾 - Windows 上发起 tcp 连接时提示 Only one usage of each socket address (protocol/network address/port) is normally permitted">问题定位回顾 - Windows 上发起 tcp 连接时提示 Only one usage of each socket address (protocol/network address/port) is normally permitted</a><time datetime="2024-11-22T03:52:37.000Z" title="发表于 2024-11-22 11:52:37">2024-11-22</time></div></div><div class="aside-list-item no-cover"><div class="content"><a class="title" href="/2024/11/19/listary-quick-clone-command/" title="Listary 命令分享 - 快捷 clone 仓库并使用 VSCode 打开">Listary 命令分享 - 快捷 clone 仓库并使用 VSCode 打开</a><time datetime="2024-11-19T06:08:32.000Z" title="发表于 2024-11-19 14:08:32">2024-11-19</time></div></div></div></div></div></div></main><footer id="footer" style="background: transparent;"><div id="footer-wrap"><div class="copyright">&copy;2022 - 2025 By Linloir</div><div class="framework-info"><span>框架 </span><a target="_blank" rel="noopener" href="https://hexo.io">Hexo</a><span class="footer-separator">|</span><span>主题 </span><a target="_blank" rel="noopener" href="https://github.com/jerryc127/hexo-theme-butterfly">Butterfly</a></div><div class="footer_custom_text">Wirtten with Love ❤</div></div></footer></div><div id="rightside"><div id="rightside-config-hide"><button id="readmode" type="button" title="阅读模式"><i class="fas fa-book-open"></i></button><button id="translateLink" type="button" title="简繁转换"></button><button id="darkmode" type="button" title="日间和夜间模式切换"><i class="fas fa-adjust"></i></button><button id="hide-aside-btn" type="button" title="单栏和双栏切换"><i class="fas fa-arrows-alt-h"></i></button></div><div id="rightside-config-show"><button id="rightside-config" type="button" title="设置"><i class="fas fa-cog fa-spin"></i></button><button class="close" id="mobile-toc-button" type="button" title="目录"><i class="fas fa-list-ul"></i></button><button id="go-up" type="button" title="回到顶部"><span class="scroll-percent"></span><i class="fas fa-arrow-up"></i></button></div></div><div><script src="/js/utils.js"></script><script src="/js/main.js"></script><script src="/js/tw_cn.js"></script><script src="https://cdn.jsdelivr.net/npm/@fancyapps/ui/dist/fancybox/fancybox.umd.min.js"></script><script src="https://cdn.jsdelivr.net/npm/instant.page/instantpage.min.js" type="module"></script><script src="https://cdn.jsdelivr.net/npm/node-snackbar/dist/snackbar.min.js"></script><script>(() => {
const panguFn = () => {
if (typeof pangu === 'object') pangu.autoSpacingPage()
else {
btf.getScript('https://cdn.jsdelivr.net/npm/pangu/dist/browser/pangu.min.js')
.then(() => {
pangu.autoSpacingPage()
})
}
}
const panguInit = () => {
if (true){
GLOBAL_CONFIG_SITE.isPost && panguFn()
} else {
panguFn()
}
}
btf.addGlobalFn('pjaxComplete', panguInit, 'pangu')
document.addEventListener('DOMContentLoaded', panguInit)
})()</script><div class="js-pjax"><script>(async () => {
const showKatex = () => {
document.querySelectorAll('#article-container .katex').forEach(el => el.classList.add('katex-show'))
}
if (!window.katex_js_css) {
window.katex_js_css = true
await btf.getCSS('https://cdn.jsdelivr.net/npm/katex/dist/katex.min.css')
if (false) {
await btf.getScript('https://cdn.jsdelivr.net/npm/katex/dist/contrib/copy-tex.min.js')
}
}
showKatex()
})()</script><script>(() => {
const runMermaid = ele => {
window.loadMermaid = true
const theme = document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'default'
ele.forEach((item, index) => {
const mermaidSrc = item.firstElementChild
const mermaidThemeConfig = `%%{init:{ 'theme':'${theme}'}}%%\n`
const mermaidID = `mermaid-${index}`
const mermaidDefinition = mermaidThemeConfig + mermaidSrc.textContent
const renderFn = mermaid.render(mermaidID, mermaidDefinition)
const renderMermaid = svg => {
mermaidSrc.insertAdjacentHTML('afterend', svg)
}
// mermaid v9 and v10 compatibility
typeof renderFn === 'string' ? renderMermaid(renderFn) : renderFn.then(({ svg }) => renderMermaid(svg))
})
}
const codeToMermaid = () => {
const codeMermaidEle = document.querySelectorAll('pre > code.mermaid')
if (codeMermaidEle.length === 0) return
codeMermaidEle.forEach(ele => {
const preEle = document.createElement('pre')
preEle.className = 'mermaid-src'
preEle.hidden = true
preEle.textContent = ele.textContent
const newEle = document.createElement('div')
newEle.className = 'mermaid-wrap'
newEle.appendChild(preEle)
ele.parentNode.replaceWith(newEle)
})
}
const loadMermaid = () => {
if (true) codeToMermaid()
const $mermaid = document.querySelectorAll('#article-container .mermaid-wrap')
if ($mermaid.length === 0) return
const runMermaidFn = () => runMermaid($mermaid)
btf.addGlobalFn('themeChange', runMermaidFn, 'mermaid')
window.loadMermaid ? runMermaidFn() : btf.getScript('https://cdn.jsdelivr.net/npm/mermaid/dist/mermaid.min.js').then(runMermaidFn)
}
btf.addGlobalFn('encrypt', loadMermaid, 'mermaid')
window.pjax ? loadMermaid() : document.addEventListener('DOMContentLoaded', loadMermaid)
})()</script></div><script id="canvas_nest" defer="defer" color="165,165,165" opacity="0.8" zIndex="-1" count="99" mobile="false" src="https://cdn.jsdelivr.net/npm/butterfly-extsrc/dist/canvas-nest.min.js"></script><script src="https://cdn.jsdelivr.net/npm/pjax/pjax.min.js"></script><script>(() => {
const pjaxSelectors = ["head > title","#config-diff","#body-wrap","#rightside-config-hide","#rightside-config-show",".js-pjax"]
window.pjax = new Pjax({
elements: 'a:not([target="_blank"])',
selectors: pjaxSelectors,
cacheBust: false,
analytics: false,
scrollRestoration: false
})
const triggerPjaxFn = (val) => {
if (!val) return
Object.values(val).forEach(fn => fn())
}
document.addEventListener('pjax:send', () => {
// removeEventListener
btf.removeGlobalFnEvent('pjaxSendOnce')
btf.removeGlobalFnEvent('themeChange')
// reset readmode
const $bodyClassList = document.body.classList
if ($bodyClassList.contains('read-mode')) $bodyClassList.remove('read-mode')
triggerPjaxFn(window.globalFn.pjaxSend)
})
document.addEventListener('pjax:complete', () => {
btf.removeGlobalFnEvent('pjaxCompleteOnce')
document.querySelectorAll('script[data-pjax]').forEach(item => {
const newScript = document.createElement('script')
const content = item.text || item.textContent || item.innerHTML || ""
Array.from(item.attributes).forEach(attr => newScript.setAttribute(attr.name, attr.value))
newScript.appendChild(document.createTextNode(content))
item.parentNode.replaceChild(newScript, item)
})
triggerPjaxFn(window.globalFn.pjaxComplete)
})
document.addEventListener('pjax:error', e => {
if (e.request.status === 404) {
pjax.loadUrl('/404')
}
})
})()</script><script async data-pjax src="//busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js"></script></div></body></html>