diff options
| author | Raphael Moll <ralf@android.com> | 2011-09-01 00:01:57 -0700 | 
|---|---|---|
| committer | Raphael <raphael@google.com> | 2011-09-01 22:48:53 -0700 | 
| commit | d7ab87816ba1da6b0cd8afecf15cd50f801e0f6d (patch) | |
| tree | d4ec7dd4f6c422eaf0e04323be39724386366c39 | |
| parent | f48ab3a9fb1d328024234510abc955dc365c9523 (diff) | |
| download | sdk-d7ab87816ba1da6b0cd8afecf15cd50f801e0f6d.zip sdk-d7ab87816ba1da6b0cd8afecf15cd50f801e0f6d.tar.gz sdk-d7ab87816ba1da6b0cd8afecf15cd50f801e0f6d.tar.bz2 | |
SDK Manager2: fix large downloads.
UrlOpener was using a BufferedHtmlEntity reader.
As a consequence its getContent() was loading large packages
in memory and resulting in OutOfMemory exceptions, whilst at
the same time blocking and thus rendering the download
progress report useless. This fixes it by simply passing
through in the underlying stream and freeing resources when
the stream is closed.
Change-Id: Ib3b265953931fecdf202197230ee9f24d8178faa
4 files changed, 39 insertions, 38 deletions
| diff --git a/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/AddonsListFetcher.java b/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/AddonsListFetcher.java index 6b0df67..cb8602f 100755 --- a/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/AddonsListFetcher.java +++ b/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/AddonsListFetcher.java @@ -196,7 +196,7 @@ public class AddonsListFetcher {              byte[] result = new byte[inc];
              try {
 -                is = UrlOpener.openURL(urlString, monitor);
 +                is = UrlOpener.openUrl(urlString, monitor);
                  int n;
                  while ((n = is.read(result, curr, result.length - curr)) != -1) {
 diff --git a/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/ArchiveInstaller.java b/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/ArchiveInstaller.java index 1eeea28..a06a88b 100755 --- a/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/ArchiveInstaller.java +++ b/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/ArchiveInstaller.java @@ -303,7 +303,7 @@ public class ArchiveInstaller {          FileOutputStream os = null;
          InputStream is = null;
          try {
 -            is = UrlOpener.openURL(urlString, monitor);
 +            is = UrlOpener.openUrl(urlString, monitor);
              os = new FileOutputStream(tmpFile);
              MessageDigest digester = archive.getChecksumType().getMessageDigest();
 diff --git a/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/SdkSource.java b/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/SdkSource.java index ed40d54..354b3aa 100755 --- a/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/SdkSource.java +++ b/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/SdkSource.java @@ -493,7 +493,7 @@ public abstract class SdkSource implements IDescription, Comparable<SdkSource> {              byte[] result = new byte[inc];
              try {
 -                is = UrlOpener.openURL(urlString, monitor);
 +                is = UrlOpener.openUrl(urlString, monitor);
                  int n;
                  while ((n = is.read(result, curr, result.length - curr)) != -1) {
 diff --git a/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/UrlOpener.java b/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/UrlOpener.java index 74f2b29..ed9dd39 100644 --- a/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/UrlOpener.java +++ b/sdkmanager/libs/sdklib/src/com/android/sdklib/internal/repository/UrlOpener.java @@ -27,12 +27,13 @@ import org.apache.http.auth.Credentials;  import org.apache.http.auth.UsernamePasswordCredentials;  import org.apache.http.client.methods.HttpGet;  import org.apache.http.client.protocol.ClientContext; -import org.apache.http.entity.BufferedHttpEntity;  import org.apache.http.impl.client.DefaultHttpClient;  import org.apache.http.impl.conn.ProxySelectorRoutePlanner;  import org.apache.http.protocol.BasicHttpContext;  import org.apache.http.protocol.HttpContext; +import java.io.FileNotFoundException; +import java.io.FilterInputStream;  import java.io.IOException;  import java.io.InputStream;  import java.net.ProxySelector; @@ -41,7 +42,7 @@ import java.util.Map;  /**   * This class holds methods for adding URLs management. - * @see #openURL(String, ITaskMonitor) + * @see #openUrl(String, ITaskMonitor)   */  public class UrlOpener { @@ -87,17 +88,14 @@ public class UrlOpener {       * @throws CanceledByUserException Exception thrown if the user cancels the       *              authentication dialog.       */ -    @SuppressWarnings("deprecation") -    static InputStream openURL(String url, ITaskMonitor monitor) +    static InputStream openUrl(String url, ITaskMonitor monitor)          throws IOException, CanceledByUserException { -        InputStream stream = null; -        HttpEntity entity = null;          Pair<String, String> result = null;          String realm = null;          // use the simple one -        DefaultHttpClient httpclient = new DefaultHttpClient(); +        final DefaultHttpClient httpClient = new DefaultHttpClient();          // create local execution context          HttpContext localContext = new BasicHttpContext(); @@ -105,15 +103,16 @@ public class UrlOpener {          // retrieve local java configured network in case there is the need to          // authenticate a proxy -        ProxySelectorRoutePlanner routePlanner = new ProxySelectorRoutePlanner(httpclient -                .getConnectionManager().getSchemeRegistry(), ProxySelector.getDefault()); -        httpclient.setRoutePlanner(routePlanner); +        ProxySelectorRoutePlanner routePlanner = new ProxySelectorRoutePlanner( +                    httpClient.getConnectionManager().getSchemeRegistry(), +                    ProxySelector.getDefault()); +        httpClient.setRoutePlanner(routePlanner);          boolean trying = true;          // loop while the response is being fetched          while (trying) {              // connect and get status code -            HttpResponse response = httpclient.execute(httpget, localContext); +            HttpResponse response = httpClient.execute(httpget, localContext);              int statusCode = response.getStatusLine().getStatusCode();              // check whether any authentication is required @@ -171,7 +170,7 @@ public class UrlOpener {                  // proceed in case there is indeed a user                  if (user != null && user.length() > 0) {                      Credentials credentials = new UsernamePasswordCredentials(user, password); -                    httpclient.getCredentialsProvider().setCredentials(authScope, credentials); +                    httpClient.getCredentialsProvider().setCredentials(authScope, credentials);                      trying = true;                  } else {                      trying = false; @@ -179,34 +178,36 @@ public class UrlOpener {              } else {                  trying = false;              } -            entity = response.getEntity(); + +            HttpEntity entity = response.getEntity(); +              if (entity != null) { -                // in case another pass to the Http Client will be performed, -                // consume the entity                  if (trying) { -                    entity.consumeContent(); -                } -                // since no pass to the Http Client is needed, retrieve the -                // entity's content -                else { -                    // Use a buffered entity because the stream in which it will -                    // be transfered, will not be closed later, unexpectedly - -                    // TODO: an unfortunate side effect is that creating the BufferedHttpEntity -                    // seems to perform the *actual* download (looking at it, there's a buffer -                    // being filled in there). So the caller doesn't have a chance to produce -                    // a meaningful callback with download speed/ETA stats. -                    // Behavior might be different with a slower network. - -                    BufferedHttpEntity bufferedEntity = new BufferedHttpEntity(entity); -                    stream = bufferedEntity.getContent(); +                    // in case another pass to the Http Client will be performed, close the entity. +                    entity.getContent().close(); +                } else { +                    // since no pass to the Http Client is needed, retrieve the +                    // entity's content. + +                    // Note: don't use something like a BufferedHttpEntity since it would consume +                    // all content and store it in memory, resulting in an OutOfMemory exception +                    // on a large download. + +                    return new FilterInputStream(entity.getContent()) { +                        @Override +                        public void close() throws IOException { +                            super.close(); + +                            // since Http Client is no longer needed, close it +                            httpClient.getConnectionManager().shutdown(); +                        } +                    };                  }              }          } -        // since Http Client is no longer needed, close it -        httpclient.getConnectionManager().shutdown(); - -        return stream; +        // We get here if we did not succeed. Callers do not expect a null result. +        httpClient.getConnectionManager().shutdown(); +        throw new FileNotFoundException(url);      }  } | 
