1 /* 2 * Copyright (C) 2008 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.mms; 18 19 import java.io.FileInputStream; 20 import java.util.ArrayList; 21 import java.util.Random; 22 23 import com.android.mms.data.Contact; 24 import com.android.mms.util.Recycler; 25 import android.provider.Telephony.Sms; 26 import android.provider.Telephony.Threads; 27 import android.provider.Telephony.Sms.Inbox; 28 29 import android.content.ContentResolver; 30 import android.content.ContentValues; 31 import android.content.Context; 32 import android.database.Cursor; 33 import android.database.sqlite.SQLiteDatabase; 34 import android.database.sqlite.SQLiteException; 35 import android.database.sqlite.SqliteWrapper; 36 import android.net.Uri; 37 import android.provider.Telephony.Sms.Conversations; 38 import android.test.AndroidTestCase; 39 import android.test.suitebuilder.annotation.LargeTest; 40 import android.util.Log; 41 42 /** 43 * Bang on the recycler and test it getting called simultaneously from two different threads 44 * NOTE: you first have to put the unix words file on the device: 45 * example: adb push ~/words /data/data/com.android.mms/files 46 * and then push a file that contains a comma separated list of numbers to send to. 47 * example: adb push ~/recipients /data/data/com.android.mms/files 48 * 49 */ 50 /** 51 * Bang on the recycler and test it getting called simultaneously from two different threads 52 * NOTE: you first have to put the unix words file on the device: 53 * example: adb push ~/words /data/data/com.android.mms/files 54 * and then push a file that contains a comma separated list of numbers to send to. 55 * example: adb push ~/recipients /data/data/com.android.mms/files 56 * 57 * To run just this test: 58 * runtest --test-class=com.android.mms.RecyclerTest mms 59 */ 60 public class RecyclerTest extends AndroidTestCase { 61 static final String TAG = "RecyclerTest"; 62 private ArrayList<String> mWords; 63 private ArrayList<String> mRecipients; 64 private int mWordCount; 65 private Random mRandom = new Random(); 66 private int mRecipientCnt; 67 private static final Uri sAllThreadsUri = 68 Threads.CONTENT_URI.buildUpon().appendQueryParameter("simple", "true").build(); 69 private static final String[] ALL_THREADS_PROJECTION = { 70 Threads._ID, Threads.DATE, Threads.MESSAGE_COUNT, Threads.RECIPIENT_IDS, 71 Threads.SNIPPET, Threads.SNIPPET_CHARSET, Threads.READ, Threads.ERROR, 72 Threads.HAS_ATTACHMENT 73 }; 74 75 @Override 76 protected void setUp() throws Exception { 77 super.setUp(); 78 Context context = getContext(); 79 80 // Read in dictionary of words 81 mWords = new ArrayList<String>(98568); // count of words in words file 82 StringBuilder sb = new StringBuilder(); 83 try { 84 Log.v(TAG, "Loading dictionary of words"); 85 FileInputStream words = context.openFileInput("words"); 86 int c; 87 while ((c = words.read()) != -1) { 88 if (c == '\r' || c == '\n') { 89 String word = sb.toString().trim(); 90 if (word.length() > 0) { 91 mWords.add(word); 92 } 93 sb.setLength(0); 94 } else { 95 sb.append((char)c); 96 } 97 } 98 words.close(); 99 mWordCount = mWords.size(); 100 Log.v(TAG, "Loaded dictionary word count: " + mWordCount); 101 } catch (Exception e) { 102 Log.e(TAG, "can't open words file at /data/data/com.android.mms/files/words"); 103 return; 104 } 105 106 // Read in list of recipients 107 mRecipients = new ArrayList<String>(); 108 try { 109 Log.v(TAG, "Loading recipients"); 110 FileInputStream recipients = context.openFileInput("recipients"); 111 int c; 112 while ((c = recipients.read()) != -1) { 113 if (c == '\r' || c == '\n' || c == ',') { 114 String recipient = sb.toString().trim(); 115 if (recipient.length() > 0) { 116 mRecipients.add(recipient); 117 } 118 sb.setLength(0); 119 } else { 120 sb.append((char)c); 121 } 122 } 123 recipients.close(); 124 Log.v(TAG, "Loaded recipients: " + mRecipients.size()); 125 } catch (Exception e) { 126 Log.e(TAG, "can't open recipients file at /data/data/com.android.mms/files/recipients"); 127 return; 128 } 129 mRecipientCnt = mRecipients.size(); 130 } 131 132 private String generateMessage() { 133 int wordsInMessage = mRandom.nextInt(9) + 1; // up to 10 words in the message 134 StringBuilder msg = new StringBuilder(); 135 for (int i = 0; i < wordsInMessage; i++) { 136 msg.append(mWords.get(mRandom.nextInt(mWordCount)) + " "); 137 } 138 return msg.toString(); 139 } 140 141 private Uri storeMessage(Context context, String address, String message) { 142 // Store the message in the content provider. 143 ContentValues values = new ContentValues(); 144 // values.put(Sms.ERROR_CODE, 0); 145 values.put(Inbox.ADDRESS, address); 146 147 // Use now for the timestamp to avoid confusion with clock 148 // drift between the handset and the SMSC. 149 values.put(Inbox.DATE, new Long(System.currentTimeMillis())); 150 values.put(Inbox.PROTOCOL, 0); 151 values.put(Inbox.READ, Integer.valueOf(0)); 152 // if (sms.getPseudoSubject().length() > 0) { 153 // values.put(Inbox.SUBJECT, sms.getPseudoSubject()); 154 // } 155 values.put(Inbox.REPLY_PATH_PRESENT, 0); 156 values.put(Inbox.SERVICE_CENTER, 0); 157 values.put(Inbox.BODY, message); 158 159 // Make sure we've got a thread id so after the insert we'll be able to delete 160 // excess messages. 161 Long threadId = 0L; 162 Contact cacheContact = Contact.get(address,true); 163 if (cacheContact != null) { 164 address = cacheContact.getNumber(); 165 } 166 167 if (((threadId == null) || (threadId == 0)) && (address != null)) { 168 values.put(Sms.THREAD_ID, Threads.getOrCreateThreadId( 169 context, address)); 170 } 171 172 ContentResolver resolver = context.getContentResolver(); 173 174 Uri insertedUri = SqliteWrapper.insert(context, resolver, Inbox.CONTENT_URI, values); 175 176 // Now make sure we're not over the limit in stored messages 177 threadId = values.getAsLong(Sms.THREAD_ID); 178 Recycler.getSmsRecycler().deleteOldMessagesByThreadId(context, threadId); 179 180 return insertedUri; 181 } 182 183 Runnable mRecyclerBang = new Runnable() { 184 public void run() { 185 final int MAXSEND = Integer.MAX_VALUE; 186 187 for (int i = 0; i < MAXSEND; i++) { 188 // Put a random message to one of the random recipients in the SMS db. 189 Uri uri = storeMessage(getContext(), 190 mRecipients.get(mRandom.nextInt(mRecipientCnt)), 191 generateMessage()); 192 Log.v(TAG, "Generating msg uri: " + uri); 193 if (i > 100) { 194 // Wait until we've sent a bunch of messages to guarantee we've got 195 // some threads built up. Then check to make sure all the threads are there 196 // on each message. All these queries will provide additional stress on the 197 // sms db. 198 Cursor cursor = null; 199 try { 200 cursor = SqliteWrapper.query(getContext(), 201 getContext().getContentResolver(), sAllThreadsUri, 202 ALL_THREADS_PROJECTION, null, null, 203 Conversations.DEFAULT_SORT_ORDER); 204 assertNotNull("Cursor from thread query is null!", cursor); 205 int cnt = cursor.getCount(); 206 assertTrue("The threads appeared to have been wiped out", 207 cursor.getCount() >= mRecipientCnt); 208 } catch (SQLiteException e) { 209 Log.v(TAG, "query for threads failed with exception: " + e); 210 fail("query for threads failed with exception: " + e); 211 } finally { 212 if (cursor != null) { 213 cursor.close(); 214 } 215 } 216 } 217 } 218 } 219 }; 220 221 Runnable mSQLMemoryReleaser = new Runnable() { 222 public void run() { 223 while (true) { 224 SQLiteDatabase.releaseMemory(); 225 try { 226 Thread.sleep(5000); 227 } catch (Exception e) { 228 229 } 230 } 231 } 232 }; 233 234 /** 235 * Send a flurry of SMS and MMS messages 236 */ 237 @LargeTest 238 public void testRecycler() throws Throwable { 239 // Start N simultaneous threads generating messages and running the recycler 240 final int THREAD_COUNT = 3; 241 ArrayList<Thread> threads = new ArrayList<Thread>(THREAD_COUNT); 242 for (int i = 0; i < THREAD_COUNT; i++) { 243 threads.add(i, new Thread(mRecyclerBang)); 244 threads.get(i).start(); 245 } 246 Thread memoryBanger = new Thread(mSQLMemoryReleaser); 247 memoryBanger.start(); 248 249 // Wait for the threads to finish 250 for (int i = 0; i < THREAD_COUNT; i++) { 251 threads.get(i).join(); 252 } 253 254 assertTrue(true); 255 } 256 } 257