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.annotation.Nullable; 20 import android.annotation.SdkConstant; 21 import android.annotation.SystemApi; 22 import android.annotation.SystemService; 23 import android.annotation.SdkConstant.SdkConstantType; 24 import android.content.ContentResolver; 25 import android.content.ContentUris; 26 import android.content.ContentValues; 27 import android.content.Context; 28 import android.content.Intent; 29 import android.database.Cursor; 30 import android.database.CursorWrapper; 31 import android.net.ConnectivityManager; 32 import android.net.NetworkPolicyManager; 33 import android.net.Uri; 34 import android.os.Build; 35 import android.os.Environment; 36 import android.os.FileUtils; 37 import android.os.ParcelFileDescriptor; 38 import android.provider.Downloads; 39 import android.provider.Settings; 40 import android.provider.MediaStore.Images; 41 import android.provider.Settings.SettingNotFoundException; 42 import android.text.TextUtils; 43 import android.util.Pair; 44 45 import java.io.File; 46 import java.io.FileNotFoundException; 47 import java.util.ArrayList; 48 import java.util.List; 49 50 /** 51 * The download manager is a system service that handles long-running HTTP downloads. Clients may 52 * request that a URI be downloaded to a particular destination file. The download manager will 53 * conduct the download in the background, taking care of HTTP interactions and retrying downloads 54 * after failures or across connectivity changes and system reboots. 55 * <p> 56 * Apps that request downloads through this API should register a broadcast receiver for 57 * {@link #ACTION_NOTIFICATION_CLICKED} to appropriately handle when the user clicks on a running 58 * download in a notification or from the downloads UI. 59 * <p> 60 * Note that the application must have the {@link android.Manifest.permission#INTERNET} 61 * permission to use this class. 62 */ 63 @SystemService(Context.DOWNLOAD_SERVICE) 64 public class DownloadManager { 65 66 /** 67 * An identifier for a particular download, unique across the system. Clients use this ID to 68 * make subsequent calls related to the download. 69 */ 70 public final static String COLUMN_ID = Downloads.Impl._ID; 71 72 /** 73 * The client-supplied title for this download. This will be displayed in system notifications. 74 * Defaults to the empty string. 75 */ 76 public final static String COLUMN_TITLE = Downloads.Impl.COLUMN_TITLE; 77 78 /** 79 * The client-supplied description of this download. This will be displayed in system 80 * notifications. Defaults to the empty string. 81 */ 82 public final static String COLUMN_DESCRIPTION = Downloads.Impl.COLUMN_DESCRIPTION; 83 84 /** 85 * URI to be downloaded. 86 */ 87 public final static String COLUMN_URI = Downloads.Impl.COLUMN_URI; 88 89 /** 90 * Internet Media Type of the downloaded file. If no value is provided upon creation, this will 91 * initially be null and will be filled in based on the server's response once the download has 92 * started. 93 * 94 * @see <a href="http://www.ietf.org/rfc/rfc1590.txt">RFC 1590, defining Media Types</a> 95 */ 96 public final static String COLUMN_MEDIA_TYPE = "media_type"; 97 98 /** 99 * Total size of the download in bytes. This will initially be -1 and will be filled in once 100 * the download starts. 101 */ 102 public final static String COLUMN_TOTAL_SIZE_BYTES = "total_size"; 103 104 /** 105 * Uri where downloaded file will be stored. If a destination is supplied by client, that URI 106 * will be used here. Otherwise, the value will initially be null and will be filled in with a 107 * generated URI once the download has started. 108 */ 109 public final static String COLUMN_LOCAL_URI = "local_uri"; 110 111 /** 112 * Path to the downloaded file on disk. 113 * <p> 114 * Note that apps may not have filesystem permissions to directly access 115 * this path. Instead of trying to open this path directly, apps should use 116 * {@link ContentResolver#openFileDescriptor(Uri, String)} to gain access. 117 * 118 * @deprecated apps should transition to using 119 * {@link ContentResolver#openFileDescriptor(Uri, String)} 120 * instead. 121 */ 122 @Deprecated 123 public final static String COLUMN_LOCAL_FILENAME = "local_filename"; 124 125 /** 126 * Current status of the download, as one of the STATUS_* constants. 127 */ 128 public final static String COLUMN_STATUS = Downloads.Impl.COLUMN_STATUS; 129 130 /** 131 * Provides more detail on the status of the download. Its meaning depends on the value of 132 * {@link #COLUMN_STATUS}. 133 * 134 * When {@link #COLUMN_STATUS} is {@link #STATUS_FAILED}, this indicates the type of error that 135 * occurred. If an HTTP error occurred, this will hold the HTTP status code as defined in RFC 136 * 2616. Otherwise, it will hold one of the ERROR_* constants. 137 * 138 * When {@link #COLUMN_STATUS} is {@link #STATUS_PAUSED}, this indicates why the download is 139 * paused. It will hold one of the PAUSED_* constants. 140 * 141 * If {@link #COLUMN_STATUS} is neither {@link #STATUS_FAILED} nor {@link #STATUS_PAUSED}, this 142 * column's value is undefined. 143 * 144 * @see <a href="http://www.w3.org/Protocols/rfc2616/rfc2616-sec6.html#sec6.1.1">RFC 2616 145 * status codes</a> 146 */ 147 public final static String COLUMN_REASON = "reason"; 148 149 /** 150 * Number of bytes download so far. 151 */ 152 public final static String COLUMN_BYTES_DOWNLOADED_SO_FAR = "bytes_so_far"; 153 154 /** 155 * Timestamp when the download was last modified, in {@link System#currentTimeMillis 156 * System.currentTimeMillis()} (wall clock time in UTC). 157 */ 158 public final static String COLUMN_LAST_MODIFIED_TIMESTAMP = "last_modified_timestamp"; 159 160 /** 161 * The URI to the corresponding entry in MediaProvider for this downloaded entry. It is 162 * used to delete the entries from MediaProvider database when it is deleted from the 163 * downloaded list. 164 */ 165 public static final String COLUMN_MEDIAPROVIDER_URI = Downloads.Impl.COLUMN_MEDIAPROVIDER_URI; 166 167 /** 168 * @hide 169 */ 170 public final static String COLUMN_ALLOW_WRITE = Downloads.Impl.COLUMN_ALLOW_WRITE; 171 172 /** 173 * Value of {@link #COLUMN_STATUS} when the download is waiting to start. 174 */ 175 public final static int STATUS_PENDING = 1 << 0; 176 177 /** 178 * Value of {@link #COLUMN_STATUS} when the download is currently running. 179 */ 180 public final static int STATUS_RUNNING = 1 << 1; 181 182 /** 183 * Value of {@link #COLUMN_STATUS} when the download is waiting to retry or resume. 184 */ 185 public final static int STATUS_PAUSED = 1 << 2; 186 187 /** 188 * Value of {@link #COLUMN_STATUS} when the download has successfully completed. 189 */ 190 public final static int STATUS_SUCCESSFUL = 1 << 3; 191 192 /** 193 * Value of {@link #COLUMN_STATUS} when the download has failed (and will not be retried). 194 */ 195 public final static int STATUS_FAILED = 1 << 4; 196 197 /** 198 * Value of COLUMN_ERROR_CODE when the download has completed with an error that doesn't fit 199 * under any other error code. 200 */ 201 public final static int ERROR_UNKNOWN = 1000; 202 203 /** 204 * Value of {@link #COLUMN_REASON} when a storage issue arises which doesn't fit under any 205 * other error code. Use the more specific {@link #ERROR_INSUFFICIENT_SPACE} and 206 * {@link #ERROR_DEVICE_NOT_FOUND} when appropriate. 207 */ 208 public final static int ERROR_FILE_ERROR = 1001; 209 210 /** 211 * Value of {@link #COLUMN_REASON} when an HTTP code was received that download manager 212 * can't handle. 213 */ 214 public final static int ERROR_UNHANDLED_HTTP_CODE = 1002; 215 216 /** 217 * Value of {@link #COLUMN_REASON} when an error receiving or processing data occurred at 218 * the HTTP level. 219 */ 220 public final static int ERROR_HTTP_DATA_ERROR = 1004; 221 222 /** 223 * Value of {@link #COLUMN_REASON} when there were too many redirects. 224 */ 225 public final static int ERROR_TOO_MANY_REDIRECTS = 1005; 226 227 /** 228 * Value of {@link #COLUMN_REASON} when there was insufficient storage space. Typically, 229 * this is because the SD card is full. 230 */ 231 public final static int ERROR_INSUFFICIENT_SPACE = 1006; 232 233 /** 234 * Value of {@link #COLUMN_REASON} when no external storage device was found. Typically, 235 * this is because the SD card is not mounted. 236 */ 237 public final static int ERROR_DEVICE_NOT_FOUND = 1007; 238 239 /** 240 * Value of {@link #COLUMN_REASON} when some possibly transient error occurred but we can't 241 * resume the download. 242 */ 243 public final static int ERROR_CANNOT_RESUME = 1008; 244 245 /** 246 * Value of {@link #COLUMN_REASON} when the requested destination file already exists (the 247 * download manager will not overwrite an existing file). 248 */ 249 public final static int ERROR_FILE_ALREADY_EXISTS = 1009; 250 251 /** 252 * Value of {@link #COLUMN_REASON} when the download has failed because of 253 * {@link NetworkPolicyManager} controls on the requesting application. 254 * 255 * @hide 256 */ 257 public final static int ERROR_BLOCKED = 1010; 258 259 /** 260 * Value of {@link #COLUMN_REASON} when the download is paused because some network error 261 * occurred and the download manager is waiting before retrying the request. 262 */ 263 public final static int PAUSED_WAITING_TO_RETRY = 1; 264 265 /** 266 * Value of {@link #COLUMN_REASON} when the download is waiting for network connectivity to 267 * proceed. 268 */ 269 public final static int PAUSED_WAITING_FOR_NETWORK = 2; 270 271 /** 272 * Value of {@link #COLUMN_REASON} when the download exceeds a size limit for downloads over 273 * the mobile network and the download manager is waiting for a Wi-Fi connection to proceed. 274 */ 275 public final static int PAUSED_QUEUED_FOR_WIFI = 3; 276 277 /** 278 * Value of {@link #COLUMN_REASON} when the download is paused for some other reason. 279 */ 280 public final static int PAUSED_UNKNOWN = 4; 281 282 /** 283 * Broadcast intent action sent by the download manager when a download completes. 284 */ 285 @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION) 286 public final static String ACTION_DOWNLOAD_COMPLETE = "android.intent.action.DOWNLOAD_COMPLETE"; 287 288 /** 289 * Broadcast intent action sent by the download manager when the user clicks on a running 290 * download, either from a system notification or from the downloads UI. 291 */ 292 @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION) 293 public final static String ACTION_NOTIFICATION_CLICKED = 294 "android.intent.action.DOWNLOAD_NOTIFICATION_CLICKED"; 295 296 /** 297 * Intent action to launch an activity to display all downloads. 298 */ 299 @SdkConstant(SdkConstantType.ACTIVITY_INTENT_ACTION) 300 public final static String ACTION_VIEW_DOWNLOADS = "android.intent.action.VIEW_DOWNLOADS"; 301 302 /** 303 * Intent extra included with {@link #ACTION_VIEW_DOWNLOADS} to start DownloadApp in 304 * sort-by-size mode. 305 */ 306 public final static String INTENT_EXTRAS_SORT_BY_SIZE = 307 "android.app.DownloadManager.extra_sortBySize"; 308 309 /** 310 * Intent extra included with {@link #ACTION_DOWNLOAD_COMPLETE} intents, indicating the ID (as a 311 * long) of the download that just completed. 312 */ 313 public static final String EXTRA_DOWNLOAD_ID = "extra_download_id"; 314 315 /** 316 * When clicks on multiple notifications are received, the following 317 * provides an array of download ids corresponding to the download notification that was 318 * clicked. It can be retrieved by the receiver of this 319 * Intent using {@link android.content.Intent#getLongArrayExtra(String)}. 320 */ 321 public static final String EXTRA_NOTIFICATION_CLICK_DOWNLOAD_IDS = "extra_click_download_ids"; 322 323 /** {@hide} */ 324 @SystemApi 325 public static final String ACTION_DOWNLOAD_COMPLETED = 326 "android.intent.action.DOWNLOAD_COMPLETED"; 327 328 /** 329 * columns to request from DownloadProvider. 330 * @hide 331 */ 332 public static final String[] UNDERLYING_COLUMNS = new String[] { 333 Downloads.Impl._ID, 334 Downloads.Impl._DATA + " AS " + COLUMN_LOCAL_FILENAME, 335 Downloads.Impl.COLUMN_MEDIAPROVIDER_URI, 336 Downloads.Impl.COLUMN_DESTINATION, 337 Downloads.Impl.COLUMN_TITLE, 338 Downloads.Impl.COLUMN_DESCRIPTION, 339 Downloads.Impl.COLUMN_URI, 340 Downloads.Impl.COLUMN_STATUS, 341 Downloads.Impl.COLUMN_FILE_NAME_HINT, 342 Downloads.Impl.COLUMN_MIME_TYPE + " AS " + COLUMN_MEDIA_TYPE, 343 Downloads.Impl.COLUMN_TOTAL_BYTES + " AS " + COLUMN_TOTAL_SIZE_BYTES, 344 Downloads.Impl.COLUMN_LAST_MODIFICATION + " AS " + COLUMN_LAST_MODIFIED_TIMESTAMP, 345 Downloads.Impl.COLUMN_CURRENT_BYTES + " AS " + COLUMN_BYTES_DOWNLOADED_SO_FAR, 346 Downloads.Impl.COLUMN_ALLOW_WRITE, 347 /* add the following 'computed' columns to the cursor. 348 * they are not 'returned' by the database, but their inclusion 349 * eliminates need to have lot of methods in CursorTranslator 350 */ 351 "'placeholder' AS " + COLUMN_LOCAL_URI, 352 "'placeholder' AS " + COLUMN_REASON 353 }; 354 355 /** 356 * This class contains all the information necessary to request a new download. The URI is the 357 * only required parameter. 358 * 359 * Note that the default download destination is a shared volume where the system might delete 360 * your file if it needs to reclaim space for system use. If this is a problem, use a location 361 * on external storage (see {@link #setDestinationUri(Uri)}. 362 */ 363 public static class Request { 364 /** 365 * Bit flag for {@link #setAllowedNetworkTypes} corresponding to 366 * {@link ConnectivityManager#TYPE_MOBILE}. 367 */ 368 public static final int NETWORK_MOBILE = 1 << 0; 369 370 /** 371 * Bit flag for {@link #setAllowedNetworkTypes} corresponding to 372 * {@link ConnectivityManager#TYPE_WIFI}. 373 */ 374 public static final int NETWORK_WIFI = 1 << 1; 375 376 /** 377 * Bit flag for {@link #setAllowedNetworkTypes} corresponding to 378 * {@link ConnectivityManager#TYPE_BLUETOOTH}. 379 * @hide 380 */ 381 @Deprecated 382 public static final int NETWORK_BLUETOOTH = 1 << 2; 383 384 private Uri mUri; 385 private Uri mDestinationUri; 386 private List<Pair<String, String>> mRequestHeaders = new ArrayList<Pair<String, String>>(); 387 private CharSequence mTitle; 388 private CharSequence mDescription; 389 private String mMimeType; 390 private int mAllowedNetworkTypes = ~0; // default to all network types allowed 391 private boolean mRoamingAllowed = true; 392 private boolean mMeteredAllowed = true; 393 private int mFlags = 0; 394 private boolean mIsVisibleInDownloadsUi = true; 395 private boolean mScannable = false; 396 private boolean mUseSystemCache = false; 397 /** if a file is designated as a MediaScanner scannable file, the following value is 398 * stored in the database column {@link Downloads.Impl#COLUMN_MEDIA_SCANNED}. 399 */ 400 private static final int SCANNABLE_VALUE_YES = 0; 401 // value of 1 is stored in the above column by DownloadProvider after it is scanned by 402 // MediaScanner 403 /** if a file is designated as a file that should not be scanned by MediaScanner, 404 * the following value is stored in the database column 405 * {@link Downloads.Impl#COLUMN_MEDIA_SCANNED}. 406 */ 407 private static final int SCANNABLE_VALUE_NO = 2; 408 409 /** 410 * This download is visible but only shows in the notifications 411 * while it's in progress. 412 */ 413 public static final int VISIBILITY_VISIBLE = 0; 414 415 /** 416 * This download is visible and shows in the notifications while 417 * in progress and after completion. 418 */ 419 public static final int VISIBILITY_VISIBLE_NOTIFY_COMPLETED = 1; 420 421 /** 422 * This download doesn't show in the UI or in the notifications. 423 */ 424 public static final int VISIBILITY_HIDDEN = 2; 425 426 /** 427 * This download shows in the notifications after completion ONLY. 428 * It is usuable only with 429 * {@link DownloadManager#addCompletedDownload(String, String, 430 * boolean, String, String, long, boolean)}. 431 */ 432 public static final int VISIBILITY_VISIBLE_NOTIFY_ONLY_COMPLETION = 3; 433 434 /** can take any of the following values: {@link #VISIBILITY_HIDDEN} 435 * {@link #VISIBILITY_VISIBLE_NOTIFY_COMPLETED}, {@link #VISIBILITY_VISIBLE}, 436 * {@link #VISIBILITY_VISIBLE_NOTIFY_ONLY_COMPLETION} 437 */ 438 private int mNotificationVisibility = VISIBILITY_VISIBLE; 439 440 /** 441 * @param uri the HTTP or HTTPS URI to download. 442 */ Request(Uri uri)443 public Request(Uri uri) { 444 if (uri == null) { 445 throw new NullPointerException(); 446 } 447 String scheme = uri.getScheme(); 448 if (scheme == null || (!scheme.equals("http") && !scheme.equals("https"))) { 449 throw new IllegalArgumentException("Can only download HTTP/HTTPS URIs: " + uri); 450 } 451 mUri = uri; 452 } 453 Request(String uriString)454 Request(String uriString) { 455 mUri = Uri.parse(uriString); 456 } 457 458 /** 459 * Set the local destination for the downloaded file. Must be a file URI to a path on 460 * external storage, and the calling application must have the WRITE_EXTERNAL_STORAGE 461 * permission. 462 * <p> 463 * The downloaded file is not scanned by MediaScanner. 464 * But it can be made scannable by calling {@link #allowScanningByMediaScanner()}. 465 * <p> 466 * By default, downloads are saved to a generated filename in the shared download cache and 467 * may be deleted by the system at any time to reclaim space. 468 * 469 * @return this object 470 */ setDestinationUri(Uri uri)471 public Request setDestinationUri(Uri uri) { 472 mDestinationUri = uri; 473 return this; 474 } 475 476 /** 477 * Set the local destination for the downloaded file to the system cache dir (/cache). 478 * This is only available to System apps with the permission 479 * {@link android.Manifest.permission#ACCESS_CACHE_FILESYSTEM}. 480 * <p> 481 * The downloaded file is not scanned by MediaScanner. 482 * But it can be made scannable by calling {@link #allowScanningByMediaScanner()}. 483 * <p> 484 * Files downloaded to /cache may be deleted by the system at any time to reclaim space. 485 * 486 * @return this object 487 * @hide 488 */ setDestinationToSystemCache()489 public Request setDestinationToSystemCache() { 490 mUseSystemCache = true; 491 return this; 492 } 493 494 /** 495 * Set the local destination for the downloaded file to a path within 496 * the application's external files directory (as returned by 497 * {@link Context#getExternalFilesDir(String)}. 498 * <p> 499 * The downloaded file is not scanned by MediaScanner. But it can be 500 * made scannable by calling {@link #allowScanningByMediaScanner()}. 501 * 502 * @param context the {@link Context} to use in determining the external 503 * files directory 504 * @param dirType the directory type to pass to 505 * {@link Context#getExternalFilesDir(String)} 506 * @param subPath the path within the external directory, including the 507 * destination filename 508 * @return this object 509 * @throws IllegalStateException If the external storage directory 510 * cannot be found or created. 511 */ setDestinationInExternalFilesDir(Context context, String dirType, String subPath)512 public Request setDestinationInExternalFilesDir(Context context, String dirType, 513 String subPath) { 514 final File file = context.getExternalFilesDir(dirType); 515 if (file == null) { 516 throw new IllegalStateException("Failed to get external storage files directory"); 517 } else if (file.exists()) { 518 if (!file.isDirectory()) { 519 throw new IllegalStateException(file.getAbsolutePath() + 520 " already exists and is not a directory"); 521 } 522 } else { 523 if (!file.mkdirs()) { 524 throw new IllegalStateException("Unable to create directory: "+ 525 file.getAbsolutePath()); 526 } 527 } 528 setDestinationFromBase(file, subPath); 529 return this; 530 } 531 532 /** 533 * Set the local destination for the downloaded file to a path within 534 * the public external storage directory (as returned by 535 * {@link Environment#getExternalStoragePublicDirectory(String)}). 536 * <p> 537 * The downloaded file is not scanned by MediaScanner. But it can be 538 * made scannable by calling {@link #allowScanningByMediaScanner()}. 539 * 540 * @param dirType the directory type to pass to {@link Environment#getExternalStoragePublicDirectory(String)} 541 * @param subPath the path within the external directory, including the 542 * destination filename 543 * @return this object 544 * @throws IllegalStateException If the external storage directory 545 * cannot be found or created. 546 */ setDestinationInExternalPublicDir(String dirType, String subPath)547 public Request setDestinationInExternalPublicDir(String dirType, String subPath) { 548 File file = Environment.getExternalStoragePublicDirectory(dirType); 549 if (file == null) { 550 throw new IllegalStateException("Failed to get external storage public directory"); 551 } else if (file.exists()) { 552 if (!file.isDirectory()) { 553 throw new IllegalStateException(file.getAbsolutePath() + 554 " already exists and is not a directory"); 555 } 556 } else { 557 if (!file.mkdirs()) { 558 throw new IllegalStateException("Unable to create directory: "+ 559 file.getAbsolutePath()); 560 } 561 } 562 setDestinationFromBase(file, subPath); 563 return this; 564 } 565 setDestinationFromBase(File base, String subPath)566 private void setDestinationFromBase(File base, String subPath) { 567 if (subPath == null) { 568 throw new NullPointerException("subPath cannot be null"); 569 } 570 mDestinationUri = Uri.withAppendedPath(Uri.fromFile(base), subPath); 571 } 572 573 /** 574 * If the file to be downloaded is to be scanned by MediaScanner, this method 575 * should be called before {@link DownloadManager#enqueue(Request)} is called. 576 */ allowScanningByMediaScanner()577 public void allowScanningByMediaScanner() { 578 mScannable = true; 579 } 580 581 /** 582 * Add an HTTP header to be included with the download request. The header will be added to 583 * the end of the list. 584 * @param header HTTP header name 585 * @param value header value 586 * @return this object 587 * @see <a href="http://www.w3.org/Protocols/rfc2616/rfc2616-sec4.html#sec4.2">HTTP/1.1 588 * Message Headers</a> 589 */ addRequestHeader(String header, String value)590 public Request addRequestHeader(String header, String value) { 591 if (header == null) { 592 throw new NullPointerException("header cannot be null"); 593 } 594 if (header.contains(":")) { 595 throw new IllegalArgumentException("header may not contain ':'"); 596 } 597 if (value == null) { 598 value = ""; 599 } 600 mRequestHeaders.add(Pair.create(header, value)); 601 return this; 602 } 603 604 /** 605 * Set the title of this download, to be displayed in notifications (if enabled). If no 606 * title is given, a default one will be assigned based on the download filename, once the 607 * download starts. 608 * @return this object 609 */ setTitle(CharSequence title)610 public Request setTitle(CharSequence title) { 611 mTitle = title; 612 return this; 613 } 614 615 /** 616 * Set a description of this download, to be displayed in notifications (if enabled) 617 * @return this object 618 */ setDescription(CharSequence description)619 public Request setDescription(CharSequence description) { 620 mDescription = description; 621 return this; 622 } 623 624 /** 625 * Set the MIME content type of this download. This will override the content type declared 626 * in the server's response. 627 * @see <a href="http://www.w3.org/Protocols/rfc2616/rfc2616-sec3.html#sec3.7">HTTP/1.1 628 * Media Types</a> 629 * @return this object 630 */ setMimeType(String mimeType)631 public Request setMimeType(String mimeType) { 632 mMimeType = mimeType; 633 return this; 634 } 635 636 /** 637 * Control whether a system notification is posted by the download manager while this 638 * download is running. If enabled, the download manager posts notifications about downloads 639 * through the system {@link android.app.NotificationManager}. By default, a notification is 640 * shown. 641 * 642 * If set to false, this requires the permission 643 * android.permission.DOWNLOAD_WITHOUT_NOTIFICATION. 644 * 645 * @param show whether the download manager should show a notification for this download. 646 * @return this object 647 * @deprecated use {@link #setNotificationVisibility(int)} 648 */ 649 @Deprecated setShowRunningNotification(boolean show)650 public Request setShowRunningNotification(boolean show) { 651 return (show) ? setNotificationVisibility(VISIBILITY_VISIBLE) : 652 setNotificationVisibility(VISIBILITY_HIDDEN); 653 } 654 655 /** 656 * Control whether a system notification is posted by the download manager while this 657 * download is running or when it is completed. 658 * If enabled, the download manager posts notifications about downloads 659 * through the system {@link android.app.NotificationManager}. 660 * By default, a notification is shown only when the download is in progress. 661 *<p> 662 * It can take the following values: {@link #VISIBILITY_HIDDEN}, 663 * {@link #VISIBILITY_VISIBLE}, 664 * {@link #VISIBILITY_VISIBLE_NOTIFY_COMPLETED}. 665 *<p> 666 * If set to {@link #VISIBILITY_HIDDEN}, this requires the permission 667 * android.permission.DOWNLOAD_WITHOUT_NOTIFICATION. 668 * 669 * @param visibility the visibility setting value 670 * @return this object 671 */ setNotificationVisibility(int visibility)672 public Request setNotificationVisibility(int visibility) { 673 mNotificationVisibility = visibility; 674 return this; 675 } 676 677 /** 678 * Restrict the types of networks over which this download may proceed. 679 * By default, all network types are allowed. Consider using 680 * {@link #setAllowedOverMetered(boolean)} instead, since it's more 681 * flexible. 682 * <p> 683 * As of {@link android.os.Build.VERSION_CODES#N}, setting only the 684 * {@link #NETWORK_WIFI} flag here is equivalent to calling 685 * {@link #setAllowedOverMetered(boolean)} with {@code false}. 686 * 687 * @param flags any combination of the NETWORK_* bit flags. 688 * @return this object 689 */ setAllowedNetworkTypes(int flags)690 public Request setAllowedNetworkTypes(int flags) { 691 mAllowedNetworkTypes = flags; 692 return this; 693 } 694 695 /** 696 * Set whether this download may proceed over a roaming connection. By default, roaming is 697 * allowed. 698 * @param allowed whether to allow a roaming connection to be used 699 * @return this object 700 */ setAllowedOverRoaming(boolean allowed)701 public Request setAllowedOverRoaming(boolean allowed) { 702 mRoamingAllowed = allowed; 703 return this; 704 } 705 706 /** 707 * Set whether this download may proceed over a metered network 708 * connection. By default, metered networks are allowed. 709 * 710 * @see ConnectivityManager#isActiveNetworkMetered() 711 */ setAllowedOverMetered(boolean allow)712 public Request setAllowedOverMetered(boolean allow) { 713 mMeteredAllowed = allow; 714 return this; 715 } 716 717 /** 718 * Specify that to run this download, the device needs to be plugged in. 719 * This defaults to false. 720 * 721 * @param requiresCharging Whether or not the device is plugged in. 722 * @see android.app.job.JobInfo.Builder#setRequiresCharging(boolean) 723 */ setRequiresCharging(boolean requiresCharging)724 public Request setRequiresCharging(boolean requiresCharging) { 725 if (requiresCharging) { 726 mFlags |= Downloads.Impl.FLAG_REQUIRES_CHARGING; 727 } else { 728 mFlags &= ~Downloads.Impl.FLAG_REQUIRES_CHARGING; 729 } 730 return this; 731 } 732 733 /** 734 * Specify that to run, the download needs the device to be in idle 735 * mode. This defaults to false. 736 * <p> 737 * Idle mode is a loose definition provided by the system, which means 738 * that the device is not in use, and has not been in use for some time. 739 * 740 * @param requiresDeviceIdle Whether or not the device need be within an 741 * idle maintenance window. 742 * @see android.app.job.JobInfo.Builder#setRequiresDeviceIdle(boolean) 743 */ setRequiresDeviceIdle(boolean requiresDeviceIdle)744 public Request setRequiresDeviceIdle(boolean requiresDeviceIdle) { 745 if (requiresDeviceIdle) { 746 mFlags |= Downloads.Impl.FLAG_REQUIRES_DEVICE_IDLE; 747 } else { 748 mFlags &= ~Downloads.Impl.FLAG_REQUIRES_DEVICE_IDLE; 749 } 750 return this; 751 } 752 753 /** 754 * Set whether this download should be displayed in the system's Downloads UI. True by 755 * default. 756 * @param isVisible whether to display this download in the Downloads UI 757 * @return this object 758 */ setVisibleInDownloadsUi(boolean isVisible)759 public Request setVisibleInDownloadsUi(boolean isVisible) { 760 mIsVisibleInDownloadsUi = isVisible; 761 return this; 762 } 763 764 /** 765 * @return ContentValues to be passed to DownloadProvider.insert() 766 */ toContentValues(String packageName)767 ContentValues toContentValues(String packageName) { 768 ContentValues values = new ContentValues(); 769 assert mUri != null; 770 values.put(Downloads.Impl.COLUMN_URI, mUri.toString()); 771 values.put(Downloads.Impl.COLUMN_IS_PUBLIC_API, true); 772 values.put(Downloads.Impl.COLUMN_NOTIFICATION_PACKAGE, packageName); 773 774 if (mDestinationUri != null) { 775 values.put(Downloads.Impl.COLUMN_DESTINATION, Downloads.Impl.DESTINATION_FILE_URI); 776 values.put(Downloads.Impl.COLUMN_FILE_NAME_HINT, mDestinationUri.toString()); 777 } else { 778 values.put(Downloads.Impl.COLUMN_DESTINATION, 779 (this.mUseSystemCache) ? 780 Downloads.Impl.DESTINATION_SYSTEMCACHE_PARTITION : 781 Downloads.Impl.DESTINATION_CACHE_PARTITION_PURGEABLE); 782 } 783 // is the file supposed to be media-scannable? 784 values.put(Downloads.Impl.COLUMN_MEDIA_SCANNED, (mScannable) ? SCANNABLE_VALUE_YES : 785 SCANNABLE_VALUE_NO); 786 787 if (!mRequestHeaders.isEmpty()) { 788 encodeHttpHeaders(values); 789 } 790 791 putIfNonNull(values, Downloads.Impl.COLUMN_TITLE, mTitle); 792 putIfNonNull(values, Downloads.Impl.COLUMN_DESCRIPTION, mDescription); 793 putIfNonNull(values, Downloads.Impl.COLUMN_MIME_TYPE, mMimeType); 794 795 values.put(Downloads.Impl.COLUMN_VISIBILITY, mNotificationVisibility); 796 values.put(Downloads.Impl.COLUMN_ALLOWED_NETWORK_TYPES, mAllowedNetworkTypes); 797 values.put(Downloads.Impl.COLUMN_ALLOW_ROAMING, mRoamingAllowed); 798 values.put(Downloads.Impl.COLUMN_ALLOW_METERED, mMeteredAllowed); 799 values.put(Downloads.Impl.COLUMN_FLAGS, mFlags); 800 values.put(Downloads.Impl.COLUMN_IS_VISIBLE_IN_DOWNLOADS_UI, mIsVisibleInDownloadsUi); 801 802 return values; 803 } 804 encodeHttpHeaders(ContentValues values)805 private void encodeHttpHeaders(ContentValues values) { 806 int index = 0; 807 for (Pair<String, String> header : mRequestHeaders) { 808 String headerString = header.first + ": " + header.second; 809 values.put(Downloads.Impl.RequestHeaders.INSERT_KEY_PREFIX + index, headerString); 810 index++; 811 } 812 } 813 putIfNonNull(ContentValues contentValues, String key, Object value)814 private void putIfNonNull(ContentValues contentValues, String key, Object value) { 815 if (value != null) { 816 contentValues.put(key, value.toString()); 817 } 818 } 819 } 820 821 /** 822 * This class may be used to filter download manager queries. 823 */ 824 public static class Query { 825 /** 826 * Constant for use with {@link #orderBy} 827 * @hide 828 */ 829 public static final int ORDER_ASCENDING = 1; 830 831 /** 832 * Constant for use with {@link #orderBy} 833 * @hide 834 */ 835 public static final int ORDER_DESCENDING = 2; 836 837 private long[] mIds = null; 838 private Integer mStatusFlags = null; 839 private String mFilterString = null; 840 private String mOrderByColumn = Downloads.Impl.COLUMN_LAST_MODIFICATION; 841 private int mOrderDirection = ORDER_DESCENDING; 842 private boolean mOnlyIncludeVisibleInDownloadsUi = false; 843 844 /** 845 * Include only the downloads with the given IDs. 846 * @return this object 847 */ setFilterById(long... ids)848 public Query setFilterById(long... ids) { 849 mIds = ids; 850 return this; 851 } 852 853 /** 854 * 855 * Include only the downloads that contains the given string in its name. 856 * @return this object 857 * @hide 858 */ setFilterByString(@ullable String filter)859 public Query setFilterByString(@Nullable String filter) { 860 mFilterString = filter; 861 return this; 862 } 863 864 /** 865 * Include only downloads with status matching any the given status flags. 866 * @param flags any combination of the STATUS_* bit flags 867 * @return this object 868 */ setFilterByStatus(int flags)869 public Query setFilterByStatus(int flags) { 870 mStatusFlags = flags; 871 return this; 872 } 873 874 /** 875 * Controls whether this query includes downloads not visible in the system's Downloads UI. 876 * @param value if true, this query will only include downloads that should be displayed in 877 * the system's Downloads UI; if false (the default), this query will include 878 * both visible and invisible downloads. 879 * @return this object 880 * @hide 881 */ setOnlyIncludeVisibleInDownloadsUi(boolean value)882 public Query setOnlyIncludeVisibleInDownloadsUi(boolean value) { 883 mOnlyIncludeVisibleInDownloadsUi = value; 884 return this; 885 } 886 887 /** 888 * Change the sort order of the returned Cursor. 889 * 890 * @param column one of the COLUMN_* constants; currently, only 891 * {@link #COLUMN_LAST_MODIFIED_TIMESTAMP} and {@link #COLUMN_TOTAL_SIZE_BYTES} are 892 * supported. 893 * @param direction either {@link #ORDER_ASCENDING} or {@link #ORDER_DESCENDING} 894 * @return this object 895 * @hide 896 */ orderBy(String column, int direction)897 public Query orderBy(String column, int direction) { 898 if (direction != ORDER_ASCENDING && direction != ORDER_DESCENDING) { 899 throw new IllegalArgumentException("Invalid direction: " + direction); 900 } 901 902 if (column.equals(COLUMN_LAST_MODIFIED_TIMESTAMP)) { 903 mOrderByColumn = Downloads.Impl.COLUMN_LAST_MODIFICATION; 904 } else if (column.equals(COLUMN_TOTAL_SIZE_BYTES)) { 905 mOrderByColumn = Downloads.Impl.COLUMN_TOTAL_BYTES; 906 } else { 907 throw new IllegalArgumentException("Cannot order by " + column); 908 } 909 mOrderDirection = direction; 910 return this; 911 } 912 913 /** 914 * Run this query using the given ContentResolver. 915 * @param projection the projection to pass to ContentResolver.query() 916 * @return the Cursor returned by ContentResolver.query() 917 */ runQuery(ContentResolver resolver, String[] projection, Uri baseUri)918 Cursor runQuery(ContentResolver resolver, String[] projection, Uri baseUri) { 919 Uri uri = baseUri; 920 List<String> selectionParts = new ArrayList<String>(); 921 String[] selectionArgs = null; 922 923 int whereArgsCount = (mIds == null) ? 0 : mIds.length; 924 whereArgsCount = (mFilterString == null) ? whereArgsCount : whereArgsCount + 1; 925 selectionArgs = new String[whereArgsCount]; 926 927 if (whereArgsCount > 0) { 928 if (mIds != null) { 929 selectionParts.add(getWhereClauseForIds(mIds)); 930 getWhereArgsForIds(mIds, selectionArgs); 931 } 932 933 if (mFilterString != null) { 934 selectionParts.add(Downloads.Impl.COLUMN_TITLE + " LIKE ?"); 935 selectionArgs[selectionArgs.length - 1] = "%" + mFilterString + "%"; 936 } 937 } 938 939 if (mStatusFlags != null) { 940 List<String> parts = new ArrayList<String>(); 941 if ((mStatusFlags & STATUS_PENDING) != 0) { 942 parts.add(statusClause("=", Downloads.Impl.STATUS_PENDING)); 943 } 944 if ((mStatusFlags & STATUS_RUNNING) != 0) { 945 parts.add(statusClause("=", Downloads.Impl.STATUS_RUNNING)); 946 } 947 if ((mStatusFlags & STATUS_PAUSED) != 0) { 948 parts.add(statusClause("=", Downloads.Impl.STATUS_PAUSED_BY_APP)); 949 parts.add(statusClause("=", Downloads.Impl.STATUS_WAITING_TO_RETRY)); 950 parts.add(statusClause("=", Downloads.Impl.STATUS_WAITING_FOR_NETWORK)); 951 parts.add(statusClause("=", Downloads.Impl.STATUS_QUEUED_FOR_WIFI)); 952 } 953 if ((mStatusFlags & STATUS_SUCCESSFUL) != 0) { 954 parts.add(statusClause("=", Downloads.Impl.STATUS_SUCCESS)); 955 } 956 if ((mStatusFlags & STATUS_FAILED) != 0) { 957 parts.add("(" + statusClause(">=", 400) 958 + " AND " + statusClause("<", 600) + ")"); 959 } 960 selectionParts.add(joinStrings(" OR ", parts)); 961 } 962 963 if (mOnlyIncludeVisibleInDownloadsUi) { 964 selectionParts.add(Downloads.Impl.COLUMN_IS_VISIBLE_IN_DOWNLOADS_UI + " != '0'"); 965 } 966 967 // only return rows which are not marked 'deleted = 1' 968 selectionParts.add(Downloads.Impl.COLUMN_DELETED + " != '1'"); 969 970 String selection = joinStrings(" AND ", selectionParts); 971 String orderDirection = (mOrderDirection == ORDER_ASCENDING ? "ASC" : "DESC"); 972 String orderBy = mOrderByColumn + " " + orderDirection; 973 974 return resolver.query(uri, projection, selection, selectionArgs, orderBy); 975 } 976 joinStrings(String joiner, Iterable<String> parts)977 private String joinStrings(String joiner, Iterable<String> parts) { 978 StringBuilder builder = new StringBuilder(); 979 boolean first = true; 980 for (String part : parts) { 981 if (!first) { 982 builder.append(joiner); 983 } 984 builder.append(part); 985 first = false; 986 } 987 return builder.toString(); 988 } 989 statusClause(String operator, int value)990 private String statusClause(String operator, int value) { 991 return Downloads.Impl.COLUMN_STATUS + operator + "'" + value + "'"; 992 } 993 } 994 995 private final ContentResolver mResolver; 996 private final String mPackageName; 997 998 private Uri mBaseUri = Downloads.Impl.CONTENT_URI; 999 private boolean mAccessFilename; 1000 1001 /** 1002 * @hide 1003 */ DownloadManager(Context context)1004 public DownloadManager(Context context) { 1005 mResolver = context.getContentResolver(); 1006 mPackageName = context.getPackageName(); 1007 1008 // Callers can access filename columns when targeting old platform 1009 // versions; otherwise we throw telling them it's deprecated. 1010 mAccessFilename = context.getApplicationInfo().targetSdkVersion < Build.VERSION_CODES.N; 1011 } 1012 1013 /** 1014 * Makes this object access the download provider through /all_downloads URIs rather than 1015 * /my_downloads URIs, for clients that have permission to do so. 1016 * @hide 1017 */ 1018 public void setAccessAllDownloads(boolean accessAllDownloads) { 1019 if (accessAllDownloads) { 1020 mBaseUri = Downloads.Impl.ALL_DOWNLOADS_CONTENT_URI; 1021 } else { 1022 mBaseUri = Downloads.Impl.CONTENT_URI; 1023 } 1024 } 1025 1026 /** {@hide} */ 1027 public void setAccessFilename(boolean accessFilename) { 1028 mAccessFilename = accessFilename; 1029 } 1030 1031 /** 1032 * Enqueue a new download. The download will start automatically once the download manager is 1033 * ready to execute it and connectivity is available. 1034 * 1035 * @param request the parameters specifying this download 1036 * @return an ID for the download, unique across the system. This ID is used to make future 1037 * calls related to this download. 1038 */ 1039 public long enqueue(Request request) { 1040 ContentValues values = request.toContentValues(mPackageName); 1041 Uri downloadUri = mResolver.insert(Downloads.Impl.CONTENT_URI, values); 1042 long id = Long.parseLong(downloadUri.getLastPathSegment()); 1043 return id; 1044 } 1045 1046 /** 1047 * Marks the specified download as 'to be deleted'. This is done when a completed download 1048 * is to be removed but the row was stored without enough info to delete the corresponding 1049 * metadata from Mediaprovider database. Actual cleanup of this row is done in DownloadService. 1050 * 1051 * @param ids the IDs of the downloads to be marked 'deleted' 1052 * @return the number of downloads actually updated 1053 * @hide 1054 */ 1055 public int markRowDeleted(long... ids) { 1056 if (ids == null || ids.length == 0) { 1057 // called with nothing to remove! 1058 throw new IllegalArgumentException("input param 'ids' can't be null"); 1059 } 1060 return mResolver.delete(mBaseUri, getWhereClauseForIds(ids), getWhereArgsForIds(ids)); 1061 } 1062 1063 /** 1064 * Cancel downloads and remove them from the download manager. Each download will be stopped if 1065 * it was running, and it will no longer be accessible through the download manager. 1066 * If there is a downloaded file, partial or complete, it is deleted. 1067 * 1068 * @param ids the IDs of the downloads to remove 1069 * @return the number of downloads actually removed 1070 */ 1071 public int remove(long... ids) { 1072 return markRowDeleted(ids); 1073 } 1074 1075 /** 1076 * Query the download manager about downloads that have been requested. 1077 * @param query parameters specifying filters for this query 1078 * @return a Cursor over the result set of downloads, with columns consisting of all the 1079 * COLUMN_* constants. 1080 */ 1081 public Cursor query(Query query) { 1082 Cursor underlyingCursor = query.runQuery(mResolver, UNDERLYING_COLUMNS, mBaseUri); 1083 if (underlyingCursor == null) { 1084 return null; 1085 } 1086 return new CursorTranslator(underlyingCursor, mBaseUri, mAccessFilename); 1087 } 1088 1089 /** 1090 * Open a downloaded file for reading. The download must have completed. 1091 * @param id the ID of the download 1092 * @return a read-only {@link ParcelFileDescriptor} 1093 * @throws FileNotFoundException if the destination file does not already exist 1094 */ 1095 public ParcelFileDescriptor openDownloadedFile(long id) throws FileNotFoundException { 1096 return mResolver.openFileDescriptor(getDownloadUri(id), "r"); 1097 } 1098 1099 /** 1100 * Returns the {@link Uri} of the given downloaded file id, if the file is 1101 * downloaded successfully. Otherwise, null is returned. 1102 * 1103 * @param id the id of the downloaded file. 1104 * @return the {@link Uri} of the given downloaded file id, if download was 1105 * successful. null otherwise. 1106 */ 1107 public Uri getUriForDownloadedFile(long id) { 1108 // to check if the file is in cache, get its destination from the database 1109 Query query = new Query().setFilterById(id); 1110 Cursor cursor = null; 1111 try { 1112 cursor = query(query); 1113 if (cursor == null) { 1114 return null; 1115 } 1116 if (cursor.moveToFirst()) { 1117 int status = cursor.getInt(cursor.getColumnIndexOrThrow(COLUMN_STATUS)); 1118 if (DownloadManager.STATUS_SUCCESSFUL == status) { 1119 return ContentUris.withAppendedId(Downloads.Impl.ALL_DOWNLOADS_CONTENT_URI, id); 1120 } 1121 } 1122 } finally { 1123 if (cursor != null) { 1124 cursor.close(); 1125 } 1126 } 1127 // downloaded file not found or its status is not 'successfully completed' 1128 return null; 1129 } 1130 1131 /** 1132 * Returns the media type of the given downloaded file id, if the file was 1133 * downloaded successfully. Otherwise, null is returned. 1134 * 1135 * @param id the id of the downloaded file. 1136 * @return the media type of the given downloaded file id, if download was successful. null 1137 * otherwise. 1138 */ 1139 public String getMimeTypeForDownloadedFile(long id) { 1140 Query query = new Query().setFilterById(id); 1141 Cursor cursor = null; 1142 try { 1143 cursor = query(query); 1144 if (cursor == null) { 1145 return null; 1146 } 1147 while (cursor.moveToFirst()) { 1148 return cursor.getString(cursor.getColumnIndexOrThrow(COLUMN_MEDIA_TYPE)); 1149 } 1150 } finally { 1151 if (cursor != null) { 1152 cursor.close(); 1153 } 1154 } 1155 // downloaded file not found or its status is not 'successfully completed' 1156 return null; 1157 } 1158 1159 /** 1160 * Restart the given downloads, which must have already completed (successfully or not). This 1161 * method will only work when called from within the download manager's process. 1162 * @param ids the IDs of the downloads 1163 * @hide 1164 */ 1165 public void restartDownload(long... ids) { 1166 Cursor cursor = query(new Query().setFilterById(ids)); 1167 try { 1168 for (cursor.moveToFirst(); !cursor.isAfterLast(); cursor.moveToNext()) { 1169 int status = cursor.getInt(cursor.getColumnIndex(COLUMN_STATUS)); 1170 if (status != STATUS_SUCCESSFUL && status != STATUS_FAILED) { 1171 throw new IllegalArgumentException("Cannot restart incomplete download: " 1172 + cursor.getLong(cursor.getColumnIndex(COLUMN_ID))); 1173 } 1174 } 1175 } finally { 1176 cursor.close(); 1177 } 1178 1179 ContentValues values = new ContentValues(); 1180 values.put(Downloads.Impl.COLUMN_CURRENT_BYTES, 0); 1181 values.put(Downloads.Impl.COLUMN_TOTAL_BYTES, -1); 1182 values.putNull(Downloads.Impl._DATA); 1183 values.put(Downloads.Impl.COLUMN_STATUS, Downloads.Impl.STATUS_PENDING); 1184 values.put(Downloads.Impl.COLUMN_FAILED_CONNECTIONS, 0); 1185 mResolver.update(mBaseUri, values, getWhereClauseForIds(ids), getWhereArgsForIds(ids)); 1186 } 1187 1188 /** 1189 * Force the given downloads to proceed even if their size is larger than 1190 * {@link #getMaxBytesOverMobile(Context)}. 1191 * 1192 * @hide 1193 */ 1194 public void forceDownload(long... ids) { 1195 ContentValues values = new ContentValues(); 1196 values.put(Downloads.Impl.COLUMN_STATUS, Downloads.Impl.STATUS_PENDING); 1197 values.put(Downloads.Impl.COLUMN_CONTROL, Downloads.Impl.CONTROL_RUN); 1198 values.put(Downloads.Impl.COLUMN_BYPASS_RECOMMENDED_SIZE_LIMIT, 1); 1199 mResolver.update(mBaseUri, values, getWhereClauseForIds(ids), getWhereArgsForIds(ids)); 1200 } 1201 1202 /** 1203 * Returns maximum size, in bytes, of downloads that may go over a mobile connection; or null if 1204 * there's no limit 1205 * 1206 * @param context the {@link Context} to use for accessing the {@link ContentResolver} 1207 * @return maximum size, in bytes, of downloads that may go over a mobile connection; or null if 1208 * there's no limit 1209 */ 1210 public static Long getMaxBytesOverMobile(Context context) { 1211 try { 1212 return Settings.Global.getLong(context.getContentResolver(), 1213 Settings.Global.DOWNLOAD_MAX_BYTES_OVER_MOBILE); 1214 } catch (SettingNotFoundException exc) { 1215 return null; 1216 } 1217 } 1218 1219 /** 1220 * Rename the given download if the download has completed 1221 * 1222 * @param context the {@link Context} to use in case need to update MediaProvider 1223 * @param id the downloaded id 1224 * @param displayName the new name to rename to 1225 * @return true if rename was successful, false otherwise 1226 * @hide 1227 */ 1228 public boolean rename(Context context, long id, String displayName) { 1229 if (!FileUtils.isValidFatFilename(displayName)) { 1230 throw new SecurityException(displayName + " is not a valid filename"); 1231 } 1232 1233 Query query = new Query().setFilterById(id); 1234 Cursor cursor = null; 1235 String oldDisplayName = null; 1236 String mimeType = null; 1237 try { 1238 cursor = query(query); 1239 if (cursor == null) { 1240 return false; 1241 } 1242 if (cursor.moveToFirst()) { 1243 int status = cursor.getInt(cursor.getColumnIndexOrThrow(COLUMN_STATUS)); 1244 if (DownloadManager.STATUS_SUCCESSFUL != status) { 1245 return false; 1246 } 1247 oldDisplayName = cursor.getString(cursor.getColumnIndexOrThrow(COLUMN_TITLE)); 1248 mimeType = cursor.getString(cursor.getColumnIndexOrThrow(COLUMN_MEDIA_TYPE)); 1249 } 1250 } finally { 1251 if (cursor != null) { 1252 cursor.close(); 1253 } 1254 } 1255 1256 if (oldDisplayName == null || mimeType == null) { 1257 throw new IllegalStateException( 1258 "Document with id " + id + " does not exist"); 1259 } 1260 1261 final File parent = Environment.getExternalStoragePublicDirectory( 1262 Environment.DIRECTORY_DOWNLOADS); 1263 1264 final File before = new File(parent, oldDisplayName); 1265 final File after = new File(parent, displayName); 1266 1267 if (after.exists()) { 1268 throw new IllegalStateException("Already exists " + after); 1269 } 1270 if (!before.renameTo(after)) { 1271 throw new IllegalStateException("Failed to rename to " + after); 1272 } 1273 1274 // Update MediaProvider if necessary 1275 if (mimeType.startsWith("image/")) { 1276 context.getContentResolver().delete(Images.Media.EXTERNAL_CONTENT_URI, 1277 Images.Media.DATA + "=?", 1278 new String[] { 1279 before.getAbsolutePath() 1280 }); 1281 1282 Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE); 1283 intent.setData(Uri.fromFile(after)); 1284 context.sendBroadcast(intent); 1285 } 1286 1287 ContentValues values = new ContentValues(); 1288 values.put(Downloads.Impl.COLUMN_TITLE, displayName); 1289 values.put(Downloads.Impl._DATA, after.toString()); 1290 values.putNull(Downloads.Impl.COLUMN_MEDIAPROVIDER_URI); 1291 long[] ids = {id}; 1292 1293 return (mResolver.update(mBaseUri, values, getWhereClauseForIds(ids), 1294 getWhereArgsForIds(ids)) == 1); 1295 } 1296 1297 /** 1298 * Returns recommended maximum size, in bytes, of downloads that may go over a mobile 1299 * connection; or null if there's no recommended limit. The user will have the option to bypass 1300 * this limit. 1301 * 1302 * @param context the {@link Context} to use for accessing the {@link ContentResolver} 1303 * @return recommended maximum size, in bytes, of downloads that may go over a mobile 1304 * connection; or null if there's no recommended limit. 1305 */ 1306 public static Long getRecommendedMaxBytesOverMobile(Context context) { 1307 try { 1308 return Settings.Global.getLong(context.getContentResolver(), 1309 Settings.Global.DOWNLOAD_RECOMMENDED_MAX_BYTES_OVER_MOBILE); 1310 } catch (SettingNotFoundException exc) { 1311 return null; 1312 } 1313 } 1314 1315 /** {@hide} */ 1316 public static boolean isActiveNetworkExpensive(Context context) { 1317 // TODO: connect to NetworkPolicyManager 1318 return false; 1319 } 1320 1321 /** {@hide} */ 1322 public static long getActiveNetworkWarningBytes(Context context) { 1323 // TODO: connect to NetworkPolicyManager 1324 return -1; 1325 } 1326 1327 /** 1328 * Adds a file to the downloads database system, so it could appear in Downloads App 1329 * (and thus become eligible for management by the Downloads App). 1330 * <p> 1331 * It is helpful to make the file scannable by MediaScanner by setting the param 1332 * isMediaScannerScannable to true. It makes the file visible in media managing 1333 * applications such as Gallery App, which could be a useful purpose of using this API. 1334 * 1335 * @param title the title that would appear for this file in Downloads App. 1336 * @param description the description that would appear for this file in Downloads App. 1337 * @param isMediaScannerScannable true if the file is to be scanned by MediaScanner. Files 1338 * scanned by MediaScanner appear in the applications used to view media (for example, 1339 * Gallery app). 1340 * @param mimeType mimetype of the file. 1341 * @param path absolute pathname to the file. The file should be world-readable, so that it can 1342 * be managed by the Downloads App and any other app that is used to read it (for example, 1343 * Gallery app to display the file, if the file contents represent a video/image). 1344 * @param length length of the downloaded file 1345 * @param showNotification true if a notification is to be sent, false otherwise 1346 * @return an ID for the download entry added to the downloads app, unique across the system 1347 * This ID is used to make future calls related to this download. 1348 */ 1349 public long addCompletedDownload(String title, String description, 1350 boolean isMediaScannerScannable, String mimeType, String path, long length, 1351 boolean showNotification) { 1352 return addCompletedDownload(title, description, isMediaScannerScannable, mimeType, path, 1353 length, showNotification, false, null, null); 1354 } 1355 1356 /** 1357 * Adds a file to the downloads database system, so it could appear in Downloads App 1358 * (and thus become eligible for management by the Downloads App). 1359 * <p> 1360 * It is helpful to make the file scannable by MediaScanner by setting the param 1361 * isMediaScannerScannable to true. It makes the file visible in media managing 1362 * applications such as Gallery App, which could be a useful purpose of using this API. 1363 * 1364 * @param title the title that would appear for this file in Downloads App. 1365 * @param description the description that would appear for this file in Downloads App. 1366 * @param isMediaScannerScannable true if the file is to be scanned by MediaScanner. Files 1367 * scanned by MediaScanner appear in the applications used to view media (for example, 1368 * Gallery app). 1369 * @param mimeType mimetype of the file. 1370 * @param path absolute pathname to the file. The file should be world-readable, so that it can 1371 * be managed by the Downloads App and any other app that is used to read it (for example, 1372 * Gallery app to display the file, if the file contents represent a video/image). 1373 * @param length length of the downloaded file 1374 * @param showNotification true if a notification is to be sent, false otherwise 1375 * @param uri the original HTTP URI of the download 1376 * @param referer the HTTP Referer for the download 1377 * @return an ID for the download entry added to the downloads app, unique across the system 1378 * This ID is used to make future calls related to this download. 1379 */ 1380 public long addCompletedDownload(String title, String description, 1381 boolean isMediaScannerScannable, String mimeType, String path, long length, 1382 boolean showNotification, Uri uri, Uri referer) { 1383 return addCompletedDownload(title, description, isMediaScannerScannable, mimeType, path, 1384 length, showNotification, false, uri, referer); 1385 } 1386 1387 /** {@hide} */ 1388 public long addCompletedDownload(String title, String description, 1389 boolean isMediaScannerScannable, String mimeType, String path, long length, 1390 boolean showNotification, boolean allowWrite) { 1391 return addCompletedDownload(title, description, isMediaScannerScannable, mimeType, path, 1392 length, showNotification, allowWrite, null, null); 1393 } 1394 1395 /** {@hide} */ 1396 public long addCompletedDownload(String title, String description, 1397 boolean isMediaScannerScannable, String mimeType, String path, long length, 1398 boolean showNotification, boolean allowWrite, Uri uri, Uri referer) { 1399 // make sure the input args are non-null/non-zero 1400 validateArgumentIsNonEmpty("title", title); 1401 validateArgumentIsNonEmpty("description", description); 1402 validateArgumentIsNonEmpty("path", path); 1403 validateArgumentIsNonEmpty("mimeType", mimeType); 1404 if (length < 0) { 1405 throw new IllegalArgumentException(" invalid value for param: totalBytes"); 1406 } 1407 1408 // if there is already an entry with the given path name in downloads.db, return its id 1409 Request request; 1410 if (uri != null) { 1411 request = new Request(uri); 1412 } else { 1413 request = new Request(NON_DOWNLOADMANAGER_DOWNLOAD); 1414 } 1415 request.setTitle(title) 1416 .setDescription(description) 1417 .setMimeType(mimeType); 1418 if (referer != null) { 1419 request.addRequestHeader("Referer", referer.toString()); 1420 } 1421 ContentValues values = request.toContentValues(null); 1422 values.put(Downloads.Impl.COLUMN_DESTINATION, 1423 Downloads.Impl.DESTINATION_NON_DOWNLOADMANAGER_DOWNLOAD); 1424 values.put(Downloads.Impl._DATA, path); 1425 values.put(Downloads.Impl.COLUMN_STATUS, Downloads.Impl.STATUS_SUCCESS); 1426 values.put(Downloads.Impl.COLUMN_TOTAL_BYTES, length); 1427 values.put(Downloads.Impl.COLUMN_MEDIA_SCANNED, 1428 (isMediaScannerScannable) ? Request.SCANNABLE_VALUE_YES : 1429 Request.SCANNABLE_VALUE_NO); 1430 values.put(Downloads.Impl.COLUMN_VISIBILITY, (showNotification) ? 1431 Request.VISIBILITY_VISIBLE_NOTIFY_ONLY_COMPLETION : Request.VISIBILITY_HIDDEN); 1432 values.put(Downloads.Impl.COLUMN_ALLOW_WRITE, allowWrite ? 1 : 0); 1433 Uri downloadUri = mResolver.insert(Downloads.Impl.CONTENT_URI, values); 1434 if (downloadUri == null) { 1435 return -1; 1436 } 1437 return Long.parseLong(downloadUri.getLastPathSegment()); 1438 } 1439 1440 private static final String NON_DOWNLOADMANAGER_DOWNLOAD = 1441 "non-dwnldmngr-download-dont-retry2download"; 1442 1443 private static void validateArgumentIsNonEmpty(String paramName, String val) { 1444 if (TextUtils.isEmpty(val)) { 1445 throw new IllegalArgumentException(paramName + " can't be null"); 1446 } 1447 } 1448 1449 /** 1450 * Get the DownloadProvider URI for the download with the given ID. 1451 * 1452 * @hide 1453 */ 1454 public Uri getDownloadUri(long id) { 1455 return ContentUris.withAppendedId(Downloads.Impl.ALL_DOWNLOADS_CONTENT_URI, id); 1456 } 1457 1458 /** 1459 * Get a parameterized SQL WHERE clause to select a bunch of IDs. 1460 */ 1461 static String getWhereClauseForIds(long[] ids) { 1462 StringBuilder whereClause = new StringBuilder(); 1463 whereClause.append("("); 1464 for (int i = 0; i < ids.length; i++) { 1465 if (i > 0) { 1466 whereClause.append("OR "); 1467 } 1468 whereClause.append(Downloads.Impl._ID); 1469 whereClause.append(" = ? "); 1470 } 1471 whereClause.append(")"); 1472 return whereClause.toString(); 1473 } 1474 1475 /** 1476 * Get the selection args for a clause returned by {@link #getWhereClauseForIds(long[])}. 1477 */ 1478 static String[] getWhereArgsForIds(long[] ids) { 1479 String[] whereArgs = new String[ids.length]; 1480 return getWhereArgsForIds(ids, whereArgs); 1481 } 1482 1483 /** 1484 * Get selection args for a clause returned by {@link #getWhereClauseForIds(long[])} 1485 * and write it to the supplied args array. 1486 */ 1487 static String[] getWhereArgsForIds(long[] ids, String[] args) { 1488 assert(args.length >= ids.length); 1489 for (int i = 0; i < ids.length; i++) { 1490 args[i] = Long.toString(ids[i]); 1491 } 1492 return args; 1493 } 1494 1495 1496 /** 1497 * This class wraps a cursor returned by DownloadProvider -- the "underlying cursor" -- and 1498 * presents a different set of columns, those defined in the DownloadManager.COLUMN_* constants. 1499 * Some columns correspond directly to underlying values while others are computed from 1500 * underlying data. 1501 */ 1502 private static class CursorTranslator extends CursorWrapper { 1503 private final Uri mBaseUri; 1504 private final boolean mAccessFilename; 1505 1506 public CursorTranslator(Cursor cursor, Uri baseUri, boolean accessFilename) { 1507 super(cursor); 1508 mBaseUri = baseUri; 1509 mAccessFilename = accessFilename; 1510 } 1511 1512 @Override 1513 public int getInt(int columnIndex) { 1514 return (int) getLong(columnIndex); 1515 } 1516 1517 @Override 1518 public long getLong(int columnIndex) { 1519 if (getColumnName(columnIndex).equals(COLUMN_REASON)) { 1520 return getReason(super.getInt(getColumnIndex(Downloads.Impl.COLUMN_STATUS))); 1521 } else if (getColumnName(columnIndex).equals(COLUMN_STATUS)) { 1522 return translateStatus(super.getInt(getColumnIndex(Downloads.Impl.COLUMN_STATUS))); 1523 } else { 1524 return super.getLong(columnIndex); 1525 } 1526 } 1527 1528 @Override 1529 public String getString(int columnIndex) { 1530 final String columnName = getColumnName(columnIndex); 1531 switch (columnName) { 1532 case COLUMN_LOCAL_URI: 1533 return getLocalUri(); 1534 case COLUMN_LOCAL_FILENAME: 1535 if (!mAccessFilename) { 1536 throw new SecurityException( 1537 "COLUMN_LOCAL_FILENAME is deprecated;" 1538 + " use ContentResolver.openFileDescriptor() instead"); 1539 } 1540 default: 1541 return super.getString(columnIndex); 1542 } 1543 } 1544 1545 private String getLocalUri() { 1546 long destinationType = getLong(getColumnIndex(Downloads.Impl.COLUMN_DESTINATION)); 1547 if (destinationType == Downloads.Impl.DESTINATION_FILE_URI || 1548 destinationType == Downloads.Impl.DESTINATION_EXTERNAL || 1549 destinationType == Downloads.Impl.DESTINATION_NON_DOWNLOADMANAGER_DOWNLOAD) { 1550 String localPath = super.getString(getColumnIndex(COLUMN_LOCAL_FILENAME)); 1551 if (localPath == null) { 1552 return null; 1553 } 1554 return Uri.fromFile(new File(localPath)).toString(); 1555 } 1556 1557 // return content URI for cache download 1558 long downloadId = getLong(getColumnIndex(Downloads.Impl._ID)); 1559 return ContentUris.withAppendedId(Downloads.Impl.ALL_DOWNLOADS_CONTENT_URI, downloadId).toString(); 1560 } 1561 1562 private long getReason(int status) { 1563 switch (translateStatus(status)) { 1564 case STATUS_FAILED: 1565 return getErrorCode(status); 1566 1567 case STATUS_PAUSED: 1568 return getPausedReason(status); 1569 1570 default: 1571 return 0; // arbitrary value when status is not an error 1572 } 1573 } 1574 1575 private long getPausedReason(int status) { 1576 switch (status) { 1577 case Downloads.Impl.STATUS_WAITING_TO_RETRY: 1578 return PAUSED_WAITING_TO_RETRY; 1579 1580 case Downloads.Impl.STATUS_WAITING_FOR_NETWORK: 1581 return PAUSED_WAITING_FOR_NETWORK; 1582 1583 case Downloads.Impl.STATUS_QUEUED_FOR_WIFI: 1584 return PAUSED_QUEUED_FOR_WIFI; 1585 1586 default: 1587 return PAUSED_UNKNOWN; 1588 } 1589 } 1590 1591 private long getErrorCode(int status) { 1592 if ((400 <= status && status < Downloads.Impl.MIN_ARTIFICIAL_ERROR_STATUS) 1593 || (500 <= status && status < 600)) { 1594 // HTTP status code 1595 return status; 1596 } 1597 1598 switch (status) { 1599 case Downloads.Impl.STATUS_FILE_ERROR: 1600 return ERROR_FILE_ERROR; 1601 1602 case Downloads.Impl.STATUS_UNHANDLED_HTTP_CODE: 1603 case Downloads.Impl.STATUS_UNHANDLED_REDIRECT: 1604 return ERROR_UNHANDLED_HTTP_CODE; 1605 1606 case Downloads.Impl.STATUS_HTTP_DATA_ERROR: 1607 return ERROR_HTTP_DATA_ERROR; 1608 1609 case Downloads.Impl.STATUS_TOO_MANY_REDIRECTS: 1610 return ERROR_TOO_MANY_REDIRECTS; 1611 1612 case Downloads.Impl.STATUS_INSUFFICIENT_SPACE_ERROR: 1613 return ERROR_INSUFFICIENT_SPACE; 1614 1615 case Downloads.Impl.STATUS_DEVICE_NOT_FOUND_ERROR: 1616 return ERROR_DEVICE_NOT_FOUND; 1617 1618 case Downloads.Impl.STATUS_CANNOT_RESUME: 1619 return ERROR_CANNOT_RESUME; 1620 1621 case Downloads.Impl.STATUS_FILE_ALREADY_EXISTS_ERROR: 1622 return ERROR_FILE_ALREADY_EXISTS; 1623 1624 default: 1625 return ERROR_UNKNOWN; 1626 } 1627 } 1628 1629 private int translateStatus(int status) { 1630 switch (status) { 1631 case Downloads.Impl.STATUS_PENDING: 1632 return STATUS_PENDING; 1633 1634 case Downloads.Impl.STATUS_RUNNING: 1635 return STATUS_RUNNING; 1636 1637 case Downloads.Impl.STATUS_PAUSED_BY_APP: 1638 case Downloads.Impl.STATUS_WAITING_TO_RETRY: 1639 case Downloads.Impl.STATUS_WAITING_FOR_NETWORK: 1640 case Downloads.Impl.STATUS_QUEUED_FOR_WIFI: 1641 return STATUS_PAUSED; 1642 1643 case Downloads.Impl.STATUS_SUCCESS: 1644 return STATUS_SUCCESSFUL; 1645 1646 default: 1647 assert Downloads.Impl.isStatusError(status); 1648 return STATUS_FAILED; 1649 } 1650 } 1651 } 1652 } 1653