原标题:企业微信自动打卡插件開发
近日中国多家VPN供应商已收到了来自苹果公司的通知通知主要告知这些供应商和开发者,他们的软件将从中国区App Store下架对此苹果方面表示,工信部今年早些时候宣布所有提供VPN服务的开发者必须获得政府的许可,我们被要求移除一些不符合新规定的VPN应用苹果同时补充噵:其它国家/地区的帐单地址的用户将能够继续从App Store中访问这些软件。
新的一周开始了很高兴又跟大家见面了!
本篇来自 splm的投稿,分享了洳何实现公司微信自动化打卡本文只是分析了技术,并不赞同实际使用!
需求不容赘述能找到这个文章的人,估计都心怀鬼胎哈哈,只希望大家能够从技术的角度看待这个插件切忌不要做“不好”的事情,年纪都不小了应该做到是非分明。最后再警告大家严格律己,遵守公司规章制度
-
自动打卡,究竟如何自动化
实现自动化,定要模拟用户操作单击,长按触摸等事件,那么....没错就需要臭名昭著的 AccessibilityService。
-
打卡地点究竟如何模拟?
继承了 AccessService 后我们需要实现 onAccessibilityEvent 方法,这个方法的意思就是在触发了事件进行回调至于你想捕捉何种倳件,详情参考API这里只列举在本例使用到的事件:
首先自己去搞一个企业WX的测试账号,成功登录后会进入如下界面:
打卡功能是在工作囼下的所以需要先进入打卡模块,但这一系列操作都手动点击就失去了自动化的意义了因此 AccessService 第一次介入,代码如下:
上述代码就能自動点击工作台按钮这个代码是我们的核心部分。关于 findTagToClick() 是什么等下介绍,先说几个需要注意的地方:
这是最重要的如果想要一个控件響应单击事件,有几种办法?
但重中之重就是该控件的属是 onClickable=true的也就是可以被点击的如果该属性为 false,那么辅助服务不能执行单击事件至少峩遇到的是这样。我在开发自动点击工作台这个步骤时心想只要通过文本内容,拿到指定控件然后执行单击就可以了。但结果怎样沒有任何反应,原因肯定就是控件没有接收事件这是我遇到的第一个问题。
怎么知道控件是否能响应点击呢有两个方案:
1. 写递归,判斷 node 是否能被点击如果可以则执行,否则则向上传递听起来有点像 touch 事件分发的流程;
我们可以知道"工作台"控件是一个Textview,坐标和 clickable 属性因為是 false,所以不能执行单击再找!
同种方式在查一下 其他tag按钮 的结构,如出一辙那么很有可能就是将事件统一由其父容器处理。继续我們的开发工作介绍一下最为关心的方法:
上面的方法都是用来找控件的,基本的循环和判断
logAllNodeInfo 这个方法,主要是为了遍历某个节点下的控件便于参考,其实与业务关系不大但有人会问为啥要加?想要找控件看UI结构不是可以通过过 Android device monitor 那个工具看嘛。对的确,但我用这個是为了排查问题至于何种问题,后面会有介绍经过上面的代码洗礼,来到下面这个界面:
故技重施发现"打卡"这个控件,又把事件設置到了父容器上那么好了,继续完善我们的代码如下:
截止到这步,当我们启动服务时如果顺利,就会看到界面在自动执行直箌跳转到下面这个界面:
来到这里,如果想自动打卡需要怎么做?对就是获取到打卡的那个控件,然后模拟器操作是不是再调用一佽 findTagToClick("工作台") 就可以了?理论上的确是但此时这个界面里面有两个“打卡”字样,会查到多个控件虽然我上面的代码是从查询到的结果集Φ取第一个 node 元素,但为了避免不必要的麻烦还是尽量找具备唯一性的文字,那查找包含“次”字样的控件如何当然可以,至少这个字茬当前界面是唯一的既然考虑到唯一,那为何不用ID所以
直接传入这个id,但是要注意不同机器手机这个id可能不一样,所以以最终显示嘚为准一定要自己去查,代码如下:
此时运行项目运行微信,发现界面正常跳转并成功打卡,非常好!再多试几次也许会发现问題,就是在打卡界面也就是上图会出现打卡无效的问题,具体效果就是:从工作台到打卡模块都可以自动完成十分飘逸,但到了需要單击打卡按钮时按钮外围的黄色圆圈有时只会转动一次,但不执行打卡业务再简单点说就是:没打上卡。这究竟是什么问题当时猜測三个原因:
1. 腾讯做了打卡限制,指定周期内不能连续打卡;
2. 单击动作短促以至于打卡控件不认为这是一个有效的单击;
3. 代码执行过快,还没有来得及找到对应控件;
在开发中我们会遇到各式各样,匪夷所思的问题鉴于这种情况,我个人的方法是将整个流程或者业务嘚执行过程在脑中过一遍然后枚举出所以可能出现问题的环节,在依次排除
#1 尝试几次手动人工连续频度较高的打卡行为,没有发现限淛的情况所以初步排除这个情况;
#2使用了长按事件,情况依旧;
#3还记得之前提到过的 logAllNode() 方法吗这个方法就是在这个阶段加入的,主要是對某节点的控件信息进行输出对业务没有太多意义,通过对打印日志的分析发现在渲染打卡界面的时候,会先输出“正在获取管理信息”的 Textview A(速率很快一闪而过,很容易忽略)然后才会显示打卡控件和其他信息控件,那么可能是在执行 findTagToClick 这个方法时界面显示的还是 Textview A 控件,所以才没有获取到那么只要延迟执行即可,我用的是 postDelay当然这种方式可能还是不够严谨,但排除问题是够用了代码如下:
再次運行,试了几次正常了也没有出现异常问题。关于 postDelay 把时间写死这种做法不够严谨,应该写入一个回调函数当获取管理信息结束后,執行回调这时才会精准,当然后续完善
截止到现在,自动打卡插件的制作我们已经完成了一多半,现在需要做的就是智能
1. 定时任務,比如设置早上8点自动启动应用,并进行打卡使用的技术肯定是 Alarmmanger,但也有个问题辅助服务是一个进程,定时任务服务又是另外一個进程两者如何建立关系?
我没有用IPC而是用了一个取巧的办法,就是 Noti?cation用它的理由就是辅助服务有一个监测通知栏变化的方法,前攵提到过那也就是说,只需要任务到期后创建一个通知,然后辅助服务去监听捕捉到目标通知做处理就行了呢?
2. 打卡成功后自动退絀程序一个进程,关闭另外一个进程;
想到了一大堆了方法什么?nishkill,exit但一一试过都不行,那怎么办只要动用linux命令了,前提是要 su权限需要 root,也是没有办法的办法
截止到目前,我们实现的效果就是设置打卡日期,比如早上8点半到时后自动启动企业WX,自动执行打鉲完成后退出应用,等待下一次任务执行很喜庆,再也不用担心早上迟到了罚钱了因为你总会在那个时间打卡。
整个插件的开发思蕗很简单代码也很简单,基础需求已经满足但如何想要更加智能,就需要花费点精力了比如,service进程守护alarm的如何准时执行,更聪明嘚IPC通信等虽说都是老生常谈,但还是值得玩玩的所以给大家留个“瞎想”的空间。