简单Espresso测试“循环超过60秒的x次迭代”错误

时间:2016-04-29 15:39:33

标签: android unit-testing android-espresso

我实际上尝试使用Espresso设置一些单元测试,经过几个小时的研究后,应用程序只进行点击并通过EditText获得焦点,但之后没有任何内容

Caused by: android.support.test.espresso.AppNotIdleException: Looped for 1996 iterations over 60 SECONDS. The following Idle Conditions failed .

我删除了所有动画& SwipeRefreshLayout因为我看到swiperefresh有一个错误

我实际上使用了一些回调来替换Activity

中的当前片段

如果有人有一些提示,我会在搜索u_u

4小时后离开

谢谢你:)

My Gradle依赖项:

// App dependencies
compile 'com.android.support:support-annotations:23.3.0'
compile 'com.google.guava:guava:18.0'
// Testing-only dependencies
// Force usage of support annotations in the test app, since it is internally used by the runner module.
androidTestCompile 'com.android.support:support-annotations:23.3.0'
androidTestCompile 'com.android.support.test:runner:0.4.1'
androidTestCompile 'com.android.support.test:rules:0.4.1'
androidTestCompile 'com.android.support.test.espresso:espresso-core:2.2.2'
androidTestCompile 'com.android.support.test.espresso:espresso-intents:2.2.2'

我在defaultConfig上添加了这个:

testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"

有我的测试:

@RunWith(AndroidJUnit4.class)
@LargeTest
public class HelloWorldEspressoTest {

    public static final String USERNAME = "do_f";

    @Rule
    public ActivityTestRule<LoginActivity> mActivityRule = new ActivityTestRule(LoginActivity.class);

    private LoginActivity mActivity = null;

    @Before
    public void setActivity() {
        mActivity = mActivityRule.getActivity();
    }

    @Test
    public void login_LoginActivity() {

        onView(withId(R.id.menu_login)).perform(click());

        onView(withId(R.id.login_username))
                .perform(typeText(USERNAME), closeSoftKeyboard());
    }
}

有我的活动:

public class LoginActivity extends AppCompatActivity
        implements MenuFragment.OnFragmentInteractionListener {

    private FragmentManager fm;

    public static void newActivity(Activity activity)
    {
        Intent i = new Intent(activity, LoginActivity.class);
        activity.startActivity(i);
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_login);
        SharedPreferences sp = getSharedPreferences(Utils.SP, Context.MODE_PRIVATE);

        if (!sp.getString(Utils.TOKEN, "null").equals("null"))
        {
            MainActivity.newActivity(this);
            finish();
        }

        fm = getFragmentManager();
        fm.beginTransaction()
                .replace(R.id.container, MenuFragment.newInstance())
                .addToBackStack(null)
                .commit();
    }

    @Override
    public void onBackPressed()
    {
        if (getFragmentManager().getBackStackEntryCount() > 1)
            getFragmentManager().popBackStack();
        else
            super.onBackPressed();
    }

    @Override
    public void showLogin() {
        fm.beginTransaction()
                .replace(R.id.container, LoginFragment.newInstance())
                .addToBackStack(null)
                .commit();
    }

    @Override
    public void showRegister() {
        fm.beginTransaction()
                .replace(R.id.container, RegisterFragment.newInstance())
                .addToBackStack(null)
                .commit();
    }
}

我的LoginFragment:

    private static final String        TAG = "LoginFragment";

@Bind(R.id.loading_spinner)
ProgressBar loading;

@Bind(R.id.content)
LinearLayout content;

@Bind(R.id.login_username)
EditText        username;

@Bind(R.id.login_password)
EditText        password;

public LoginFragment() {

}

public static LoginFragment newInstance() {
    return new LoginFragment();
}

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
}

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
                         Bundle savedInstanceState) {
    View v = inflater.inflate(R.layout.login_fragment_login, container, false);
    ButterKnife.bind(this, v);
    return v;
}

@Override
public void onActivityCreated(Bundle saveInstanceState) {
    super.onActivityCreated(saveInstanceState);
}

@OnClick(R.id.login_submit)
public void onSubmit(View v)
{
    if (username.getText().length() == 0
            || password.getText().length() == 0)
    {
        Snackbar.make(getView(), "blabla", Snackbar.LENGTH_SHORT).show();
        return ;
    }
    //hideContent();
    LoginPost p = new LoginPost(username.getText().toString(), password.getText().toString());
    Call<LoginResponse> call = RestClient.get().login(p);
    call.enqueue(new Callback<LoginResponse>() {
        @Override
        public void onResponse(Call<LoginResponse> call, Response<LoginResponse> response) {
            if (response.body() == null) {
                String msg = getResources().getString(R.string.login_error_bad_credentials);
                Snackbar.make(getView(), msg, Snackbar.LENGTH_SHORT).show();
                //showContent();
            } else {
                SharedPreferences sp = getActivity().getSharedPreferences(Utils.SP, Context.MODE_PRIVATE);
                sp.edit().putString(Utils.TOKEN, response.body().getToken()).apply();
                sp.edit().putString(Utils.USERNAME, username.getText().toString()).apply();
                MainActivity.newActivity(getActivity());
                getActivity().finish();
            }
        }

        @Override
        public void onFailure(Call<LoginResponse> call, Throwable t) {
            Snackbar.make(getView(), "onFailure", Snackbar.LENGTH_SHORT).show();
            //showContent();
        }
    });
}

2 个答案:

答案 0 :(得分:6)

我找到了解决方案,循环迭代是由此引起的

<ProgressBar
    android:id="@+id/loading_spinner"
    android:layout_width="150dp"
    android:layout_height="150dp"
    android:layout_centerVertical="true"
    android:layout_centerHorizontal="true"
    android:indeterminateTintMode="src_atop"
    android:indeterminateTint="@color/ganjify"
    android:alpha="0"
    android:layout_gravity="center" />

因为当我不使用时,我没有将可见性设置为GONE!

loading.setVisibility(View.GONE);

答案 1 :(得分:2)

Espresso旨在等待直到您的应用程序中没有待处理的UI动画或AsyncTasks之后,再进行测试。在the docs中,重点是我:

  

Espresso提供了一套完善的同步功能。但是,该框架的这一特征仅适用于将消息发布到MessageQueue上的操作,例如,在屏幕上绘制其内容的View的子类。

android.support.test.espresso.AppNotIdleException的最可能原因是Espresso正在等待其完成的UI动画或AsyncTask。

例如,如果您的布局中有一个ProgressBar元素,则您需要设置可见性loading.setVisibility(View.GONE);,因为Espresso将该元素解释为正在进行的UI动画,并且将等待它消失以继续进行测试