大規模なプロジェクトの一部として、Android ライブラリ (と呼ばれるapi) gradle モジュールがあります。プロジェクト全体を AndroidX に移行しました。apiライブラリでインストルメンテーション テストを実行すると、次のエラーが発生します。
Task :api:checkDebugAndroidTestDuplicateClasses FAILED
FAILURE: Build failed with an exception.
* What went wrong:
Execution failed for task ':api:checkDebugAndroidTestDuplicateClasses'.
> 1 exception was raised by workers:
java.lang.RuntimeException: java.lang.RuntimeException: Duplicate class com.google.common.util.concurrent.ListenableFuture found in modules jetified-guava-25.1-android.jar (com.google.guava:guava:25.1-android) and listenablefuture-1.0.jar (com.google.guava:listenablefuture:1.0)
debugAndroidTestバリアントのランタイム クラス パスを確認すると、次のようになります。
./gradlew api:dependencies --configuration debugAndroidTestRuntimeClasspath | grep --color -E "guava|$"
この出力が得られます。私は問題を見ることができます:
------------------------------------------------------------
Project :api
------------------------------------------------------------
debugAndroidTestRuntimeClasspath - Resolved configuration for runtime for variant: debugAndroidTest
+--- project :test_utils
| +--- project :core
...
| +--- project :api (*)
| +--- com.google.android.material:material:1.1.0-alpha03
| | +--- androidx.annotation:annotation:1.0.1 -> 1.1.0-alpha01
| | +--- androidx.appcompat:appcompat:1.1.0-alpha01
| | | +--- androidx.annotation:annotation:1.0.0 -> 1.1.0-alpha01
| | | +--- androidx.core:core:1.1.0-alpha01 -> 1.1.0-alpha03
| | | | +--- com.google.guava:listenablefuture:1.0 // <------ GUAVA
| | | | +--- androidx.annotation:annotation:1.0.1 -> 1.1.0-alpha01
...
+--- com.google.truth:truth:0.42
| +--- com.google.guava:guava:25.1-android / <------ MORE GUAVA
| | +--- com.google.code.findbugs:jsr305:3.0.2
| | +--- com.google.errorprone:error_prone_annotations:2.1.3 -> 2.3.1
| | +--- com.google.j2objc:j2objc-annotations:1.1
| | \--- org.codehaus.mojo:animal-sniffer-annotations:1.14
| +--- org.checkerframework:checker-compat-qual:2.5.3
| +--- org.checkerframework:checker-qual:2.5.3
| +--- junit:junit:4.12 (*)
| +--- com.googlecode.java-diff-utils:diffutils:1.3.0
| +--- com.google.auto.value:auto-value-annotations:1.6.2
| \--- com.google.errorprone:error_prone_annotations:2.3.1
...
AndroidX コアはグアバの新しい「ListableFuture-only」ビルドに依存しており、Truth は完全な Guava 25 に依存しています。
ListenableFuture の根本的な問題を理解していると思います: https://groups.google.com/forum/#!topic/guava-announce/Km82fZG68Sw
ここで正しい解決策は何ですか?
Guava を Truth から完全に除外したくありません (そうしないと、Truth はコンパイルされません)
androidTestImplementation("com.google.truth:truth:0.42") {
exclude group: 'com.google.guava', module: 'guava'
}
最初のレベルの依存関係にすることで、Guava 27 への更新を除外 + 強制することはできますか?
androidTestImplementation("com.google.truth:truth:$rootProject.ext.truthVersion") {
exclude group: 'com.google.guava', module: 'guava'
}
// must add guava as top level dependency to force Truth to use latest version
androidTestImplementation 'com.google.guava:guava:27.0.1-android'
これを行う場合、アンドロイドまたは JRE バージョンのグアバを使用する必要がありますか?
副次的な質問:
compileクラスパスを見たときにグアバの依存関係が表示されないのはなぜですか? エラーは実行時エラーではなく、コンパイル時エラーです
./gradlew api:dependencies --configuration debugAndroidTestCompileClasspath | grep --color -E "guava|$"
結果の deps:
debugAndroidTestCompileClasspath - Resolved configuration for compilation for variant: debugAndroidTest
+--- project :test_utils // <----------- why are test_utils deps not listed here???
...
+--- com.google.truth:truth:0.42
| +--- com.google.guava:guava:25.1-android <------ GUAVA
| | +--- com.google.code.findbugs:jsr305:3.0.2
| | +--- org.checkerframework:checker-compat-qual:2.0.0 -> 2.5.3
| | +--- com.google.errorprone:error_prone_annotations:2.1.3 -> 2.3.1
| | +--- com.google.j2objc:j2objc-annotations:1.1
| | \--- org.codehaus.mojo:animal-sniffer-annotations:1.14
| +--- org.checkerframework:checker-compat-qual:2.5.3
| +--- org.checkerframework:checker-qual:2.5.3
| +--- junit:junit:4.12 (*)
| +--- com.googlecode.java-diff-utils:diffutils:1.3.0
| +--- com.google.auto.value:auto-value-annotations:1.6.2
| \--- com.google.errorprone:error_prone_annotations:2.3.1
...
+--- com.google.truth:truth:{strictly 0.42} -> 0.42 (c)
+--- com.google.guava:guava:{strictly 25.1-android} -> 25.1-android (c) // <--------- why is this listed again here at top level?
アップデート:
Truth 0.43 にアップデートした後、次のエラーが表示されます。
> Could not resolve all artifacts for configuration ':mymodule:debugAndroidTestRuntimeClasspath'.
> Could not resolve com.google.guava:listenablefuture:{strictly 1.0}.
Required by:
project :mymodule
> Cannot find a version of 'com.google.guava:listenablefuture' that satisfies the version constraints:
Dependency path 'example:mymodule:unspecified' --> 'com.google.truth:truth:0.43' --> 'com.google.guava:guava:27.0.1-android' --> 'com.google.guava:listenablefuture:9999.0-empty-to-avoid-conflict-with-guava'
Constraint path 'example:mymodule:unspecified' --> 'com.google.guava:listenablefuture:{strictly 1.0}' because of the following reason: debugRuntimeClasspath uses version 1.0
Dependency path 'example:mymodule:unspecified' --> 'example:myothermodule:unspecified' --> 'andrexampleoidx.core:core:1.1.0-alpha04' --> 'com.google.guava:listenablefuture:1.0'
Constraint path 'example:mymodule:unspecified' --> 'com.google.guava:listenablefuture:{strictly 1.0}' because of the following reason: debugRuntimeClasspath uses version 1.0
...
Dependency path 'example:mymodule:unspecified' --> 'example:myothermodule:unspecified' --> 'androidx.core:core:1.1.0-alpha04' --> 'androidx.concurrent:concurrent-futures:1.0.0-alpha02' --> 'com.google.guava:listenablefuture:1.0'
Constraint path 'example:mymodule:unspecified' --> 'com.google.guava:listenablefuture:{strictly 1.0}' because of the following reason: debugRuntimeClasspath uses version 1.0
...
> Could not resolve com.google.guava:listenablefuture:9999.0-empty-to-avoid-conflict-with-guava.
Required by:
project :mymodule > com.google.truth:truth:0.43 > com.google.guava:guava:27.0.1-android
> Cannot find a version of 'com.google.guava:listenablefuture' that satisfies the version constraints:
Dependency path 'example:mymodule:unspecified' --> 'com.google.truth:truth:0.43' --> 'com.google.guava:guava:27.0.1-android' --> 'com.google.guava:listenablefuture:9999.0-empty-to-avoid-conflict-with-guava'
Constraint path 'example:mymodule:unspecified' --> 'com.google.guava:listenablefuture:{strictly 1.0}' because of the following reason: debugRuntimeClasspath uses version 1.0
Dependency path 'example:mymodule:unspecified' --> 'example:myothermodule:unspecified' --> 'androidx.core:core:1.1.0-alpha04' --> 'com.google.guava:listenablefuture:1.0'
...
Dependency path 'example:mymodule:unspecified' --> 'example:myothermodule:unspecified' --> 'androidx.core:core:1.1.0-alpha04' --> 'androidx.concurrent:concurrent-futures:1.0.0-alpha02' --> 'com.google.guava:listenablefuture:1.0'
Constraint path 'example:mymodule:unspecified' --> 'com.google.guava:listenablefuture:{strictly 1.0}' because of the following reason: debugRuntimeClasspath uses version 1.0
...
> Could not resolve com.google.guava:listenablefuture:1.0.
Required by:
project :mymodule > androidx.core:core:1.1.0-alpha04
project :mymodule > androidx.concurrent:concurrent-futures:1.0.0-alpha02
> Cannot find a version of 'com.google.guava:listenablefuture' that satisfies the version constraints:
Dependency path 'example:mymodule:unspecified' --> 'com.google.truth:truth:0.43' --> 'com.google.guava:guava:27.0.1-android' --> 'com.google.guava:listenablefuture:9999.0-empty-to-avoid-conflict-with-guava'
Constraint path 'example:mymodule:unspecified' --> 'com.google.guava:listenablefuture:{strictly 1.0}' because of the following reason: debugRuntimeClasspath uses version 1.0
...
Dependency path 'example:mymodule:unspecified' --> 'example:myothermodule:unspecified' --> 'androidx.core:core:1.1.0-alpha04' --> 'androidx.concurrent:concurrent-futures:1.0.0-alpha02' --> 'com.google.guava:listenablefuture:1.0'
Constraint path 'example:mymodule:unspecified' --> 'com.google.guava:listenablefuture:{strictly 1.0}' because of the following reason: debugRuntimeClasspath uses version 1.0
...
私が正しく理解していれば:
- Truth 0.43 は Guava 27 に依存します
- Guava 27 は、バージョン 9999.0-emptyのコピーが内部
9999.0-emptyに含まれているため、依存しています。ListenableFuture - AndroidXコアのような他のライブラリが依存しています
listenablefuture:1.0 - 競合を解決する場合 (バージョン 1.0 または 9999.0 を使用する必要がありますか?)、Gradle は最新バージョン (この場合は 9999.0) を選択します。
- したがって、AndroidX コアは実際
ListenableFutureにはGuava を介してクラスパスから取得していますが、賢明ではありません。 - ただし、
strictlyキーワードは 1.0 の使用を強制しています - これは、Android Gradle プラグインが実行時とコンパイル時のクラスパスを強制的に同じバージョンに依存させるためです =(詳細はこちら
そのため、今何をすべきか悩んでいるのが現状です。