summaryrefslogtreecommitdiffstats
path: root/tests
diff options
context:
space:
mode:
authorChiao Cheng <chiaocheng@google.com>2012-09-14 12:00:21 -0700
committerChiao Cheng <chiaocheng@google.com>2012-09-17 15:15:21 -0700
commitbf1f360793c466dae12f8bc386161353355f2b28 (patch)
tree426c813c050a9a32a801f5653c3eae0d861309a7 /tests
parent5a6b519447147a91eda94142c5667e86b2d4456c (diff)
downloadandroid_packages_apps_ContactsCommon-bf1f360793c466dae12f8bc386161353355f2b28.tar.gz
android_packages_apps_ContactsCommon-bf1f360793c466dae12f8bc386161353355f2b28.tar.bz2
android_packages_apps_ContactsCommon-bf1f360793c466dae12f8bc386161353355f2b28.zip
Adding custom AsyncQueryHandler to prevent null cursors.
In safe mode, a cursor can come back as null even though our providers never allow for that. NoNullCursorAsyncQueryHandler should be used instead of AsyncQueryHandler from now on. It will return EmptyCursor in place of a null. Bug: 7012260 Change-Id: I5b8ba6fb85ae0f984ee8b0ff2f2f068232a03848
Diffstat (limited to 'tests')
-rw-r--r--tests/Android.mk38
-rw-r--r--tests/AndroidManifest.xml30
-rw-r--r--tests/proguard.flags9
-rw-r--r--tests/src/com/android/contacts/common/database/NoNullCursorAsyncQueryHandlerTest.java152
4 files changed, 229 insertions, 0 deletions
diff --git a/tests/Android.mk b/tests/Android.mk
new file mode 100644
index 00000000..4ae1a0d1
--- /dev/null
+++ b/tests/Android.mk
@@ -0,0 +1,38 @@
+# Copyright 2012, The Android Open Source Project
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+LOCAL_PATH:= $(call my-dir)
+include $(CLEAR_VARS)
+
+# We only want this apk build for tests.
+LOCAL_MODULE_TAGS := tests
+LOCAL_CERTIFICATE := shared
+
+LOCAL_JAVA_LIBRARIES := android.test.runner
+
+# Include all test java files.
+src_dirs := src ../src
+res_dirs := res ../res
+LOCAL_SRC_FILES := $(call all-java-files-under, $(src_dirs))
+LOCAL_RESOURCE_DIR := $(addprefix $(LOCAL_PATH)/, $(res_dirs))
+
+LOCAL_AAPT_FLAGS := \
+ --auto-add-overlay \
+ --extra-packages com.android.contacts.common
+
+LOCAL_STATIC_JAVA_LIBRARIES += com.android.contacts.common.test com.android.phone.shared
+
+LOCAL_PACKAGE_NAME := ContactsCommonTests
+
+include $(BUILD_PACKAGE)
diff --git a/tests/AndroidManifest.xml b/tests/AndroidManifest.xml
new file mode 100644
index 00000000..9687a662
--- /dev/null
+++ b/tests/AndroidManifest.xml
@@ -0,0 +1,30 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--
+ ~ Copyright (C) 2012 The Android Open Source Project
+ ~
+ ~ Licensed under the Apache License, Version 2.0 (the "License");
+ ~ you may not use this file except in compliance with the License.
+ ~ You may obtain a copy of the License at
+ ~
+ ~ http://www.apache.org/licenses/LICENSE-2.0
+ ~
+ ~ Unless required by applicable law or agreed to in writing, software
+ ~ distributed under the License is distributed on an "AS IS" BASIS,
+ ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ ~ See the License for the specific language governing permissions and
+ ~ limitations under the License
+ -->
+
+<manifest xmlns:android="http://schemas.android.com/apk/res/android"
+ package="com.android.contacts.common.unittest">
+
+ <application>
+ <uses-library android:name="android.test.runner" />
+ </application>
+
+ <instrumentation android:name="android.test.InstrumentationTestRunner"
+ android:targetPackage="com.android.contacts.common.unittest"
+ android:label="Contacts common unit tests">
+ </instrumentation>
+
+</manifest>
diff --git a/tests/proguard.flags b/tests/proguard.flags
new file mode 100644
index 00000000..883dde6d
--- /dev/null
+++ b/tests/proguard.flags
@@ -0,0 +1,9 @@
+# Any class or method annotated with NeededForTesting or NeededForReflection.
+-keep @com.android.contacts.test.NeededForTesting class *
+-keep @com.android.contacts.test.NeededForReflection class *
+-keepclassmembers class * {
+@com.android.contacts.test.NeededForTesting *;
+@com.android.contacts.test.NeededForReflection *;
+}
+
+-verbose
diff --git a/tests/src/com/android/contacts/common/database/NoNullCursorAsyncQueryHandlerTest.java b/tests/src/com/android/contacts/common/database/NoNullCursorAsyncQueryHandlerTest.java
new file mode 100644
index 00000000..a2b635d0
--- /dev/null
+++ b/tests/src/com/android/contacts/common/database/NoNullCursorAsyncQueryHandlerTest.java
@@ -0,0 +1,152 @@
+/*
+ * Copyright (C) 2012 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License
+ */
+
+package com.android.contacts.common.database;
+
+import android.database.Cursor;
+import android.net.Uri;
+import android.test.InstrumentationTestCase;
+import android.test.mock.MockContentProvider;
+import android.test.mock.MockContentResolver;
+
+import java.util.concurrent.CountDownLatch;
+import java.util.concurrent.TimeUnit;
+
+/**
+ * Unit test for {@link NoNullCursorAsyncQueryHandler}
+ */
+public class NoNullCursorAsyncQueryHandlerTest extends InstrumentationTestCase {
+
+ private MockContentResolver mMockContentResolver;
+
+ private static final String AUTHORITY = "com.android.contacts.common.unittest";
+ private static final Uri URI = Uri.parse("content://" + AUTHORITY);
+ private static final String[] PROJECTION = new String[]{"column1", "column2"};
+
+ @Override
+ public void setUp() throws Exception {
+ super.setUp();
+ mMockContentResolver = new MockContentResolver();
+ final MockContentProvider mMockContentProvider = new MockContentProvider() {
+ @Override
+ public Cursor query(Uri uri, String[] projection, String selection,
+ String[] selectionArgs,
+ String sortOrder) {
+ return null;
+ }
+ };
+
+ mMockContentResolver.addProvider(AUTHORITY, mMockContentProvider);
+ }
+
+ public void testCursorIsNotNull() throws Throwable {
+
+ final CountDownLatch latch = new CountDownLatch(1);
+ final ObjectHolder<Cursor> cursorHolder = ObjectHolder.newInstance();
+ final ObjectHolder<Boolean> ranHolder = ObjectHolder.newInstance(false);
+
+ runTestOnUiThread(new Runnable() {
+
+ @Override
+ public void run() {
+
+ NoNullCursorAsyncQueryHandler handler = new NoNullCursorAsyncQueryHandler(
+ mMockContentResolver) {
+ @Override
+ protected void onNotNullableQueryComplete(int token, Object cookie,
+ Cursor cursor) {
+ cursorHolder.obj = cursor;
+ ranHolder.obj = true;
+ latch.countDown();
+ }
+ };
+ handler.startQuery(1, null, URI, PROJECTION, null, null, null);
+ }
+ });
+
+ latch.await(5, TimeUnit.SECONDS);
+ assertFalse(cursorHolder.obj == null);
+ assertTrue(ranHolder.obj);
+ }
+
+ public void testCursorContainsCorrectCookies() throws Throwable {
+ final ObjectHolder<Boolean> ranHolder = ObjectHolder.newInstance(false);
+ final CountDownLatch latch = new CountDownLatch(1);
+ final ObjectHolder<Object> cookieHolder = ObjectHolder.newInstance();
+ final String cookie = "TEST COOKIE";
+ runTestOnUiThread(new Runnable() {
+ @Override
+ public void run() {
+ final NoNullCursorAsyncQueryHandler handler = new NoNullCursorAsyncQueryHandler(
+ mMockContentResolver) {
+ @Override
+ protected void onNotNullableQueryComplete(int token, Object cookie,
+ Cursor cursor) {
+ ranHolder.obj = true;
+ cookieHolder.obj = cookie;
+ latch.countDown();
+ }
+ };
+ handler.startQuery(1, cookie, URI, PROJECTION, null, null, null);
+ }
+ });
+
+ latch.await(5, TimeUnit.SECONDS);
+ assertSame(cookie, cookieHolder.obj);
+ assertTrue(ranHolder.obj);
+ }
+
+ public void testCursorContainsCorrectColumns() throws Throwable {
+ final ObjectHolder<Boolean> ranHolder = ObjectHolder.newInstance(false);
+ final CountDownLatch latch = new CountDownLatch(1);
+ final ObjectHolder<Cursor> cursorHolder = ObjectHolder.newInstance();
+ final String cookie = "TEST COOKIE";
+ runTestOnUiThread(new Runnable() {
+ @Override
+ public void run() {
+ final NoNullCursorAsyncQueryHandler handler = new NoNullCursorAsyncQueryHandler(
+ mMockContentResolver) {
+ @Override
+ protected void onNotNullableQueryComplete(int token, Object cookie,
+ Cursor cursor) {
+ ranHolder.obj = true;
+ cursorHolder.obj = cursor;
+ latch.countDown();
+ }
+ };
+ handler.startQuery(1, cookie, URI, PROJECTION, null, null, null);
+ }
+ });
+
+ latch.await(5, TimeUnit.SECONDS);
+ assertSame(PROJECTION, cursorHolder.obj.getColumnNames());
+ assertTrue(ranHolder.obj);
+ }
+
+ private static class ObjectHolder<T> {
+ public T obj;
+
+ public static <E> ObjectHolder<E> newInstance() {
+ return new ObjectHolder<E>();
+ }
+
+ public static <E> ObjectHolder<E> newInstance(E value) {
+ ObjectHolder<E> holder = new ObjectHolder<E>();
+ holder.obj = value;
+ return holder;
+ }
+ }
+}