1 /*
2  * Copyright (C) 2010 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 android.app;
18 
19 import android.app.DownloadManager.Query;
20 import android.app.DownloadManager.Request;
21 import android.content.BroadcastReceiver;
22 import android.content.Context;
23 import android.content.Intent;
24 import android.content.IntentFilter;
25 import android.database.Cursor;
26 import android.net.ConnectivityManager;
27 import android.net.NetworkInfo;
28 import android.net.Uri;
29 import android.net.wifi.WifiManager;
30 import android.os.Environment;
31 import android.os.ParcelFileDescriptor;
32 import android.os.ParcelFileDescriptor.AutoCloseInputStream;
33 import android.os.SystemClock;
34 import android.os.UserHandle;
35 import android.provider.Settings;
36 import android.test.InstrumentationTestCase;
37 import android.util.Log;
38 
39 import libcore.io.Streams;
40 
41 import com.google.mockwebserver.MockResponse;
42 import com.google.mockwebserver.MockWebServer;
43 
44 import java.io.DataInputStream;
45 import java.io.DataOutputStream;
46 import java.io.File;
47 import java.io.FileInputStream;
48 import java.io.FileNotFoundException;
49 import java.io.FileOutputStream;
50 import java.io.IOException;
51 import java.net.URL;
52 import java.util.ArrayList;
53 import java.util.Collections;
54 import java.util.HashSet;
55 import java.util.Random;
56 import java.util.Set;
57 import java.util.concurrent.TimeoutException;
58 
59 /**
60  * Base class for Instrumented tests for the Download Manager.
61  */
62 public class DownloadManagerBaseTest extends InstrumentationTestCase {
63     private static final String TAG = "DownloadManagerBaseTest";
64     protected DownloadManager mDownloadManager = null;
65     private MockWebServer mServer = null;
66     protected String mFileType = "text/plain";
67     protected Context mContext = null;
68     protected MultipleDownloadsCompletedReceiver mReceiver = null;
69     protected static final int DEFAULT_FILE_SIZE = 10 * 1024;  // 10kb
70     protected static final int FILE_BLOCK_READ_SIZE = 1024 * 1024;
71 
72     protected static final String LOG_TAG = "android.net.DownloadManagerBaseTest";
73     protected static final int HTTP_OK = 200;
74     protected static final int HTTP_REDIRECT = 307;
75     protected static final int HTTP_PARTIAL_CONTENT = 206;
76     protected static final int HTTP_NOT_FOUND = 404;
77     protected static final int HTTP_SERVICE_UNAVAILABLE = 503;
78     protected String DEFAULT_FILENAME = "somefile.txt";
79 
80     protected static final int DEFAULT_MAX_WAIT_TIME = 2 * 60 * 1000;  // 2 minutes
81     protected static final int DEFAULT_WAIT_POLL_TIME = 5 * 1000;  // 5 seconds
82 
83     protected static final int WAIT_FOR_DOWNLOAD_POLL_TIME = 1 * 1000;  // 1 second
84     protected static final int MAX_WAIT_FOR_DOWNLOAD_TIME = 30 * 1000; // 30 seconds
85 
86     // Just a few popular file types used to return from a download
87     protected enum DownloadFileType {
88         PLAINTEXT,
89         APK,
90         GIF,
91         GARBAGE,
92         UNRECOGNIZED,
93         ZIP
94     }
95 
96     protected enum DataType {
97         TEXT,
98         BINARY
99     }
100 
101     public static class LoggingRng extends Random {
102 
103         /**
104          * Constructor
105          *
106          * Creates RNG with self-generated seed value.
107          */
LoggingRng()108         public LoggingRng() {
109             this(SystemClock.uptimeMillis());
110         }
111 
112         /**
113          * Constructor
114          *
115          * Creats RNG with given initial seed value
116 
117          * @param seed The initial seed value
118          */
LoggingRng(long seed)119         public LoggingRng(long seed) {
120             super(seed);
121             Log.i(LOG_TAG, "Seeding RNG with value: " + seed);
122         }
123     }
124 
125     public static class MultipleDownloadsCompletedReceiver extends BroadcastReceiver {
126         private volatile int mNumDownloadsCompleted = 0;
127         private Set<Long> downloadIds = Collections.synchronizedSet(new HashSet<Long>());
128 
129         /**
130          * {@inheritDoc}
131          */
132         @Override
onReceive(Context context, Intent intent)133         public void onReceive(Context context, Intent intent) {
134             if (intent.getAction().equalsIgnoreCase(DownloadManager.ACTION_DOWNLOAD_COMPLETE)) {
135                 synchronized(this) {
136                     long id = intent.getExtras().getLong(DownloadManager.EXTRA_DOWNLOAD_ID);
137                     Log.i(LOG_TAG, "Received Notification for download: " + id);
138                     if (!downloadIds.contains(id)) {
139                         ++mNumDownloadsCompleted;
140                         Log.i(LOG_TAG, "MultipleDownloadsCompletedReceiver got intent: " +
141                                 intent.getAction() + " --> total count: " + mNumDownloadsCompleted);
142                         downloadIds.add(id);
143 
144                         DownloadManager dm = (DownloadManager)context.getSystemService(
145                                 Context.DOWNLOAD_SERVICE);
146 
147                         Cursor cursor = dm.query(new Query().setFilterById(id));
148                         try {
149                             if (cursor.moveToFirst()) {
150                                 int status = cursor.getInt(cursor.getColumnIndex(
151                                         DownloadManager.COLUMN_STATUS));
152                                 Log.i(LOG_TAG, "Download status is: " + status);
153                             } else {
154                                 fail("No status found for completed download!");
155                             }
156                         } finally {
157                             cursor.close();
158                         }
159                     } else {
160                         Log.i(LOG_TAG, "Notification for id: " + id + " has already been made.");
161                     }
162                 }
163             }
164         }
165 
166         /**
167          * Gets the number of times the {@link #onReceive} callback has been called for the
168          * {@link DownloadManager.ACTION_DOWNLOAD_COMPLETED} action, indicating the number of
169          * downloads completed thus far.
170          *
171          * @return the number of downloads completed so far.
172          */
numDownloadsCompleted()173         public int numDownloadsCompleted() {
174             return mNumDownloadsCompleted;
175         }
176 
177         /**
178          * Gets the list of download IDs.
179          * @return A Set<Long> with the ids of the completed downloads.
180          */
getDownloadIds()181         public Set<Long> getDownloadIds() {
182             synchronized(this) {
183                 Set<Long> returnIds = new HashSet<Long>(downloadIds);
184                 return returnIds;
185             }
186         }
187 
188     }
189 
190     public static class WiFiChangedReceiver extends BroadcastReceiver {
191         private Context mContext = null;
192 
193         /**
194          * Constructor
195          *
196          * Sets the current state of WiFi.
197          *
198          * @param context The current app {@link Context}.
199          */
WiFiChangedReceiver(Context context)200         public WiFiChangedReceiver(Context context) {
201             mContext = context;
202         }
203 
204         /**
205          * {@inheritDoc}
206          */
207         @Override
onReceive(Context context, Intent intent)208         public void onReceive(Context context, Intent intent) {
209             if (intent.getAction().equalsIgnoreCase(ConnectivityManager.CONNECTIVITY_ACTION)) {
210                 Log.i(LOG_TAG, "ConnectivityManager state change: " + intent.getAction());
211                 synchronized (this) {
212                     this.notify();
213                 }
214             }
215         }
216 
217         /**
218          * Gets the current state of WiFi.
219          *
220          * @return Returns true if WiFi is on, false otherwise.
221          */
getWiFiIsOn()222         public boolean getWiFiIsOn() {
223             ConnectivityManager connManager = (ConnectivityManager)mContext.getSystemService(
224                     Context.CONNECTIVITY_SERVICE);
225             NetworkInfo info = connManager.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
226             Log.i(LOG_TAG, "WiFi Connection state is currently: " + info.isConnected());
227             return info.isConnected();
228         }
229     }
230 
231     /**
232      * {@inheritDoc}
233      */
234     @Override
setUp()235     public void setUp() throws Exception {
236         mContext = getInstrumentation().getContext();
237         mDownloadManager = (DownloadManager)mContext.getSystemService(Context.DOWNLOAD_SERVICE);
238         mServer = new MockWebServer();
239         mServer.play();
240         mReceiver = registerNewMultipleDownloadsReceiver();
241         // Note: callers overriding this should call mServer.play() with the desired port #
242     }
243 
244     @Override
tearDown()245     public void tearDown() throws Exception {
246         mServer.shutdown();
247         super.tearDown();
248     }
249 
250     /**
251      * Helper to build a response from the MockWebServer with no body.
252      *
253      * @param status The HTTP status code to return for this response
254      * @return Returns the mock web server response that was queued (which can be modified)
255      */
buildResponse(int status)256     protected MockResponse buildResponse(int status) {
257         MockResponse response = new MockResponse().setResponseCode(status);
258         response.setHeader("Content-type", mFileType);
259         return response;
260     }
261 
262     /**
263      * Helper to build a response from the MockWebServer.
264      *
265      * @param status The HTTP status code to return for this response
266      * @param body The body to return in this response
267      * @return Returns the mock web server response that was queued (which can be modified)
268      */
buildResponse(int status, byte[] body)269     protected MockResponse buildResponse(int status, byte[] body) {
270         return buildResponse(status).setBody(body);
271     }
272 
273     /**
274      * Helper to build a response from the MockWebServer.
275      *
276      * @param status The HTTP status code to return for this response
277      * @param bodyFile The body to return in this response
278      * @return Returns the mock web server response that was queued (which can be modified)
279      */
buildResponse(int status, File bodyFile)280     protected MockResponse buildResponse(int status, File bodyFile)
281             throws FileNotFoundException, IOException {
282         final byte[] body = Streams.readFully(new FileInputStream(bodyFile));
283         return buildResponse(status).setBody(body);
284     }
285 
enqueueResponse(MockResponse resp)286     protected void enqueueResponse(MockResponse resp) {
287         mServer.enqueue(resp);
288     }
289 
290     /**
291      * Helper to generate a random blob of bytes.
292      *
293      * @param size The size of the data to generate
294      * @param type The type of data to generate: currently, one of {@link DataType#TEXT} or
295      *         {@link DataType#BINARY}.
296      * @return The random data that is generated.
297      */
generateData(int size, DataType type)298     protected byte[] generateData(int size, DataType type) {
299         return generateData(size, type, null);
300     }
301 
302     /**
303      * Helper to generate a random blob of bytes using a given RNG.
304      *
305      * @param size The size of the data to generate
306      * @param type The type of data to generate: currently, one of {@link DataType#TEXT} or
307      *         {@link DataType#BINARY}.
308      * @param rng (optional) The RNG to use; pass null to use
309      * @return The random data that is generated.
310      */
generateData(int size, DataType type, Random rng)311     protected byte[] generateData(int size, DataType type, Random rng) {
312         int min = Byte.MIN_VALUE;
313         int max = Byte.MAX_VALUE;
314 
315         // Only use chars in the HTTP ASCII printable character range for Text
316         if (type == DataType.TEXT) {
317             min = 32;
318             max = 126;
319         }
320         byte[] result = new byte[size];
321         Log.i(LOG_TAG, "Generating data of size: " + size);
322 
323         if (rng == null) {
324             rng = new LoggingRng();
325         }
326 
327         for (int i = 0; i < size; ++i) {
328             result[i] = (byte) (min + rng.nextInt(max - min + 1));
329         }
330         return result;
331     }
332 
333     /**
334      * Helper to verify the size of a file.
335      *
336      * @param pfd The input file to compare the size of
337      * @param size The expected size of the file
338      */
verifyFileSize(ParcelFileDescriptor pfd, long size)339     protected void verifyFileSize(ParcelFileDescriptor pfd, long size) {
340         assertEquals(pfd.getStatSize(), size);
341     }
342 
343     /**
344      * Helper to verify the contents of a downloaded file versus a byte[].
345      *
346      * @param actual The file of whose contents to verify
347      * @param expected The data we expect to find in the aforementioned file
348      * @throws IOException if there was a problem reading from the file
349      */
verifyFileContents(ParcelFileDescriptor actual, byte[] expected)350     protected void verifyFileContents(ParcelFileDescriptor actual, byte[] expected)
351             throws IOException {
352         AutoCloseInputStream input = new ParcelFileDescriptor.AutoCloseInputStream(actual);
353         long fileSize = actual.getStatSize();
354 
355         assertTrue(fileSize <= Integer.MAX_VALUE);
356         assertEquals(expected.length, fileSize);
357 
358         byte[] actualData = new byte[expected.length];
359         assertEquals(input.read(actualData), fileSize);
360         compareByteArrays(actualData, expected);
361     }
362 
363     /**
364      * Helper to compare 2 byte arrays.
365      *
366      * @param actual The array whose data we want to verify
367      * @param expected The array of data we expect to see
368      */
compareByteArrays(byte[] actual, byte[] expected)369     protected void compareByteArrays(byte[] actual, byte[] expected) {
370         assertEquals(actual.length, expected.length);
371         int length = actual.length;
372         for (int i = 0; i < length; ++i) {
373             // assert has a bit of overhead, so only do the assert when the values are not the same
374             if (actual[i] != expected[i]) {
375                 fail("Byte arrays are not equal.");
376             }
377         }
378     }
379 
380     /**
381      * Verifies the contents of a downloaded file versus the contents of a File.
382      *
383      * @param pfd The file whose data we want to verify
384      * @param file The file containing the data we expect to see in the aforementioned file
385      * @throws IOException If there was a problem reading either of the two files
386      */
verifyFileContents(ParcelFileDescriptor pfd, File file)387     protected void verifyFileContents(ParcelFileDescriptor pfd, File file) throws IOException {
388         byte[] actual = new byte[FILE_BLOCK_READ_SIZE];
389         byte[] expected = new byte[FILE_BLOCK_READ_SIZE];
390 
391         AutoCloseInputStream input = new ParcelFileDescriptor.AutoCloseInputStream(pfd);
392 
393         assertEquals(file.length(), pfd.getStatSize());
394 
395         DataInputStream inFile = new DataInputStream(new FileInputStream(file));
396         int actualRead = 0;
397         int expectedRead = 0;
398 
399         while (((actualRead = input.read(actual)) != -1) &&
400                 ((expectedRead = inFile.read(expected)) != -1)) {
401             assertEquals(actualRead, expectedRead);
402             compareByteArrays(actual, expected);
403         }
404     }
405 
406     /**
407      * Sets the MIME type of file that will be served from the mock server
408      *
409      * @param type The MIME type to return from the server
410      */
setServerMimeType(DownloadFileType type)411     protected void setServerMimeType(DownloadFileType type) {
412         mFileType = getMimeMapping(type);
413     }
414 
415     /**
416      * Gets the MIME content string for a given type
417      *
418      * @param type The MIME type to return
419      * @return the String representation of that MIME content type
420      */
getMimeMapping(DownloadFileType type)421     protected String getMimeMapping(DownloadFileType type) {
422         switch (type) {
423             case APK:
424                 return "application/vnd.android.package-archive";
425             case GIF:
426                 return "image/gif";
427             case ZIP:
428                 return "application/x-zip-compressed";
429             case GARBAGE:
430                 return "zip\\pidy/doo/da";
431             case UNRECOGNIZED:
432                 return "application/new.undefined.type.of.app";
433         }
434         return "text/plain";
435     }
436 
437     /**
438      * Gets the Uri that should be used to access the mock server
439      *
440      * @param filename The name of the file to try to retrieve from the mock server
441      * @return the Uri to use for access the file on the mock server
442      */
getServerUri(String filename)443     protected Uri getServerUri(String filename) throws Exception {
444         URL url = mServer.getUrl("/" + filename);
445         return Uri.parse(url.toString());
446     }
447 
448    /**
449     * Gets the Uri that should be used to access the mock server
450     *
451     * @param filename The name of the file to try to retrieve from the mock server
452     * @return the Uri to use for access the file on the mock server
453     */
logDBColumnData(Cursor cursor, String column)454     protected void logDBColumnData(Cursor cursor, String column) {
455         int index = cursor.getColumnIndex(column);
456         Log.i(LOG_TAG, "columnName: " + column);
457         Log.i(LOG_TAG, "columnValue: " + cursor.getString(index));
458     }
459 
460     /**
461      * Helper to create and register a new MultipleDownloadCompletedReciever
462      *
463      * This is used to track many simultaneous downloads by keeping count of all the downloads
464      * that have completed.
465      *
466      * @return A new receiver that records and can be queried on how many downloads have completed.
467      */
registerNewMultipleDownloadsReceiver()468     protected MultipleDownloadsCompletedReceiver registerNewMultipleDownloadsReceiver() {
469         MultipleDownloadsCompletedReceiver receiver = new MultipleDownloadsCompletedReceiver();
470         mContext.registerReceiver(receiver, new IntentFilter(
471                 DownloadManager.ACTION_DOWNLOAD_COMPLETE));
472         return receiver;
473     }
474 
475     /**
476      * Helper to verify a standard single-file download from the mock server, and clean up after
477      * verification
478      *
479      * Note that this also calls the Download manager's remove, which cleans up the file from cache.
480      *
481      * @param requestId The id of the download to remove
482      * @param fileData The data to verify the file contains
483      */
verifyAndCleanupSingleFileDownload(long requestId, byte[] fileData)484     protected void verifyAndCleanupSingleFileDownload(long requestId, byte[] fileData)
485             throws Exception {
486         int fileSize = fileData.length;
487         ParcelFileDescriptor pfd = mDownloadManager.openDownloadedFile(requestId);
488         Cursor cursor = mDownloadManager.query(new Query().setFilterById(requestId));
489 
490         try {
491             assertEquals(1, cursor.getCount());
492             assertTrue(cursor.moveToFirst());
493 
494             verifyFileSize(pfd, fileSize);
495             verifyFileContents(pfd, fileData);
496         } finally {
497             pfd.close();
498             cursor.close();
499             mDownloadManager.remove(requestId);
500         }
501     }
502 
503     /**
504      * Enables or disables WiFi.
505      *
506      * Note: Needs the following permissions:
507      *  android.permission.ACCESS_WIFI_STATE
508      *  android.permission.CHANGE_WIFI_STATE
509      * @param enable true if it should be enabled, false if it should be disabled
510      */
setWiFiStateOn(boolean enable)511     protected void setWiFiStateOn(boolean enable) throws Exception {
512         Log.i(LOG_TAG, "Setting WiFi State to: " + enable);
513         WifiManager manager = (WifiManager)mContext.getSystemService(Context.WIFI_SERVICE);
514 
515         manager.setWifiEnabled(enable);
516 
517         String timeoutMessage = "Timed out waiting for Wifi to be "
518             + (enable ? "enabled!" : "disabled!");
519 
520         WiFiChangedReceiver receiver = new WiFiChangedReceiver(mContext);
521         mContext.registerReceiver(receiver, new IntentFilter(
522                 ConnectivityManager.CONNECTIVITY_ACTION));
523 
524         synchronized (receiver) {
525             long timeoutTime = SystemClock.elapsedRealtime() + DEFAULT_MAX_WAIT_TIME;
526             boolean timedOut = false;
527 
528             while (receiver.getWiFiIsOn() != enable && !timedOut) {
529                 try {
530                     receiver.wait(DEFAULT_WAIT_POLL_TIME);
531 
532                     if (SystemClock.elapsedRealtime() > timeoutTime) {
533                         timedOut = true;
534                     }
535                 }
536                 catch (InterruptedException e) {
537                     // ignore InterruptedExceptions
538                 }
539             }
540             if (timedOut) {
541                 fail(timeoutMessage);
542             }
543         }
544         assertEquals(enable, receiver.getWiFiIsOn());
545     }
546 
547     /**
548      * Helper to enables or disables airplane mode. If successful, it also broadcasts an intent
549      * indicating that the mode has changed.
550      *
551      * Note: Needs the following permission:
552      *  android.permission.WRITE_SETTINGS
553      * @param enable true if airplane mode should be ON, false if it should be OFF
554      */
setAirplaneModeOn(boolean enable)555     protected void setAirplaneModeOn(boolean enable) throws Exception {
556         int state = enable ? 1 : 0;
557 
558         // Change the system setting
559         Settings.Global.putInt(mContext.getContentResolver(), Settings.Global.AIRPLANE_MODE_ON,
560                 state);
561 
562         String timeoutMessage = "Timed out waiting for airplane mode to be " +
563                 (enable ? "enabled!" : "disabled!");
564 
565         // wait for airplane mode to change state
566         int currentWaitTime = 0;
567         while (Settings.Global.getInt(mContext.getContentResolver(),
568                 Settings.Global.AIRPLANE_MODE_ON, -1) != state) {
569             timeoutWait(currentWaitTime, DEFAULT_WAIT_POLL_TIME, DEFAULT_MAX_WAIT_TIME,
570                     timeoutMessage);
571         }
572 
573         // Post the intent
574         Intent intent = new Intent(Intent.ACTION_AIRPLANE_MODE_CHANGED);
575         intent.putExtra("state", true);
576         mContext.sendBroadcastAsUser(intent, UserHandle.ALL);
577     }
578 
579     /**
580      * Helper to create a large file of random data on the SD card.
581      *
582      * @param filename (optional) The name of the file to create on the SD card; pass in null to
583      *          use a default temp filename.
584      * @param type The type of file to create
585      * @param subdirectory If not null, the subdirectory under the SD card where the file should go
586      * @return The File that was created
587      * @throws IOException if there was an error while creating the file.
588      */
createFileOnSD(String filename, long fileSize, DataType type, String subdirectory)589     protected File createFileOnSD(String filename, long fileSize, DataType type,
590             String subdirectory) throws IOException {
591 
592         // Build up the file path and name
593         String sdPath = Environment.getExternalStorageDirectory().getPath();
594         StringBuilder fullPath = new StringBuilder(sdPath);
595         if (subdirectory != null) {
596             fullPath.append(File.separatorChar).append(subdirectory);
597         }
598 
599         File file = null;
600         if (filename == null) {
601             file = File.createTempFile("DMTEST_", null, new File(fullPath.toString()));
602         }
603         else {
604             fullPath.append(File.separatorChar).append(filename);
605             file = new File(fullPath.toString());
606             file.createNewFile();
607         }
608 
609         // Fill the file with random data
610         DataOutputStream output = new DataOutputStream(new FileOutputStream(file));
611         final int CHUNK_SIZE = 1000000;  // copy random data in 1000000-char chunks
612         long remaining = fileSize;
613         int nextChunkSize = CHUNK_SIZE;
614         byte[] randomData = null;
615         Random rng = new LoggingRng();
616         byte[] chunkSizeData = generateData(nextChunkSize, type, rng);
617 
618         try {
619             while (remaining > 0) {
620                 if (remaining < CHUNK_SIZE) {
621                     nextChunkSize = (int)remaining;
622                     remaining = 0;
623                     randomData = generateData(nextChunkSize, type, rng);
624                 }
625                 else {
626                     remaining -= CHUNK_SIZE;
627                     randomData = chunkSizeData;
628                 }
629                 output.write(randomData);
630                 Log.i(TAG, "while creating " + fileSize + " file, " +
631                         "remaining bytes to be written: " + remaining);
632             }
633         } catch (IOException e) {
634             Log.e(LOG_TAG, "Error writing to file " + file.getAbsolutePath());
635             file.delete();
636             throw e;
637         } finally {
638             output.close();
639         }
640         return file;
641     }
642 
643     /**
644      * Helper to wait for a particular download to finish, or else a timeout to occur
645      *
646      * Does not wait for a receiver notification of the download.
647      *
648      * @param id The download id to query on (wait for)
649      */
waitForDownloadOrTimeout_skipNotification(long id)650     protected void waitForDownloadOrTimeout_skipNotification(long id) throws TimeoutException,
651             InterruptedException {
652         waitForDownloadOrTimeout(id, WAIT_FOR_DOWNLOAD_POLL_TIME, MAX_WAIT_FOR_DOWNLOAD_TIME);
653     }
654 
655     /**
656      * Helper to wait for a particular download to finish, or else a timeout to occur
657      *
658      * Also guarantees a notification has been posted for the download.
659      *
660      * @param id The download id to query on (wait for)
661      */
waitForDownloadOrTimeout(long id)662     protected void waitForDownloadOrTimeout(long id) throws TimeoutException,
663             InterruptedException {
664         waitForDownloadOrTimeout_skipNotification(id);
665         waitForReceiverNotifications(1);
666     }
667 
668     /**
669      * Helper to wait for a particular download to finish, or else a timeout to occur
670      *
671      * Also guarantees a notification has been posted for the download.
672      *
673      * @param id The download id to query on (wait for)
674      * @param poll The amount of time to wait
675      * @param timeoutMillis The max time (in ms) to wait for the download(s) to complete
676      */
waitForDownloadOrTimeout(long id, long poll, long timeoutMillis)677     protected void waitForDownloadOrTimeout(long id, long poll, long timeoutMillis)
678             throws TimeoutException, InterruptedException {
679         doWaitForDownloadsOrTimeout(new Query().setFilterById(id), poll, timeoutMillis);
680         waitForReceiverNotifications(1);
681     }
682 
683     /**
684      * Helper to wait for all downloads to finish, or else a specified timeout to occur
685      *
686      * Makes no guaranee that notifications have been posted for all downloads.
687      *
688      * @param poll The amount of time to wait
689      * @param timeoutMillis The max time (in ms) to wait for the download(s) to complete
690      */
waitForDownloadsOrTimeout(long poll, long timeoutMillis)691     protected void waitForDownloadsOrTimeout(long poll, long timeoutMillis) throws TimeoutException,
692             InterruptedException {
693         doWaitForDownloadsOrTimeout(new Query(), poll, timeoutMillis);
694     }
695 
696     /**
697      * Helper to wait for all downloads to finish, or else a timeout to occur, but does not throw
698      *
699      * Also guarantees a notification has been posted for the download.
700      *
701      * @param id The id of the download to query against
702      * @param poll The amount of time to wait
703      * @param timeoutMillis The max time (in ms) to wait for the download(s) to complete
704      * @return true if download completed successfully (didn't timeout), false otherwise
705      */
waitForDownloadOrTimeoutNoThrow(long id, long poll, long timeoutMillis)706     protected boolean waitForDownloadOrTimeoutNoThrow(long id, long poll, long timeoutMillis) {
707         try {
708             doWaitForDownloadsOrTimeout(new Query().setFilterById(id), poll, timeoutMillis);
709             waitForReceiverNotifications(1);
710         } catch (TimeoutException e) {
711             return false;
712         }
713         return true;
714     }
715 
716     /**
717      * Helper function to synchronously wait, or timeout if the maximum threshold has been exceeded.
718      *
719      * @param currentTotalWaitTime The total time waited so far
720      * @param poll The amount of time to wait
721      * @param maxTimeoutMillis The total wait time threshold; if we've waited more than this long,
722      *          we timeout and fail
723      * @param timedOutMessage The message to display in the failure message if we timeout
724      * @return The new total amount of time we've waited so far
725      * @throws TimeoutException if timed out waiting for SD card to mount
726      */
timeoutWait(int currentTotalWaitTime, long poll, long maxTimeoutMillis, String timedOutMessage)727     protected int timeoutWait(int currentTotalWaitTime, long poll, long maxTimeoutMillis,
728             String timedOutMessage) throws TimeoutException {
729         long now = SystemClock.elapsedRealtime();
730         long end = now + poll;
731 
732         // if we get InterruptedException's, ignore them and just keep sleeping
733         while (now < end) {
734             try {
735                 Thread.sleep(end - now);
736             } catch (InterruptedException e) {
737                 // ignore interrupted exceptions
738             }
739             now = SystemClock.elapsedRealtime();
740         }
741 
742         currentTotalWaitTime += poll;
743         if (currentTotalWaitTime > maxTimeoutMillis) {
744             throw new TimeoutException(timedOutMessage);
745         }
746         return currentTotalWaitTime;
747     }
748 
749     /**
750      * Helper to wait for all downloads to finish, or else a timeout to occur
751      *
752      * @param query The query to pass to the download manager
753      * @param poll The poll time to wait between checks
754      * @param timeoutMillis The max amount of time (in ms) to wait for the download(s) to complete
755      */
doWaitForDownloadsOrTimeout(Query query, long poll, long timeoutMillis)756     protected void doWaitForDownloadsOrTimeout(Query query, long poll, long timeoutMillis)
757             throws TimeoutException {
758         int currentWaitTime = 0;
759         while (true) {
760             query.setFilterByStatus(DownloadManager.STATUS_PENDING | DownloadManager.STATUS_PAUSED
761                     | DownloadManager.STATUS_RUNNING);
762             Cursor cursor = mDownloadManager.query(query);
763 
764             try {
765                 if (cursor.getCount() == 0) {
766                     Log.i(LOG_TAG, "All downloads should be done...");
767                     break;
768                 }
769                 currentWaitTime = timeoutWait(currentWaitTime, poll, timeoutMillis,
770                         "Timed out waiting for all downloads to finish");
771             } finally {
772                 cursor.close();
773             }
774         }
775     }
776 
777     /**
778      * Synchronously waits for external store to be mounted (eg: SD Card).
779      *
780      * @throws InterruptedException if interrupted
781      * @throws Exception if timed out waiting for SD card to mount
782      */
waitForExternalStoreMount()783     protected void waitForExternalStoreMount() throws Exception {
784         String extStorageState = Environment.getExternalStorageState();
785         int currentWaitTime = 0;
786         while (!extStorageState.equals(Environment.MEDIA_MOUNTED)) {
787             Log.i(LOG_TAG, "Waiting for SD card...");
788             currentWaitTime = timeoutWait(currentWaitTime, DEFAULT_WAIT_POLL_TIME,
789                     DEFAULT_MAX_WAIT_TIME, "Timed out waiting for SD Card to be ready!");
790             extStorageState = Environment.getExternalStorageState();
791         }
792     }
793 
794     /**
795      * Synchronously waits for a download to start.
796      *
797      * @param dlRequest the download request id used by Download Manager to track the download.
798      * @throws Exception if timed out while waiting for SD card to mount
799      */
waitForDownloadToStart(long dlRequest)800     protected void waitForDownloadToStart(long dlRequest) throws Exception {
801         Cursor cursor = getCursor(dlRequest);
802         try {
803             int columnIndex = cursor.getColumnIndex(DownloadManager.COLUMN_STATUS);
804             int value = cursor.getInt(columnIndex);
805             int currentWaitTime = 0;
806 
807             while (value != DownloadManager.STATUS_RUNNING &&
808                     (value != DownloadManager.STATUS_FAILED) &&
809                     (value != DownloadManager.STATUS_SUCCESSFUL)) {
810                 Log.i(LOG_TAG, "Waiting for download to start...");
811                 currentWaitTime = timeoutWait(currentWaitTime, WAIT_FOR_DOWNLOAD_POLL_TIME,
812                         MAX_WAIT_FOR_DOWNLOAD_TIME, "Timed out waiting for download to start!");
813                 cursor.requery();
814                 assertTrue(cursor.moveToFirst());
815                 columnIndex = cursor.getColumnIndex(DownloadManager.COLUMN_STATUS);
816                 value = cursor.getInt(columnIndex);
817             }
818             assertFalse("Download failed immediately after start",
819                     value == DownloadManager.STATUS_FAILED);
820         } finally {
821             cursor.close();
822         }
823     }
824 
825     /**
826      * Convenience function to wait for just 1 notification of a download.
827      *
828      * @throws Exception if timed out while waiting
829      */
waitForReceiverNotification()830     protected void waitForReceiverNotification() throws Exception {
831         waitForReceiverNotifications(1);
832     }
833 
834     /**
835      * Synchronously waits for our receiver to receive notification for a given number of
836      * downloads.
837      *
838      * @param targetNumber The number of notifications for unique downloads to wait for; pass in
839      *         -1 to not wait for notification.
840      * @throws Exception if timed out while waiting
841      */
waitForReceiverNotifications(int targetNumber)842     protected void waitForReceiverNotifications(int targetNumber) throws TimeoutException {
843         int count = mReceiver.numDownloadsCompleted();
844         int currentWaitTime = 0;
845 
846         while (count < targetNumber) {
847             Log.i(LOG_TAG, "Waiting for notification of downloads...");
848             currentWaitTime = timeoutWait(currentWaitTime, WAIT_FOR_DOWNLOAD_POLL_TIME,
849                     MAX_WAIT_FOR_DOWNLOAD_TIME, "Timed out waiting for download notifications!"
850                     + " Received " + count + "notifications.");
851             count = mReceiver.numDownloadsCompleted();
852         }
853     }
854 
855     /**
856      * Synchronously waits for a file to increase in size (such as to monitor that a download is
857      * progressing).
858      *
859      * @param file The file whose size to track.
860      * @throws Exception if timed out while waiting for the file to grow in size.
861      */
waitForFileToGrow(File file)862     protected void waitForFileToGrow(File file) throws Exception {
863         int currentWaitTime = 0;
864 
865         // File may not even exist yet, so wait until it does (or we timeout)
866         while (!file.exists()) {
867             Log.i(LOG_TAG, "Waiting for file to exist...");
868             currentWaitTime = timeoutWait(currentWaitTime, WAIT_FOR_DOWNLOAD_POLL_TIME,
869                     MAX_WAIT_FOR_DOWNLOAD_TIME, "Timed out waiting for file to be created.");
870         }
871 
872         // Get original file size...
873         long originalSize = file.length();
874 
875         while (file.length() <= originalSize) {
876             Log.i(LOG_TAG, "Waiting for file to be written to...");
877             currentWaitTime = timeoutWait(currentWaitTime, WAIT_FOR_DOWNLOAD_POLL_TIME,
878                     MAX_WAIT_FOR_DOWNLOAD_TIME, "Timed out waiting for file to be written to.");
879         }
880     }
881 
882     /**
883      * Helper to remove all downloads that are registered with the DL Manager.
884      *
885      * Note: This gives us a clean slate b/c it includes downloads that are pending, running,
886      * paused, or have completed.
887      */
removeAllCurrentDownloads()888     protected void removeAllCurrentDownloads() {
889         Log.i(LOG_TAG, "Removing all current registered downloads...");
890         ArrayList<Long> ids = new ArrayList<Long>();
891         Cursor cursor = mDownloadManager.query(new Query());
892         try {
893             if (cursor.moveToFirst()) {
894                 do {
895                     int index = cursor.getColumnIndex(DownloadManager.COLUMN_ID);
896                     long downloadId = cursor.getLong(index);
897                     ids.add(downloadId);
898                 } while (cursor.moveToNext());
899             }
900         } finally {
901             cursor.close();
902         }
903         // delete all ids
904         for (long id : ids) {
905             mDownloadManager.remove(id);
906         }
907         // make sure the database is empty
908         cursor = mDownloadManager.query(new Query());
909         try {
910             assertEquals(0, cursor.getCount());
911         } finally {
912             cursor.close();
913         }
914     }
915 
916     /**
917      * Helper to perform a standard enqueue of data to the mock server.
918      * download is performed to the downloads cache dir (NOT systemcache dir)
919      *
920      * @param body The body to return in the response from the server
921      */
doStandardEnqueue(byte[] body)922     protected long doStandardEnqueue(byte[] body) throws Exception {
923         return enqueueDownloadRequest(body);
924     }
925 
enqueueDownloadRequest(byte[] body)926     protected long enqueueDownloadRequest(byte[] body) throws Exception {
927         // Prepare the mock server with a standard response
928         mServer.enqueue(buildResponse(HTTP_OK, body));
929         return doEnqueue();
930     }
931 
932     /**
933      * Helper to perform a standard enqueue of data to the mock server.
934      *
935      * @param body The body to return in the response from the server, contained in the file
936      */
doStandardEnqueue(File body)937     protected long doStandardEnqueue(File body) throws Exception {
938         return enqueueDownloadRequest(body);
939     }
940 
enqueueDownloadRequest(File body)941     protected long enqueueDownloadRequest(File body) throws Exception {
942         // Prepare the mock server with a standard response
943         mServer.enqueue(buildResponse(HTTP_OK, body));
944         return doEnqueue();
945     }
946 
947     /**
948      * Helper to do the additional steps (setting title and Uri of default filename) when
949      * doing a standard enqueue request to the server.
950      */
doCommonStandardEnqueue()951     protected long doCommonStandardEnqueue() throws Exception {
952         return doEnqueue();
953     }
954 
doEnqueue()955     private long doEnqueue() throws Exception {
956         Uri uri = getServerUri(DEFAULT_FILENAME);
957         Request request = new Request(uri).setTitle(DEFAULT_FILENAME);
958         return mDownloadManager.enqueue(request);
959     }
960 
961     /**
962      * Helper to verify an int value in a Cursor
963      *
964      * @param cursor The cursor containing the query results
965      * @param columnName The name of the column to query
966      * @param expected The expected int value
967      */
verifyInt(Cursor cursor, String columnName, int expected)968     protected void verifyInt(Cursor cursor, String columnName, int expected) {
969         int index = cursor.getColumnIndex(columnName);
970         int actual = cursor.getInt(index);
971         assertEquals(String.format("Expected = %d : Actual = %d", expected, actual), expected, actual);
972     }
973 
974     /**
975      * Helper to verify a String value in a Cursor
976      *
977      * @param cursor The cursor containing the query results
978      * @param columnName The name of the column to query
979      * @param expected The expected String value
980      */
verifyString(Cursor cursor, String columnName, String expected)981     protected void verifyString(Cursor cursor, String columnName, String expected) {
982         int index = cursor.getColumnIndex(columnName);
983         String actual = cursor.getString(index);
984         Log.i(LOG_TAG, ": " + actual);
985         assertEquals(expected, actual);
986     }
987 
988     /**
989      * Performs a query based on ID and returns a Cursor for the query.
990      *
991      * @param id The id of the download in DL Manager; pass -1 to query all downloads
992      * @return A cursor for the query results
993      */
getCursor(long id)994     protected Cursor getCursor(long id) throws Exception {
995         Query query = new Query();
996         if (id != -1) {
997             query.setFilterById(id);
998         }
999 
1000         Cursor cursor = mDownloadManager.query(query);
1001         int currentWaitTime = 0;
1002 
1003         try {
1004             while (!cursor.moveToFirst()) {
1005                 Thread.sleep(DEFAULT_WAIT_POLL_TIME);
1006                 currentWaitTime += DEFAULT_WAIT_POLL_TIME;
1007                 if (currentWaitTime > DEFAULT_MAX_WAIT_TIME) {
1008                     fail("timed out waiting for a non-null query result");
1009                 }
1010                 cursor.requery();
1011             }
1012         } catch (Exception e) {
1013             cursor.close();
1014             throw e;
1015         }
1016         return cursor;
1017     }
1018 
1019     /**
1020      * Helper that does the actual basic download verification.
1021      */
doBasicDownload(byte[] blobData)1022     protected long doBasicDownload(byte[] blobData) throws Exception {
1023         long dlRequest = enqueueDownloadRequest(blobData);
1024 
1025         // wait for the download to complete
1026         waitForDownloadOrTimeout(dlRequest);
1027 
1028         assertEquals(1, mReceiver.numDownloadsCompleted());
1029         return dlRequest;
1030     }
1031 }
1032