1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.dialer.janktests;
18 
19 import android.content.ContentProviderOperation;
20 import android.content.ContentValues;
21 import android.content.Intent;
22 import android.content.OperationApplicationException;
23 import android.content.pm.PackageManager;
24 import android.database.Cursor;
25 import android.net.Uri;
26 import android.os.RemoteException;
27 import android.os.SystemClock;
28 import android.provider.BaseColumns;
29 import android.provider.CallLog;
30 import android.provider.ContactsContract;
31 import android.provider.ContactsContract.CommonDataKinds;
32 import android.provider.ContactsContract.CommonDataKinds.Phone;
33 import android.provider.ContactsContract.CommonDataKinds.StructuredName;
34 import android.provider.ContactsContract.RawContacts;
35 
36 import androidx.test.jank.GfxMonitor;
37 import androidx.test.jank.JankTest;
38 import androidx.test.jank.JankTestBase;
39 import androidx.test.uiautomator.By;
40 import androidx.test.uiautomator.Direction;
41 import androidx.test.uiautomator.UiDevice;
42 import androidx.test.uiautomator.UiObject2;
43 import androidx.test.uiautomator.UiObjectNotFoundException;
44 import androidx.test.uiautomator.Until;
45 
46 import java.util.ArrayList;
47 import java.util.Random;
48 
49 /**
50  * Jank test for Dialer app
51  * open a contact, initiate call to open the dialing screen
52  * fling call log
53  */
54 public class DialerJankTests extends JankTestBase {
55     private static final int TIMEOUT = 5000;
56     private static final int INNER_LOOP = 5;
57     private static final int EXPECTED_FRAMES = 100;
58     private static final String PACKAGE_NAME = "com.google.android.dialer";
59     private static final String RES_PACKAGE_NAME = "com.android.dialer";
60     private static final String RES_PACKAGE_NAME2 = "com.android.contacts";
61     private static final String RES_PACKAGE_NAME3 = "android";
62     private static final String APP_NAME = "Phone";
63     private static final String CONTACT_NAME = "A AAA Test Account";
64     private static final String CONTACT_NUMBER = "2468";
65     private UiDevice mDevice;
66     static final int PICK_CONTACT_REQUEST = 1;
67 
68     @Override
setUp()69     public void setUp() throws Exception {
70         super.setUp();
71         mDevice = UiDevice.getInstance(getInstrumentation());
72         mDevice.setOrientationNatural();
73     }
74 
75     @Override
tearDown()76     protected void tearDown() throws Exception {
77         mDevice.unfreezeRotation();
78         super.tearDown();
79     }
80 
launchApp(String packageName)81     public void launchApp(String packageName) {
82         PackageManager pm = getInstrumentation().getContext().getPackageManager();
83         Intent appIntent = pm.getLaunchIntentForPackage(packageName);
84         appIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
85         getInstrumentation().getContext().startActivity(appIntent);
86         mDevice.waitForIdle();
87     }
88 
launchDialer()89     public void launchDialer () throws OperationApplicationException, RemoteException {
90         if (!doesContactExist()) {
91             insertNewContacts();
92         }
93         launchApp(PACKAGE_NAME);
94         mDevice.waitForIdle();
95 
96         // Open contacts list
97         UiObject2 contacts = mDevice.wait(Until.findObject(By.desc("Contacts")), TIMEOUT);
98         assertNotNull("Contacts can't be found", contacts);
99         contacts.clickAndWait(Until.newWindow(), TIMEOUT);
100         // Find a contact by a given contact-name
101         UiObject2 contactName = mDevice.wait(Until.findObject(
102             By.res(RES_PACKAGE_NAME, "cliv_name_textview").text(CONTACT_NAME)), TIMEOUT);
103         assertNotNull("Contactname can't be found", contactName);
104         contactName.clickAndWait(Until.newWindow(), TIMEOUT);
105         // Click on dial-icon beside contact-number to ensure test is ready to be executed
106         UiObject2 contactNumber = mDevice.wait(Until.findObject(
107             By.res(RES_PACKAGE_NAME2,"header").text(CONTACT_NUMBER)), TIMEOUT);
108         assertNotNull("Contact number can't be found", contactNumber);
109         contactNumber.clickAndWait(Until.newWindow(), TIMEOUT);
110 
111         UiObject2 endCall = mDevice.wait(Until.findObject(By.res(RES_PACKAGE_NAME,
112               "floating_end_call_action_button")), 2 * TIMEOUT);
113         endCall.clickAndWait(Until.newWindow(), TIMEOUT);;
114         SystemClock.sleep(200);
115     }
116 
117     @JankTest(beforeTest="launchDialer", expectedFrames=EXPECTED_FRAMES)
118     @GfxMonitor(processName=PACKAGE_NAME)
testDialerCallInit()119     public void testDialerCallInit() {
120         for (int i = 0; i < INNER_LOOP; i++) {
121             UiObject2 contactNumber = mDevice.wait(Until.findObject(
122                     By.res(RES_PACKAGE_NAME2,"header").text(CONTACT_NUMBER)), TIMEOUT);
123             assertNotNull("Contact number can't be found", contactNumber);
124             contactNumber.clickAndWait(Until.newWindow(), TIMEOUT);
125             UiObject2 endCall = mDevice.wait(Until.findObject(By.res(RES_PACKAGE_NAME,
126                       "floating_end_call_action_button")), 2 * TIMEOUT);
127             endCall.clickAndWait(Until.newWindow(), TIMEOUT);
128             SystemClock.sleep(200);
129         }
130     }
131 
launchCallLog()132     public void launchCallLog() throws UiObjectNotFoundException {
133         if (getCallLogCount() < 100) {
134             for (int i = 0; i < 100; i++) {
135                 addNumToCalLog(getRandomPhoneNumber());
136             }
137         }
138         launchApp(PACKAGE_NAME);
139         mDevice.waitForIdle();
140         // Find 'Call History' and click
141         mDevice.wait(Until.findObject(By.desc("Call History")), TIMEOUT).click();
142         mDevice.wait(Until.findObject(By.res(RES_PACKAGE_NAME,"lists_pager")), TIMEOUT);
143     }
144 
145     @JankTest(beforeTest="launchCallLog", expectedFrames=EXPECTED_FRAMES)
146     @GfxMonitor(processName=PACKAGE_NAME)
testDialerCallLogFling()147     public void testDialerCallLogFling() {
148         UiObject2 callLog = mDevice.wait(Until.findObject(
149                 By.res(RES_PACKAGE_NAME,"lists_pager")), TIMEOUT);
150         assertNotNull("Call log can't be found", callLog);
151         for (int i = 0; i < INNER_LOOP; i++) {
152             callLog.fling(Direction.DOWN);
153             SystemClock.sleep(100);
154             callLog.fling(Direction.UP);
155             SystemClock.sleep(100);
156         }
157     }
158 
159     // Method to insert a new contact
insertNewContacts()160     public void insertNewContacts() throws OperationApplicationException, RemoteException {
161         ArrayList<ContentProviderOperation> ops = new ArrayList<ContentProviderOperation>();
162         int rawContactID = ops.size();
163         // to insert a new raw contact in the table ContactsContract.RawContacts
164         ops.add(ContentProviderOperation.newInsert(ContactsContract.RawContacts.CONTENT_URI)
165                 .withValue(ContactsContract.RawContacts.ACCOUNT_TYPE, "Test")
166                 .withValue(RawContacts.ACCOUNT_NAME, CONTACT_NAME)
167                 .build());
168 
169         // to insert display name in the table ContactsContract.Data
170         ops.add(ContentProviderOperation.newInsert(ContactsContract.Data.CONTENT_URI)
171                 .withValueBackReference(ContactsContract.Data.RAW_CONTACT_ID, rawContactID)
172                 .withValue(ContactsContract.Data.MIMETYPE, StructuredName.CONTENT_ITEM_TYPE)
173                 .withValue(StructuredName.DISPLAY_NAME, CONTACT_NAME)
174                 .build());
175 
176         // to insert Mobile Number in the table ContactsContract.Data
177         ops.add(ContentProviderOperation.newInsert(ContactsContract.Data.CONTENT_URI)
178                 .withValueBackReference(ContactsContract.Data.RAW_CONTACT_ID, rawContactID)
179                 .withValue(ContactsContract.Data.MIMETYPE, Phone.CONTENT_ITEM_TYPE)
180                 .withValue(Phone.NUMBER, CONTACT_NUMBER)
181                 .withValue(Phone.TYPE, CommonDataKinds.Phone.TYPE_MOBILE)
182                 .build());
183 
184             // Executing all the insert operations as a single database transaction
185         getInstrumentation().getContext().getContentResolver()
186                 .applyBatch(ContactsContract.AUTHORITY, ops);
187     }
188 
189     // Checks whether certain contact exists or not
doesContactExist()190     public boolean doesContactExist() {
191         Uri uri = Uri.withAppendedPath(
192                 ContactsContract.PhoneLookup.CONTENT_FILTER_URI, Uri.encode(CONTACT_NUMBER));
193         Cursor contactLookup = getInstrumentation().getContext().getContentResolver().query(
194                 uri, new String[] {
195                         BaseColumns._ID,
196                         ContactsContract.PhoneLookup.DISPLAY_NAME },
197                         null,
198                         null,
199                         null);
200         boolean found = false;
201         try {
202             if (contactLookup != null && contactLookup.getCount() > 0) {
203                 contactLookup.moveToNext();
204                 if (contactLookup.getString(contactLookup.getColumnIndex(
205                             ContactsContract.Data.DISPLAY_NAME)).equals(CONTACT_NAME))
206                     found = true;
207             }
208         } finally {
209             if (contactLookup != null) {
210                 contactLookup.close();
211             }
212         }
213 
214         return found;
215     }
216 
217     // Inserts a new entry in the call log
addNumToCalLog(String number)218     public void addNumToCalLog(String number){
219         ContentValues values = new ContentValues();
220         values.put(CallLog.Calls.NUMBER, number);
221         values.put(CallLog.Calls.DATE, System.currentTimeMillis());
222         values.put(CallLog.Calls.DURATION, 0);
223         values.put(CallLog.Calls.TYPE, CallLog.Calls.OUTGOING_TYPE);
224         values.put(CallLog.Calls.NEW, 1);
225         values.put(CallLog.Calls.CACHED_NAME, "");
226         values.put(CallLog.Calls.CACHED_NUMBER_TYPE, 0);
227         values.put(CallLog.Calls.CACHED_NUMBER_LABEL, "");
228         getInstrumentation().getContext().getContentResolver()
229                 .insert(CallLog.Calls.CONTENT_URI, values);
230     }
231 
232     // Gets call log count
getCallLogCount()233     public int getCallLogCount() {
234        Cursor cursor = getInstrumentation().getContext().getContentResolver()
235                .query(CallLog.Calls.CONTENT_URI, null, null, null, null);
236        return cursor.getCount();
237     }
238 
239     // Generates a random phone number
getRandomPhoneNumber()240     public String getRandomPhoneNumber() {
241         Random rand = new Random();
242         int num1 = (rand.nextInt(7) + 1) * 100 + (rand.nextInt(8) * 10) + rand.nextInt(8);
243         int num2 = rand.nextInt(743);
244         int num3 = rand.nextInt(10000);
245 
246         return String.format("%03d-%03d-%04d", num1, num2, num3);
247     }
248 }
249