Appium UiWatchers 监听解决各种非期待弹窗,弹层,弹弹弹等问题

success · 2018年04月13日 · 最后由 success 回复于 2018年05月11日 · 1442 次阅读

app自动化时,各种不期待的弹层弹窗,升级广告等时有飞出,由于弹窗具有不定时,不定页面等很多不确定性。有的弹窗很不友好,不×掉,很难进行下一步操作,造成 测试用例失败。而判断是否有弹窗,弹层很麻烦。
研究一下 appium和手机通信的原理就不难发现,运行appium时推送手机AppiumBootstrap.jar的中,有这么一段代码再listenForever

/**
 * The Bootstrap class runs the socket server.
 * 
 */
public class Bootstrap extends UiAutomatorTestCase {

  public void testRunServer() {
    Find.params = getParams();
    final boolean disableAndroidWatchers = Boolean.parseBoolean(getParams()
        .getString("disableAndroidWatchers"));
    final boolean acceptSSLCerts = Boolean.parseBoolean(getParams().getString(
        "acceptSslCerts"));

    SocketServer server;
    try {
      server = new SocketServer(4724);
      server.listenForever(disableAndroidWatchers, acceptSSLCerts);
    } catch (final SocketServerException e) {
      Logger.error(e.getError());
      System.exit(1);
    }

  }
}

那么我们可以利用这个长监听,干点点掉弹窗的事儿会不会很爽,弹窗一弹出,就把他点掉,点掉,点掉。
其实在UiWatchers.java中,作者已经写了很多UI监听了,贴一小段代码。
public class UiWatchers {
  private static final String LOG_TAG = UiWatchers.class.getSimpleName();
  private final List<String>  mErrors = new ArrayList<String>();

  /**
   * We can use the UiDevice registerWatcher to register a small script to be
   * executed when the framework is waiting for a control to appear. Waiting may
   * be the cause of an unexpected dialog on the screen and it is the time when
   * the framework runs the registered watchers. This is a sample watcher
   * looking for ANR and crashes. it closes it and moves on. You should create
   * your own watchers and handle error logging properly for your type of tests.
   */
  public void registerAnrAndCrashWatchers() {

    UiDevice.getInstance().registerWatcher("ANR", new UiWatcher() {
      @Override
      public boolean checkForCondition() {
        UiObject window = new UiObject(new UiSelector()
            .className("com.android.server.am.AppNotRespondingDialog"));
        String errorText = null;
        if (window.exists()) {
          try {
            errorText = window.getText();
          } catch (UiObjectNotFoundException e) {
            Log.e(LOG_TAG, "dialog gone?", e);
          }
          onAnrDetected(errorText);
          postHandler();
          return true; // triggered
        }
        return false; // no trigger
      }
    });

在这个类中我们加入自己的弹窗监听
  public void registerMyPopupWatcher() {
        UiDevice.getInstance().registerWatcher("myPopup", new UiWatcher() {
          @Override
          public boolean checkForCondition() {

            //广告提示
            UiObject ggPop = new UiObject(new UiSelector().resourceId("com.gift.android:id/close_view"));

            //站点切换
            UiObject addChgPop = new UiObject(new UiSelector().resourceId("com.gift.android:id/bt_cancel"));

            //升级提示
            UiObject upPop = new UiObject(new UiSelector().resourceId("com.gift.android:id/close"));

            if (upPop.exists()) {
                System.out.println("you have a updateApp popup window");
                try {
                    upPop.clickAndWaitForNewWindow();
                    return true; // triggered
                } catch (UiObjectNotFoundException e) {
                    Log.e(LOG_TAG, "Exception", e);
                }
            }else if (ggPop.exists()) {
                System.out.println("you have a popup window");
                try {
                    ggPop.clickAndWaitForNewWindow();
                    return true; // triggered
                } catch (UiObjectNotFoundException e) {
                    Log.e(LOG_TAG, "Exception", e);
                }
            }else if (addChgPop.exists()) {
                System.out.println("you have a change site popup window");
                try {
                    addChgPop.clickAndWaitForNewWindow();
                    return true; // triggered
                } catch (UiObjectNotFoundException e) {
                    Log.e(LOG_TAG, "Exception", e);
                }
            }
            return false; // no trigger
          }
        });

        Log.i(LOG_TAG, "Registered myPopupPopup Watchers");
      }
listenForever的方法中去调用一下,这样我们也监听Forever了。

写完编译成AppiumBootstrap.jar包,放在\Appium\node_modules\appium\build\android_bootstrap\下面。随着appium服务启动时,推送到手机中。
跑一下看效果。
app中弹窗中有如下resourceId的,统统被点掉了。
resourceId="com.gift.android:id/close_view";
resourceId="com.gift.android:id/bt_cancel";
resourceId="com.gift.android:id/close";
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
共收到 18 条回复 时间 点赞

这个修改对uiautomator2有效吗?

layasa 回复

uiautomator2没试,UI2应该也有类似的方法吧。你研究一下

你这效果图被吃了

哈哈哈,抱歉啊、不知道为啥贴图没有贴上

ua1和ua2里面都有对应的实现,不过他们写死了很多代码,应该是通过读取capability里面的参数动态设置的。

seveniruby 回复

capability具体配置方式没有研究,发现代码里面有,改代码也蛮方便的

@success 干货! 谢谢 比较通用了.

harsayer 回复

谢谢小马哥。

seveniruby 回复

动态配置的方式,我再研究研究。能动态配置确实不错,代码写死了的话,就有点不灵活了。多谢给予的思路点拨

success 回复

回头咱们社区成立以appium的改进开源团队吧,一起研究下如何改进。

seveniruby 回复

不错不错,期待中。

io.appium.java_client.events.api.general.AppiumWebDriverEventListener api其实有提供相关的监听器的

samfu 回复

多谢大神。这api我好好学学😙

楼主,你好

你的方法是,修改 AppiumBootstrap.jar 达到监控弹出框的效果。
如果不修改这个jar,Appium 服务器能监控到手机报错了吗?

chujidiy 回复

不修改这个jar,就要自己在手机里部署个服务开启个监听,来做类似的事情了把?感觉这样得不偿失啊。

感觉Appium既然能监听到手机端的报错,应该有比较简便的方法可以把监听到的报错信息保存下来吧

不过我找了很久,没有找到好方法。

所以我最后用了一个很笨的放方法,用 adb monitor 监控报错。

chujidiy 回复

Appium 其实也是和这个appiumbootstrap交互的。保存信息的话。可以写log的

需要 登录 后方可回复, 如果你还没有账号请点击这里 注册