Commit 4359d443 by tonihei Committed by Oliver Woodman

Rename CronetEngineFactory to CronetEngineWrapper

In addition, the class now accepts available Cronet instances and returns the
source of the current CronetEngine.

-------------
Created by MOE: https://github.com/google/moe
MOE_MIGRATED_REVID=156185701
parent 1594e719
...@@ -43,7 +43,7 @@ public final class CronetDataSourceFactory extends BaseFactory { ...@@ -43,7 +43,7 @@ public final class CronetDataSourceFactory extends BaseFactory {
public static final int DEFAULT_READ_TIMEOUT_MILLIS = public static final int DEFAULT_READ_TIMEOUT_MILLIS =
CronetDataSource.DEFAULT_READ_TIMEOUT_MILLIS; CronetDataSource.DEFAULT_READ_TIMEOUT_MILLIS;
private final CronetEngineFactory cronetEngineFactory; private final CronetEngineWrapper cronetEngineWrapper;
private final Executor executor; private final Executor executor;
private final Predicate<String> contentTypePredicate; private final Predicate<String> contentTypePredicate;
private final TransferListener<? super DataSource> transferListener; private final TransferListener<? super DataSource> transferListener;
...@@ -55,14 +55,14 @@ public final class CronetDataSourceFactory extends BaseFactory { ...@@ -55,14 +55,14 @@ public final class CronetDataSourceFactory extends BaseFactory {
/** /**
* Constructs a CronetDataSourceFactory. * Constructs a CronetDataSourceFactory.
* <p> * <p>
* If the {@link CronetEngineFactory} fails to provide a suitable {@link CronetEngine}, the * If the {@link CronetEngineWrapper} fails to provide a {@link CronetEngine}, the provided
* provided fallback {@link HttpDataSource.Factory} will be used instead. * fallback {@link HttpDataSource.Factory} will be used instead.
* *
* Sets {@link CronetDataSource#DEFAULT_CONNECT_TIMEOUT_MILLIS} as the connection timeout, {@link * Sets {@link CronetDataSource#DEFAULT_CONNECT_TIMEOUT_MILLIS} as the connection timeout, {@link
* CronetDataSource#DEFAULT_READ_TIMEOUT_MILLIS} as the read timeout and disables * CronetDataSource#DEFAULT_READ_TIMEOUT_MILLIS} as the read timeout and disables
* cross-protocol redirects. * cross-protocol redirects.
* *
* @param cronetEngineFactory A {@link CronetEngineFactory}. * @param cronetEngineWrapper A {@link CronetEngineWrapper}.
* @param executor The {@link java.util.concurrent.Executor} that will perform the requests. * @param executor The {@link java.util.concurrent.Executor} that will perform the requests.
* @param contentTypePredicate An optional {@link Predicate}. If a content type is rejected by the * @param contentTypePredicate An optional {@link Predicate}. If a content type is rejected by the
* predicate then an {@link InvalidContentTypeException} is thrown from * predicate then an {@link InvalidContentTypeException} is thrown from
...@@ -71,25 +71,25 @@ public final class CronetDataSourceFactory extends BaseFactory { ...@@ -71,25 +71,25 @@ public final class CronetDataSourceFactory extends BaseFactory {
* @param fallbackFactory A {@link HttpDataSource.Factory} which is used as a fallback in case * @param fallbackFactory A {@link HttpDataSource.Factory} which is used as a fallback in case
* no suitable CronetEngine can be build. * no suitable CronetEngine can be build.
*/ */
public CronetDataSourceFactory(CronetEngineFactory cronetEngineFactory, public CronetDataSourceFactory(CronetEngineWrapper cronetEngineWrapper,
Executor executor, Predicate<String> contentTypePredicate, Executor executor, Predicate<String> contentTypePredicate,
TransferListener<? super DataSource> transferListener, TransferListener<? super DataSource> transferListener,
HttpDataSource.Factory fallbackFactory) { HttpDataSource.Factory fallbackFactory) {
this(cronetEngineFactory, executor, contentTypePredicate, transferListener, this(cronetEngineWrapper, executor, contentTypePredicate, transferListener,
DEFAULT_CONNECT_TIMEOUT_MILLIS, DEFAULT_READ_TIMEOUT_MILLIS, false, fallbackFactory); DEFAULT_CONNECT_TIMEOUT_MILLIS, DEFAULT_READ_TIMEOUT_MILLIS, false, fallbackFactory);
} }
/** /**
* Constructs a CronetDataSourceFactory. * Constructs a CronetDataSourceFactory.
* <p> * <p>
* If the {@link CronetEngineFactory} fails to provide a suitable {@link CronetEngine}, a * If the {@link CronetEngineWrapper} fails to provide a {@link CronetEngine}, a
* {@link DefaultHttpDataSourceFactory} will be used instead. * {@link DefaultHttpDataSourceFactory} will be used instead.
* *
* Sets {@link CronetDataSource#DEFAULT_CONNECT_TIMEOUT_MILLIS} as the connection timeout, {@link * Sets {@link CronetDataSource#DEFAULT_CONNECT_TIMEOUT_MILLIS} as the connection timeout, {@link
* CronetDataSource#DEFAULT_READ_TIMEOUT_MILLIS} as the read timeout and disables * CronetDataSource#DEFAULT_READ_TIMEOUT_MILLIS} as the read timeout and disables
* cross-protocol redirects. * cross-protocol redirects.
* *
* @param cronetEngineFactory A {@link CronetEngineFactory}. * @param cronetEngineWrapper A {@link CronetEngineWrapper}.
* @param executor The {@link java.util.concurrent.Executor} that will perform the requests. * @param executor The {@link java.util.concurrent.Executor} that will perform the requests.
* @param contentTypePredicate An optional {@link Predicate}. If a content type is rejected by the * @param contentTypePredicate An optional {@link Predicate}. If a content type is rejected by the
* predicate then an {@link InvalidContentTypeException} is thrown from * predicate then an {@link InvalidContentTypeException} is thrown from
...@@ -97,10 +97,10 @@ public final class CronetDataSourceFactory extends BaseFactory { ...@@ -97,10 +97,10 @@ public final class CronetDataSourceFactory extends BaseFactory {
* @param transferListener An optional listener. * @param transferListener An optional listener.
* @param userAgent A user agent used to create a fallback HttpDataSource if needed. * @param userAgent A user agent used to create a fallback HttpDataSource if needed.
*/ */
public CronetDataSourceFactory(CronetEngineFactory cronetEngineFactory, public CronetDataSourceFactory(CronetEngineWrapper cronetEngineWrapper,
Executor executor, Predicate<String> contentTypePredicate, Executor executor, Predicate<String> contentTypePredicate,
TransferListener<? super DataSource> transferListener, String userAgent) { TransferListener<? super DataSource> transferListener, String userAgent) {
this(cronetEngineFactory, executor, contentTypePredicate, transferListener, this(cronetEngineWrapper, executor, contentTypePredicate, transferListener,
DEFAULT_CONNECT_TIMEOUT_MILLIS, DEFAULT_READ_TIMEOUT_MILLIS, false, DEFAULT_CONNECT_TIMEOUT_MILLIS, DEFAULT_READ_TIMEOUT_MILLIS, false,
new DefaultHttpDataSourceFactory(userAgent, transferListener, new DefaultHttpDataSourceFactory(userAgent, transferListener,
DEFAULT_CONNECT_TIMEOUT_MILLIS, DEFAULT_READ_TIMEOUT_MILLIS, false)); DEFAULT_CONNECT_TIMEOUT_MILLIS, DEFAULT_READ_TIMEOUT_MILLIS, false));
...@@ -109,10 +109,10 @@ public final class CronetDataSourceFactory extends BaseFactory { ...@@ -109,10 +109,10 @@ public final class CronetDataSourceFactory extends BaseFactory {
/** /**
* Constructs a CronetDataSourceFactory. * Constructs a CronetDataSourceFactory.
* <p> * <p>
* If the {@link CronetEngineFactory} fails to provide a suitable {@link CronetEngine}, a * If the {@link CronetEngineWrapper} fails to provide a {@link CronetEngine}, a
* {@link DefaultHttpDataSourceFactory} will be used instead. * {@link DefaultHttpDataSourceFactory} will be used instead.
* *
* @param cronetEngineFactory A {@link CronetEngineFactory}. * @param cronetEngineWrapper A {@link CronetEngineWrapper}.
* @param executor The {@link java.util.concurrent.Executor} that will perform the requests. * @param executor The {@link java.util.concurrent.Executor} that will perform the requests.
* @param contentTypePredicate An optional {@link Predicate}. If a content type is rejected by the * @param contentTypePredicate An optional {@link Predicate}. If a content type is rejected by the
* predicate then an {@link InvalidContentTypeException} is thrown from * predicate then an {@link InvalidContentTypeException} is thrown from
...@@ -123,11 +123,11 @@ public final class CronetDataSourceFactory extends BaseFactory { ...@@ -123,11 +123,11 @@ public final class CronetDataSourceFactory extends BaseFactory {
* @param resetTimeoutOnRedirects Whether the connect timeout is reset when a redirect occurs. * @param resetTimeoutOnRedirects Whether the connect timeout is reset when a redirect occurs.
* @param userAgent A user agent used to create a fallback HttpDataSource if needed. * @param userAgent A user agent used to create a fallback HttpDataSource if needed.
*/ */
public CronetDataSourceFactory(CronetEngineFactory cronetEngineFactory, public CronetDataSourceFactory(CronetEngineWrapper cronetEngineWrapper,
Executor executor, Predicate<String> contentTypePredicate, Executor executor, Predicate<String> contentTypePredicate,
TransferListener<? super DataSource> transferListener, int connectTimeoutMs, TransferListener<? super DataSource> transferListener, int connectTimeoutMs,
int readTimeoutMs, boolean resetTimeoutOnRedirects, String userAgent) { int readTimeoutMs, boolean resetTimeoutOnRedirects, String userAgent) {
this(cronetEngineFactory, executor, contentTypePredicate, transferListener, this(cronetEngineWrapper, executor, contentTypePredicate, transferListener,
DEFAULT_CONNECT_TIMEOUT_MILLIS, DEFAULT_READ_TIMEOUT_MILLIS, resetTimeoutOnRedirects, DEFAULT_CONNECT_TIMEOUT_MILLIS, DEFAULT_READ_TIMEOUT_MILLIS, resetTimeoutOnRedirects,
new DefaultHttpDataSourceFactory(userAgent, transferListener, connectTimeoutMs, new DefaultHttpDataSourceFactory(userAgent, transferListener, connectTimeoutMs,
readTimeoutMs, resetTimeoutOnRedirects)); readTimeoutMs, resetTimeoutOnRedirects));
...@@ -136,10 +136,10 @@ public final class CronetDataSourceFactory extends BaseFactory { ...@@ -136,10 +136,10 @@ public final class CronetDataSourceFactory extends BaseFactory {
/** /**
* Constructs a CronetDataSourceFactory. * Constructs a CronetDataSourceFactory.
* <p> * <p>
* If the {@link CronetEngineFactory} fails to provide a suitable {@link CronetEngine}, the * If the {@link CronetEngineWrapper} fails to provide a {@link CronetEngine}, the provided
* provided fallback {@link HttpDataSource.Factory} will be used instead. * fallback {@link HttpDataSource.Factory} will be used instead.
* *
* @param cronetEngineFactory A {@link CronetEngineFactory}. * @param cronetEngineWrapper A {@link CronetEngineWrapper}.
* @param executor The {@link java.util.concurrent.Executor} that will perform the requests. * @param executor The {@link java.util.concurrent.Executor} that will perform the requests.
* @param contentTypePredicate An optional {@link Predicate}. If a content type is rejected by the * @param contentTypePredicate An optional {@link Predicate}. If a content type is rejected by the
* predicate then an {@link InvalidContentTypeException} is thrown from * predicate then an {@link InvalidContentTypeException} is thrown from
...@@ -151,12 +151,12 @@ public final class CronetDataSourceFactory extends BaseFactory { ...@@ -151,12 +151,12 @@ public final class CronetDataSourceFactory extends BaseFactory {
* @param fallbackFactory A {@link HttpDataSource.Factory} which is used as a fallback in case * @param fallbackFactory A {@link HttpDataSource.Factory} which is used as a fallback in case
* no suitable CronetEngine can be build. * no suitable CronetEngine can be build.
*/ */
public CronetDataSourceFactory(CronetEngineFactory cronetEngineFactory, public CronetDataSourceFactory(CronetEngineWrapper cronetEngineWrapper,
Executor executor, Predicate<String> contentTypePredicate, Executor executor, Predicate<String> contentTypePredicate,
TransferListener<? super DataSource> transferListener, int connectTimeoutMs, TransferListener<? super DataSource> transferListener, int connectTimeoutMs,
int readTimeoutMs, boolean resetTimeoutOnRedirects, int readTimeoutMs, boolean resetTimeoutOnRedirects,
HttpDataSource.Factory fallbackFactory) { HttpDataSource.Factory fallbackFactory) {
this.cronetEngineFactory = cronetEngineFactory; this.cronetEngineWrapper = cronetEngineWrapper;
this.executor = executor; this.executor = executor;
this.contentTypePredicate = contentTypePredicate; this.contentTypePredicate = contentTypePredicate;
this.transferListener = transferListener; this.transferListener = transferListener;
...@@ -169,13 +169,12 @@ public final class CronetDataSourceFactory extends BaseFactory { ...@@ -169,13 +169,12 @@ public final class CronetDataSourceFactory extends BaseFactory {
@Override @Override
protected HttpDataSource createDataSourceInternal(HttpDataSource.RequestProperties protected HttpDataSource createDataSourceInternal(HttpDataSource.RequestProperties
defaultRequestProperties) { defaultRequestProperties) {
CronetEngine cronetEngine = cronetEngineFactory.createCronetEngine(); CronetEngine cronetEngine = cronetEngineWrapper.getCronetEngine();
if (cronetEngine == null) { if (cronetEngine == null) {
return fallbackFactory.createDataSource(); return fallbackFactory.createDataSource();
} }
return new CronetDataSource(cronetEngineFactory.createCronetEngine(), executor, return new CronetDataSource(cronetEngine, executor, contentTypePredicate, transferListener,
contentTypePredicate, transferListener, connectTimeoutMs, readTimeoutMs, connectTimeoutMs, readTimeoutMs, resetTimeoutOnRedirects, defaultRequestProperties);
resetTimeoutOnRedirects, defaultRequestProperties);
} }
} }
...@@ -16,7 +16,10 @@ ...@@ -16,7 +16,10 @@
package com.google.android.exoplayer2.ext.cronet; package com.google.android.exoplayer2.ext.cronet;
import android.content.Context; import android.content.Context;
import android.support.annotation.IntDef;
import android.util.Log; import android.util.Log;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.reflect.Field; import java.lang.reflect.Field;
import java.util.Collections; import java.util.Collections;
import java.util.Comparator; import java.util.Comparator;
...@@ -25,70 +28,127 @@ import org.chromium.net.CronetEngine; ...@@ -25,70 +28,127 @@ import org.chromium.net.CronetEngine;
import org.chromium.net.CronetProvider; import org.chromium.net.CronetProvider;
/** /**
* A factory class which creates or reuses a {@link CronetEngine}. * A wrapper class for a {@link CronetEngine}.
*/ */
public final class CronetEngineFactory { public final class CronetEngineWrapper {
private static final String TAG = "CronetEngineFactory"; private static final String TAG = "CronetEngineWrapper";
private final Context context; private final CronetEngine cronetEngine;
private final boolean preferGMSCoreCronet; private final @CronetEngineSource int cronetEngineSource;
private CronetEngine cronetEngine = null;
/** /**
* Creates the factory for a {@link CronetEngine}. Sets factory to prefer natively bundled Cronet * Source of {@link CronetEngine}.
* over GMSCore Cronet if both are available.
*
* @param context A context.
*/ */
public CronetEngineFactory(Context context) { @Retention(RetentionPolicy.SOURCE)
this(context, false); @IntDef({SOURCE_NATIVE, SOURCE_GMS, SOURCE_UNKNOWN, SOURCE_USER_PROVIDED, SOURCE_UNAVAILABLE})
} public @interface CronetEngineSource {}
/**
* Natively bundled Cronet implementation.
*/
public static final int SOURCE_NATIVE = 0;
/**
* Cronet implementation from GMSCore.
*/
public static final int SOURCE_GMS = 1;
/**
* Other (unknown) Cronet implementation.
*/
public static final int SOURCE_UNKNOWN = 2;
/**
* User-provided Cronet engine.
*/
public static final int SOURCE_USER_PROVIDED = 3;
/**
* No Cronet implementation available. Fallback Http provider is used if possible.
*/
public static final int SOURCE_UNAVAILABLE = 4;
/** /**
* Creates the factory for a {@link CronetEngine} and specifies whether Cronet from GMSCore should * Creates a wrapper for a {@link CronetEngine} which automatically selects the most suitable
* be preferred over natively bundled Cronet if both are available. * {@link CronetProvider}. Sets wrapper to prefer natively bundled Cronet over GMSCore Cronet
* if both are available.
* *
* @param context A context. * @param context A context.
*/ */
public CronetEngineFactory(Context context, boolean preferGMSCoreCronet) { public CronetEngineWrapper(Context context) {
this.context = context.getApplicationContext(); this(context, false);
this.preferGMSCoreCronet = preferGMSCoreCronet;
} }
/** /**
* Create or reuse a {@link CronetEngine}. If no CronetEngine is available, the method returns * Creates a wrapper for a {@link CronetEngine} which automatically selects the most suitable
* null. * {@link CronetProvider} based on user preference.
* *
* @return The CronetEngine, or null if no CronetEngine is available. * @param context A context.
* @param preferGMSCoreCronet Whether Cronet from GMSCore should be preferred over natively
* bundled Cronet if both are available.
*/ */
/* package */ CronetEngine createCronetEngine() { public CronetEngineWrapper(Context context, boolean preferGMSCoreCronet) {
if (cronetEngine == null) { CronetEngine cronetEngine = null;
List<CronetProvider> cronetProviders = CronetProvider.getAllProviders(context); @CronetEngineSource int cronetEngineSource = SOURCE_UNAVAILABLE;
// Remove disabled and fallback Cronet providers from list List<CronetProvider> cronetProviders = CronetProvider.getAllProviders(context);
for (int i = cronetProviders.size() - 1; i >= 0; i--) { // Remove disabled and fallback Cronet providers from list
if (!cronetProviders.get(i).isEnabled() for (int i = cronetProviders.size() - 1; i >= 0; i--) {
|| CronetProvider.PROVIDER_NAME_FALLBACK.equals(cronetProviders.get(i).getName())) { if (!cronetProviders.get(i).isEnabled()
cronetProviders.remove(i); || CronetProvider.PROVIDER_NAME_FALLBACK.equals(cronetProviders.get(i).getName())) {
} cronetProviders.remove(i);
} }
// Sort remaining providers by type and version. }
Collections.sort(cronetProviders, new CronetProviderComparator(preferGMSCoreCronet)); // Sort remaining providers by type and version.
for (int i = 0; i < cronetProviders.size(); i++) { CronetProviderComparator providerComparator = new CronetProviderComparator(preferGMSCoreCronet);
String providerName = cronetProviders.get(i).getName(); Collections.sort(cronetProviders, providerComparator);
try { for (int i = 0; i < cronetProviders.size() && cronetEngine == null; i++) {
cronetEngine = cronetProviders.get(i).createBuilder().build(); String providerName = cronetProviders.get(i).getName();
Log.d(TAG, "CronetEngine built using " + providerName); try {
} catch (UnsatisfiedLinkError e) { cronetEngine = cronetProviders.get(i).createBuilder().build();
Log.w(TAG, "Failed to link Cronet binaries. Please check if native Cronet binaries are " if (providerComparator.isNativeProvider(providerName)) {
+ "bundled into your app."); cronetEngineSource = SOURCE_NATIVE;
} else if (providerComparator.isGMSCoreProvider(providerName)) {
cronetEngineSource = SOURCE_GMS;
} else {
cronetEngineSource = SOURCE_UNKNOWN;
} }
Log.d(TAG, "CronetEngine built using " + providerName);
} catch (SecurityException e) {
Log.w(TAG, "Failed to build CronetEngine. Please check if current process has "
+ "android.permission.ACCESS_NETWORK_STATE.");
} catch (UnsatisfiedLinkError e) {
Log.w(TAG, "Failed to link Cronet binaries. Please check if native Cronet binaries are "
+ "bundled into your app.");
} }
} }
if (cronetEngine == null) { if (cronetEngine == null) {
Log.w(TAG, "Cronet not available. Using fallback provider."); Log.w(TAG, "Cronet not available. Using fallback provider.");
} }
this.cronetEngine = cronetEngine;
this.cronetEngineSource = cronetEngineSource;
}
/**
* Creates a wrapper for an existing CronetEngine.
*
* @param cronetEngine An existing CronetEngine.
*/
public CronetEngineWrapper(CronetEngine cronetEngine) {
this.cronetEngine = cronetEngine;
this.cronetEngineSource = SOURCE_USER_PROVIDED;
}
/**
* Returns the source of the wrapped {@link CronetEngine}.
*
* @return A {@link CronetEngineSource} value.
*/
public @CronetEngineSource int getCronetEngineSource() {
return cronetEngineSource;
}
/**
* Returns the wrapped {@link CronetEngine}.
*
* @return The CronetEngine, or null if no CronetEngine is available.
*/
/* package */ CronetEngine getCronetEngine() {
return cronetEngine; return cronetEngine;
} }
...@@ -127,15 +187,23 @@ public final class CronetEngineFactory { ...@@ -127,15 +187,23 @@ public final class CronetEngineFactory {
return -compareVersionStrings(providerLeft.getVersion(), providerRight.getVersion()); return -compareVersionStrings(providerLeft.getVersion(), providerRight.getVersion());
} }
public boolean isNativeProvider(String providerName) {
return CronetProvider.PROVIDER_NAME_APP_PACKAGED.equals(providerName);
}
public boolean isGMSCoreProvider(String providerName) {
return gmsCoreCronetName != null && gmsCoreCronetName.equals(providerName);
}
/** /**
* Convert Cronet provider name into a sortable preference value. * Convert Cronet provider name into a sortable preference value.
* Smaller values are preferred. * Smaller values are preferred.
*/ */
private int evaluateCronetProviderType(String providerName) { private int evaluateCronetProviderType(String providerName) {
if (CronetProvider.PROVIDER_NAME_APP_PACKAGED.equals(providerName)) { if (isNativeProvider(providerName)) {
return 1; return 1;
} }
if (gmsCoreCronetName != null && gmsCoreCronetName.equals(providerName)) { if (isGMSCoreProvider(providerName)) {
return preferGMSCoreCronet ? 0 : 2; return preferGMSCoreCronet ? 0 : 2;
} }
// Unknown provider type. // Unknown provider type.
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or sign in to comment