1 | // Copyright 2014 The Flutter Authors. All rights reserved. |
2 | // Use of this source code is governed by a BSD-style license that can be |
3 | // found in the LICENSE file. |
4 | |
5 | import 'package:meta/meta.dart' ; |
6 | import 'package:process/process.dart' ; |
7 | import 'package:unified_analytics/unified_analytics.dart' ; |
8 | |
9 | import '../base/common.dart'; |
10 | import '../base/file_system.dart'; |
11 | import '../base/io.dart'; |
12 | import '../base/logger.dart'; |
13 | import '../base/os.dart'; |
14 | import '../base/platform.dart'; |
15 | import '../base/utils.dart'; |
16 | import '../base/version.dart'; |
17 | import '../base/version_range.dart'; |
18 | import '../build_info.dart'; |
19 | import '../cache.dart'; |
20 | import '../globals.dart' as globals; |
21 | import '../project.dart'; |
22 | import 'android_sdk.dart'; |
23 | |
24 | // These are the versions used in the project templates. |
25 | // |
26 | // In general, Flutter aims to default to the latest stable version. |
27 | // However, this currently requires to migrate existing integration tests to the |
28 | // latest supported values. |
29 | // |
30 | // Please see the README before changing any of these values. |
31 | |
32 | // See https://gradle.org/releases |
33 | const templateDefaultGradleVersion = '8.12' ; |
34 | |
35 | // When bumping, also update: |
36 | // * ndkVersion constant in this file |
37 | // * ndkVersion in FlutterExtension in packages/flutter_tools/gradle/src/main/kotlin/FlutterExtension.kt |
38 | // * AGP version constants in packages/flutter_tools/gradle/build.gradle.kts |
39 | // * AGP test constants in packages/flutter_tools/gradle/src/test/kotlin/DependencyVersionCheckerTest.kt |
40 | // See https://mvnrepository.com/artifact/com.android.tools.build/gradle |
41 | const templateAndroidGradlePluginVersion = '8.9.1' ; |
42 | const templateAndroidGradlePluginVersionForModule = '8.9.1' ; |
43 | |
44 | // See https://kotlinlang.org/docs/releases.html#release-details |
45 | const templateKotlinGradlePluginVersion = '2.1.0' ; |
46 | |
47 | // The Flutter Gradle Plugin is only applied to app projects, and modules that |
48 | // are built from source using (`include_flutter.groovy`). The remaining |
49 | // projects are: plugins, and modules compiled as AARs. In modules, the |
50 | // ephemeral directory `.android` is always regenerated after `flutter pub get`, |
51 | // so new versions are picked up after a Flutter upgrade. |
52 | // |
53 | // Please see the README before changing any of these values. |
54 | const compileSdkVersion = '36' ; |
55 | const minSdkVersion = '24' ; |
56 | const targetSdkVersion = '36' ; |
57 | const ndkVersion = '27.0.12077973' ; |
58 | |
59 | // Update these when new major versions of Java are supported by new Gradle |
60 | // versions that we support. |
61 | // Source of truth: https://docs.gradle.org/current/userguide/compatibility.html |
62 | const oneMajorVersionHigherJavaVersion = '24' ; |
63 | |
64 | // Update this when new versions of Gradle come out including minor versions |
65 | // and should correspond to the maximum Gradle version we test in CI. |
66 | // |
67 | // Supported here means supported by the tooling for |
68 | // flutter analyze --suggestions and does not imply broader flutter support. |
69 | const maxKnownAndSupportedGradleVersion = '8.12' ; |
70 | |
71 | // Update this with new KGP versions come out including minor versions. |
72 | // |
73 | // Supported here means supported by the tooling for |
74 | // flutter analyze --suggestions and does not imply broader flutter support. |
75 | const maxKnownAndSupportedKgpVersion = '2.1.20' ; |
76 | |
77 | // Update this when new versions of AGP come out. |
78 | // |
79 | // Supported here means tooling is aware of this version's Java <-> AGP |
80 | // compatibility. |
81 | @visibleForTesting |
82 | const maxKnownAndSupportedAgpVersion = '8.9.1' ; |
83 | |
84 | // Update this when new versions of AGP come out. |
85 | const maxKnownAgpVersion = '8.9.1' ; |
86 | |
87 | // Supported here means tooling is aware of this versions |
88 | // Java <-> AGP compatibility and does not imply broader flutter support. |
89 | // For use in flutter see the code in: |
90 | // flutter_tools/gradle/src/main/kotlin/DependencyVersionChecker.kt |
91 | @visibleForTesting |
92 | const oldestConsideredAgpVersion = '3.3.0' ; |
93 | |
94 | // Supported here means tooling is aware of this versions |
95 | // gradle compatibility and does not imply broader flutter support. |
96 | @visibleForTesting |
97 | const oldestConsideredGradleVersion = '4.10.1' ; |
98 | |
99 | // Supported here means tooling is aware of this versions |
100 | // gradle/AGP compatibility and does not imply broader flutter support. |
101 | @visibleForTesting |
102 | const oldestDocumentedKgpCompatabilityVersion = '1.6.20' ; |
103 | |
104 | // Oldest documented version of AGP that has a listed minimum |
105 | // compatible Java version. |
106 | const oldestDocumentedJavaAgpCompatibilityVersion = '4.2' ; |
107 | |
108 | // Constant used in [_buildAndroidGradlePluginRegExp] and |
109 | // [_settingsAndroidGradlePluginRegExp] and [_kotlinGradlePluginRegExpFromId] |
110 | // to identify the version section. |
111 | const _versionGroupName = 'version' ; |
112 | |
113 | // AGP can be defined in the dependencies block of [build.gradle] or [build.gradle.kts]. |
114 | // Expected content (covers both classpath and compileOnly cases): |
115 | // Groovy DSL with single quotes - 'com.android.tools.build:gradle:{{agpVersion}}' |
116 | // Groovy DSL with double quotes - "com.android.tools.build:gradle:{{agpVersion}}" |
117 | // Kotlin DSL - ("com.android.tools.build.gradle:{{agpVersion}}") |
118 | // ? is used to name the version group which helps with extraction. |
119 | final _androidGradlePluginRegExpFromDependencies = RegExp( |
120 | r"""[^\/]*\s*((\bclasspath\b)|(\bcompileOnly\b))\s*\(?['"]com\.android\.tools\.build:gradle:(?<version>\d+(\.\d+){1,2})\)?""" , |
121 | multiLine: true, |
122 | ); |
123 | |
124 | // AGP can be defined in the plugins block of [build.gradle], |
125 | // [build.gradle.kts], [settings.gradle], or [settings.gradle.kts]. |
126 | // Expected content: |
127 | // Groovy DSL with single quotes - id 'com.android.application' version '{{agpVersion}}' |
128 | // Groovy DSL with double quotes - id "com.android.application" version "{{agpVersion}}" |
129 | // Kotlin DSL - id("com.android.application") version "{{agpVersion}}" |
130 | // ? is used to name the version group which helps with extraction. |
131 | final _androidGradlePluginRegExpFromId = RegExp( |
132 | r"""[^\/]*s*id\s*\(?['"]com\.android\.application['"]\)?\s+version\s+['"](?<version>\d+(\.\d+){1,2})\)?""" , |
133 | multiLine: true, |
134 | ); |
135 | |
136 | // KGP is defined in several places this code only checks in plugins block |
137 | // of [settings.gradle] and [settings.gradle.kts]. |
138 | // Expected content: |
139 | // Groovy DSL - id "org.jetbrains.kotlin.android" version "{{kgpVersion}}" |
140 | // Kotlin DSL - id("org.jetbrains.kotlin.android") version "{{kgpVersion}}" |
141 | // ? is used to name the version group which helps with extraction. |
142 | final _kotlinGradlePluginRegExpFromId = RegExp( |
143 | r"""[^\/]*s*id\s*\(?['"]org\.jetbrains\.kotlin\.android['"]\)?\s+version\s+['"](?<version>\d+(\.\d+){1,2})\)?""" , |
144 | multiLine: true, |
145 | ); |
146 | |
147 | // Expected content format (with lines above and below). |
148 | // Version can have 2 or 3 numbers. |
149 | // 'distributionUrl=https\://services.gradle.org/distributions/gradle-7.4.2-all.zip' |
150 | // '^\s*' protects against commented out lines. |
151 | final distributionUrlRegex = RegExp(r'^\s*distributionUrl\s*=\s*.*\.zip' , multiLine: true); |
152 | |
153 | // Modified version of the gradle distribution url match designed to only match |
154 | // gradle.org urls so that we can guarantee any modifications to the url |
155 | // still points to a hosted zip. |
156 | final gradleOrgVersionMatch = RegExp( |
157 | r'^\s*distributionUrl\s*=\s*https\\://services\.gradle\.org/distributions/gradle-((?:\d|\.)+)-(.*)\.zip', |
158 | multiLine: true, |
159 | ); |
160 | |
161 | // This matches uncommented minSdkVersion lines in the module-level build.gradle |
162 | // file which have minSdkVersion 16, 17, 18, 19, 20, 21, 22, 23 set with space sytax, |
163 | // equals syntax and when using minSdk or minSdkVersion. |
164 | final tooOldMinSdkVersionMatch = RegExp( |
165 | r'(?<=^\s*)minSdk(Version)?\s=?\s?(1[6789]|2[0123])(?=\s*(?://|$))', |
166 | multiLine: true,
|
167 | );
|
168 |
|
169 | // From https://docs.gradle.org/current/userguide/command_line_interface.html#command_line_interface
|
170 | // Flag to print the versions for gradle, kotlin dsl, groovy, etc.
|
171 | const gradleVersionsFlag = r'--version' ;
|
172 |
|
173 | // Directory under android/ that gradle uses to store gradle information.
|
174 | // Regularly used with [gradleWrapperDirectory] and
|
175 | // [gradleWrapperPropertiesFilename].
|
176 | // Different from the directory of gradle files stored in
|
177 | // `_cache.getArtifactDirectory('gradle_wrapper')`
|
178 | const gradleDirectoryName = 'gradle' ;
|
179 | const gradleWrapperDirectoryName = 'wrapper' ;
|
180 | const gradleWrapperPropertiesFilename = 'gradle-wrapper.properties' ;
|
181 |
|
182 | /// Provides utilities to run a Gradle task, such as finding the Gradle executable
|
183 | /// or constructing a Gradle project.
|
184 | class GradleUtils {
|
185 | GradleUtils({
|
186 | required Platform platform,
|
187 | required Logger logger,
|
188 | required Cache cache,
|
189 | required OperatingSystemUtils operatingSystemUtils,
|
190 | }) : _platform = platform,
|
191 | _logger = logger,
|
192 | _cache = cache,
|
193 | _operatingSystemUtils = operatingSystemUtils;
|
194 |
|
195 | final Cache _cache;
|
196 | final Platform _platform;
|
197 | final Logger _logger;
|
198 | final OperatingSystemUtils _operatingSystemUtils;
|
199 |
|
200 | /// Gets the Gradle executable path and prepares the Gradle project.
|
201 | /// This is the `gradlew` or `gradlew.bat` script in the `android/` directory.
|
202 | String getExecutable(FlutterProject project) {
|
203 | final Directory androidDir = project.android.hostAppGradleRoot;
|
204 | injectGradleWrapperIfNeeded(androidDir);
|
205 |
|
206 | final File gradle = androidDir.childFile(getGradlewFileName(_platform));
|
207 |
|
208 | if (gradle.existsSync()) {
|
209 | _logger.printTrace('Using gradle from ${gradle.absolute.path}.' );
|
210 | // If the Gradle executable doesn't have execute permission,
|
211 | // then attempt to set it.
|
212 | _operatingSystemUtils.makeExecutable(gradle);
|
213 | return gradle.absolute.path;
|
214 | }
|
215 | throwToolExit(
|
216 | 'Unable to locate gradlew script. Please check that ${gradle.path} '
|
217 | 'exists or that ${gradle.dirname} can be read.' ,
|
218 | );
|
219 | }
|
220 |
|
221 | /// Injects the Gradle wrapper files if any of these files don't exist in [directory].
|
222 | void injectGradleWrapperIfNeeded(Directory directory) {
|
223 | copyDirectory(
|
224 | _cache.getArtifactDirectory('gradle_wrapper' ),
|
225 | directory,
|
226 | shouldCopyFile: (File sourceFile, File destinationFile) {
|
227 | // Don't override the existing files in the project.
|
228 | return !destinationFile.existsSync();
|
229 | },
|
230 | onFileCopied: (File source, File dest) {
|
231 | _operatingSystemUtils.makeExecutable(dest);
|
232 | },
|
233 | );
|
234 | // Add the `gradle-wrapper.properties` file if it doesn't exist.
|
235 | final Directory propertiesDirectory = directory
|
236 | .childDirectory(gradleDirectoryName)
|
237 | .childDirectory(gradleWrapperDirectoryName);
|
238 | final File propertiesFile = propertiesDirectory.childFile(gradleWrapperPropertiesFilename);
|
239 |
|
240 | if (propertiesFile.existsSync()) {
|
241 | return;
|
242 | }
|
243 | propertiesDirectory.createSync(recursive: true);
|
244 | final String gradleVersion = getGradleVersionForAndroidPlugin(directory, _logger);
|
245 | final propertyContents =
|
246 | '''
|
247 | distributionBase=GRADLE_USER_HOME
|
248 | distributionPath=wrapper/dists
|
249 | zipStoreBase=GRADLE_USER_HOME
|
250 | zipStorePath=wrapper/dists
|
251 | distributionUrl=https\\://services.gradle.org/distributions/gradle-$gradleVersion-all.zip
|
252 | ''';
|
253 | propertiesFile.writeAsStringSync(propertyContents);
|
254 | }
|
255 | }
|
256 |
|
257 | /// Returns the Gradle version that the current Android plugin depends on when found,
|
258 | /// otherwise it returns a default version.
|
259 | ///
|
260 | /// The Android plugin version is specified in the `build.gradle`,
|
261 | /// `build.gradle.kts`, `settings.gradle`, or `settings.gradle.kts` file within
|
262 | /// the project's Android directory.
|
263 | String getGradleVersionForAndroidPlugin(Directory directory, Logger logger) {
|
264 | final String? androidPluginVersion = getAgpVersion(directory, logger);
|
265 | if (androidPluginVersion == null) {
|
266 | logger.printTrace(
|
267 | 'AGP version cannot be determined, assuming Gradle version: $templateDefaultGradleVersion',
|
268 | );
|
269 | return templateDefaultGradleVersion;
|
270 | }
|
271 | return getGradleVersionFor(androidPluginVersion);
|
272 | }
|
273 |
|
274 | /// Returns the gradle file from the top level directory.
|
275 | /// The returned file is not guaranteed to be present.
|
276 | File getGradleWrapperFile(Directory directory) {
|
277 | return directory
|
278 | .childDirectory(gradleDirectoryName)
|
279 | .childDirectory(gradleWrapperDirectoryName)
|
280 | .childFile(gradleWrapperPropertiesFilename);
|
281 | }
|
282 |
|
283 | /// Parses the gradle wrapper distribution url to return a string containing
|
284 | /// the version number.
|
285 | ///
|
286 | /// Expected input is of the form '...gradle-7.4.2-all.zip', and the output
|
287 | /// would be of the form '7.4.2'.
|
288 | String? parseGradleVersionFromDistributionUrl(String? distributionUrl) {
|
289 | if (distributionUrl == null) {
|
290 | return null;
|
291 | }
|
292 | final List<String> zipParts = distributionUrl.split('-');
|
293 | if (zipParts.length < 2) {
|
294 | return null;
|
295 | }
|
296 | return zipParts[1];
|
297 | }
|
298 |
|
299 | /// Returns either the gradle-wrapper.properties value from the passed in
|
300 | /// [directory] or if not present the version available in local path.
|
301 | ///
|
302 | /// If gradle version is not found null is returned.
|
303 | /// [directory] should be an android directory with a build.gradle file.
|
304 | Future<String?> getGradleVersion(
|
305 | Directory directory,
|
306 | Logger logger,
|
307 | ProcessManager processManager,
|
308 | ) async {
|
309 | final File propertiesFile = getGradleWrapperFile(directory);
|
310 |
|
311 | if (propertiesFile.existsSync()) {
|
312 | final String wrapperFileContent = propertiesFile.readAsStringSync();
|
313 |
|
314 | final RegExpMatch? distributionUrl = distributionUrlRegex.firstMatch(wrapperFileContent);
|
315 | if (distributionUrl != null) {
|
316 | final String? gradleVersion = parseGradleVersionFromDistributionUrl(distributionUrl.group(0));
|
317 | if (gradleVersion != null) {
|
318 | return gradleVersion;
|
319 | } else {
|
320 | // Did not find gradle zip url. Likely this is a bug in our parsing.
|
321 | logger.printWarning(_formatParseWarning(wrapperFileContent, type: 'gradle'));
|
322 | }
|
323 | } else {
|
324 | // If no distributionUrl log then treat as if there was no propertiesFile.
|
325 | logger.printTrace(
|
326 | '$propertiesFile does not provide a Gradle version falling back to system gradle.',
|
327 | );
|
328 | }
|
329 | } else {
|
330 | // Could not find properties file.
|
331 | logger.printTrace('$propertiesFile does not exist falling back to system gradle');
|
332 | }
|
333 | // System installed Gradle version.
|
334 | // TODO(reidbaker): Modify this gradle execution to use gradlew.
|
335 | if (processManager.canRun('gradle')) {
|
336 | final gradleVersionsVerbose =
|
337 | (await processManager.run(<String>['gradle', gradleVersionsFlag])).stdout as String;
|
338 | // Expected format:
|
339 | /*
|
340 |
|
341 | ------------------------------------------------------------
|
342 | Gradle 7.6
|
343 | ------------------------------------------------------------
|
344 |
|
345 | Build time: 2022-11-25 13:35:10 UTC
|
346 | Revision: daece9dbc5b79370cc8e4fd6fe4b2cd400e150a8
|
347 |
|
348 | Kotlin: 1.7.10
|
349 | Groovy: 3.0.13
|
350 | Ant: Apache Ant(TM) version 1.10.11 compiled on July 10 2021
|
351 | JVM: 17.0.6 (Homebrew 17.0.6+0)
|
352 | OS: Mac OS X 13.2.1 aarch64
|
353 | */
|
354 | // Observation shows that the version can have 2 or 3 numbers.
|
355 | // Inner parentheticals `(\.\d+)?` denote the optional third value.
|
356 | // Outer parentheticals `Gradle (...)` denote a grouping used to extract
|
357 | // the version number.
|
358 | final gradleVersionRegex = RegExp(r'Gradle\s+(\d+\.\d+(?:\.\d+)?)');
|
359 | final RegExpMatch? version = gradleVersionRegex.firstMatch(gradleVersionsVerbose);
|
360 | if (version == null) {
|
361 | // Most likely a bug in our parse implementation/regex.
|
362 | logger.printWarning(_formatParseWarning(gradleVersionsVerbose, type: 'gradle'));
|
363 | return null;
|
364 | }
|
365 | return version.group(1);
|
366 | } else {
|
367 | logger.printTrace('Could not run system gradle');
|
368 | return null;
|
369 | }
|
370 | }
|
371 |
|
372 | /// Returns the Kotlin Gradle Plugin (KGP) version that the current project
|
373 | /// depends on if found, `null` otherwise.
|
374 | /// [androidDirectory] should be an android directory with a `build.gradle` file.
|
375 | Future<String?> getKgpVersion(
|
376 | Directory androidDirectory,
|
377 | Logger logger,
|
378 | ProcessManager processManager,
|
379 | ) async {
|
380 | // Maintainers of the kotlin dsl and the kotlin gradle plugin are different.
|
381 | //
|
382 | // Android Docs refer to the kotlin gradle plugin with either the full name or KGP.
|
383 | // Kotlin docs refer to the kotlin gradle plugin as kotlin android plugin.
|
384 | //
|
385 | // gradle --version or ./gradlew --version will print the kotlin dsl version.
|
386 | // This version normally changes with the version of gradle.
|
387 | // https://github.com/gradle/gradle/blob/cefbee263181a924ac4efcaace6bda97a55bc0f7/platforms/core-runtime/gradle-cli/src/main/java/org/gradle/launcher/cli/DefaultCommandLineActionFactory.java#L260
|
388 | // This vesion is NOT the version of KGP that the project uses.
|
389 | //
|
390 | // Instead the kgpVersion task is a custom flutter task dynamiclly added that can
|
391 | // print the kgp version if gradle can run successfuly.
|
392 |
|
393 | if (processManager.canRun('./gradlew', workingDirectory: androidDirectory.path)) {
|
394 | final ProcessResult command = await processManager.run(<String>[
|
395 | './gradlew',
|
396 | 'kgpVersion',
|
397 | '-q',
|
398 | ], workingDirectory: androidDirectory.path);
|
399 | if (command.exitCode == 0) {
|
400 | final kgpVersionOutput = command.stdout as String;
|
401 |
|
402 | // See expected output defined in
|
403 | // flutter/packages/flutter_tools/gradle/src/main/kotlin/FlutterPluginUtils.kt addTaskForKGPVersion
|
404 | final kotlinVersionRegex = RegExp(r'KGP Version:\s+(\d+\.\d+(?:\.\d+)?)');
|
405 | final RegExpMatch? version = kotlinVersionRegex.firstMatch(kgpVersionOutput);
|
406 | if (version != null) {
|
407 | return version.group(1);
|
408 | }
|
409 | // Most likely a bug in our parse implementation/regex.
|
410 | logger.printWarning(_formatParseWarning(kgpVersionOutput, type: 'kotlin'));
|
411 | } else {
|
412 | logger.printTrace('Non zero exit code from gradle task kgpVersion.');
|
413 | }
|
414 | } else {
|
415 | logger.printTrace('Could not run gradle task kgpVersion.');
|
416 | }
|
417 |
|
418 | // Project valiation code is regularly run on projects that can not build.
|
419 | // Because of that this code also attempts to search through known template
|
420 | // locations for kotlin versions.
|
421 |
|
422 | logger.printTrace('Checking settings for kgp version.');
|
423 | File settingsFile = androidDirectory.childFile('settings.gradle');
|
424 | if (!settingsFile.existsSync()) {
|
425 | settingsFile = androidDirectory.childFile('settings.gradle.kts');
|
426 | }
|
427 |
|
428 | if (settingsFile.existsSync()) {
|
429 | final String settingsFileContent = settingsFile.readAsStringSync();
|
430 | final RegExpMatch? settingsMatch = _kotlinGradlePluginRegExpFromId.firstMatch(
|
431 | settingsFileContent,
|
432 | );
|
433 |
|
434 | if (settingsMatch != null) {
|
435 | final String? kgpVersion = settingsMatch.namedGroup(_versionGroupName);
|
436 | logger.printTrace('$settingsFile provides KGP version: $kgpVersion');
|
437 | return kgpVersion;
|
438 | }
|
439 | } else {
|
440 | logger.printTrace('No settings.gradle.kts');
|
441 | }
|
442 |
|
443 | return null;
|
444 | }
|
445 |
|
446 | /// Returns the Android Gradle Plugin (AGP) version that the current project
|
447 | /// depends on when found, null otherwise.
|
448 | ///
|
449 | /// The Android plugin version is specified in the `build.gradle`,
|
450 | /// `build.gradle.kts`, `settings.gradle, or `settings.gradle.kts`
|
451 | /// files within the project's Android directory ([androidDirectory]).
|
452 | String? getAgpVersion(Directory androidDirectory, Logger logger) {
|
453 | File buildFile = androidDirectory.childFile('build.gradle');
|
454 | if (!buildFile.existsSync()) {
|
455 | buildFile = androidDirectory.childFile('build.gradle.kts');
|
456 | }
|
457 | if (!buildFile.existsSync()) {
|
458 | logger.printTrace('Cannot find build.gradle/build.gradle.kts in $androidDirectory');
|
459 | return null;
|
460 | }
|
461 | final String buildFileContent = buildFile.readAsStringSync();
|
462 | final RegExpMatch? buildMatchClasspath = _androidGradlePluginRegExpFromDependencies.firstMatch(
|
463 | buildFileContent,
|
464 | );
|
465 | if (buildMatchClasspath != null) {
|
466 | final String? androidPluginVersion = buildMatchClasspath.namedGroup(_versionGroupName);
|
467 | logger.printTrace('$buildFile provides AGP version from classpath: $androidPluginVersion');
|
468 | return androidPluginVersion;
|
469 | }
|
470 | final RegExpMatch? buildMatchId = _androidGradlePluginRegExpFromId.firstMatch(buildFileContent);
|
471 | if (buildMatchId != null) {
|
472 | final String? androidPluginVersion = buildMatchId.namedGroup(_versionGroupName);
|
473 | logger.printTrace('$buildFile provides AGP version from plugin id: $androidPluginVersion');
|
474 | return androidPluginVersion;
|
475 | }
|
476 |
|
477 | logger.printTrace("$buildFile doesn't provide an AGP version. Checking settings.");
|
478 | File settingsFile = androidDirectory.childFile('settings.gradle');
|
479 | if (!settingsFile.existsSync()) {
|
480 | settingsFile = androidDirectory.childFile('settings.gradle.kts');
|
481 | }
|
482 | if (!settingsFile.existsSync()) {
|
483 | logger.printTrace('Cannot find settings.gradle/settings.gradle.kts in $androidDirectory');
|
484 | return null;
|
485 | }
|
486 | final String settingsFileContent = settingsFile.readAsStringSync();
|
487 | final RegExpMatch? settingsMatch = _androidGradlePluginRegExpFromId.firstMatch(
|
488 | settingsFileContent,
|
489 | );
|
490 |
|
491 | if (settingsMatch != null) {
|
492 | final String? androidPluginVersion = settingsMatch.namedGroup(_versionGroupName);
|
493 | logger.printTrace('$settingsFile provides AGP version: $androidPluginVersion');
|
494 | return androidPluginVersion;
|
495 | }
|
496 | logger.printTrace("$settingsFile doesn't provide an AGP version.");
|
497 | return null;
|
498 | }
|
499 |
|
500 | String _formatParseWarning(String content, {required String type}) {
|
501 | return 'Could not parse $type version from: \n'
|
502 | '$content \n'
|
503 | 'If there is a version please look for an existing bug '
|
504 | 'https://github.com/flutter/flutter/issues/'
|
505 | ' and if one does not exist file a new issue.';
|
506 | }
|
507 |
|
508 | // Validate that KGP and Gradle are compatible with each other.
|
509 | //
|
510 | // Returns true if versions are compatible.
|
511 | // Null or empty Gradle or KGP version returns false.
|
512 | // If compatibility cannot be evaluated returns false.
|
513 | // If versions are newer than the max known version a warning is logged and true
|
514 | // returned.
|
515 | //
|
516 | // Source of truth found here:
|
517 | // https://kotlinlang.org/docs/gradle-configure-project.html#apply-the-plugin
|
518 | bool validateGradleAndKGP(Logger logger, {required String? kgpV, required String? gradleV}) {
|
519 | if (gradleV == null || kgpV == null || gradleV.isEmpty || kgpV.isEmpty) {
|
520 | logger.printTrace('Gradle or KGP version unknown ($gradleV, $kgpV).');
|
521 | return false;
|
522 | }
|
523 |
|
524 | if (isWithinVersionRange(gradleV, min: '0.0', max: oldestConsideredGradleVersion)) {
|
525 | logger.printTrace(
|
526 | 'Gradle version $gradleV older than oldest considered $oldestConsideredGradleVersion',
|
527 | );
|
528 | return false;
|
529 | }
|
530 |
|
531 | if (isWithinVersionRange(
|
532 | kgpV,
|
533 | min: maxKnownAndSupportedKgpVersion,
|
534 | max: '100.100',
|
535 | inclusiveMin: false,
|
536 | )) {
|
537 | logger.printTrace(
|
538 | 'Newer than known KGP version ($kgpV), gradle ($gradleV).'
|
539 | '\n Treating as valid configuration.',
|
540 | );
|
541 | return true;
|
542 | }
|
543 |
|
544 | // https://kotlinlang.org/docs/gradle-configure-project.html#apply-the-plugin
|
545 | // Documenation is non continuous, past versions are known to the
|
546 | // publishers of KGP. When covering version ranges beyond what is documented
|
547 | // add a comment with the documented value.
|
548 | // Continuous KGP version handling is prefered in case an emergency patch to a
|
549 | // past release is shipped this code will assume the version range that is closest.
|
550 | if (isWithinVersionRange(kgpV, min: '2.1.20', max: '2.1.20')) {
|
551 | // Documented max is 8.11, using 8.12 non inclusive covers patch versions.
|
552 | return isWithinVersionRange(gradleV, min: '7.6.3', max: '8.12', inclusiveMax: false);
|
553 | }
|
554 | if (isWithinVersionRange(kgpV, min: '2.1.0', max: '2.1.10')) {
|
555 | // Documented max is 8.10, using 8.11 non inclusive covers patch versions.
|
556 | return isWithinVersionRange(gradleV, min: '7.6.3', max: '8.11', inclusiveMax: false);
|
557 | }
|
558 | // Documented max is 2.0.21.
|
559 | if (isWithinVersionRange(kgpV, min: '2.0.20', max: '2.1', inclusiveMax: false)) {
|
560 | // Documented max is 8.5, using 8.9 non inclusive covers patch versions.
|
561 | // Kotlin Multiplatform can throw warnings on 8.8.
|
562 | return isWithinVersionRange(gradleV, min: '6.8.3', max: '8.9', inclusiveMax: false);
|
563 | }
|
564 | if (isWithinVersionRange(kgpV, min: '2.0', max: '2.0.20', inclusiveMax: false)) {
|
565 | // Documented max is 8.5, using 8.6 non inclusive covers patch versions.
|
566 | return isWithinVersionRange(gradleV, min: '6.8.3', max: '8.6', inclusiveMax: false);
|
567 | }
|
568 | // Documented max is 1.9.25.
|
569 | if (isWithinVersionRange(kgpV, min: '1.9.20', max: '2.0', inclusiveMax: false)) {
|
570 | return isWithinVersionRange(gradleV, min: '6.8.3', max: '8.1.1');
|
571 | }
|
572 | // Documented max is 1.9.10.
|
573 | if (isWithinVersionRange(kgpV, min: '1.8.20', max: '1.9.20', inclusiveMax: false)) {
|
574 | return isWithinVersionRange(gradleV, min: '6.8.3', max: '7.6.0');
|
575 | }
|
576 | // Documented max is 1.8.11.
|
577 | if (isWithinVersionRange(kgpV, min: '1.8.0', max: '1.8.20', inclusiveMax: false)) {
|
578 | return isWithinVersionRange(gradleV, min: '6.8.3', max: '7.3.3');
|
579 | }
|
580 | // Documented max is 1.7.22.
|
581 | if (isWithinVersionRange(kgpV, min: '1.7.20', max: '1.8.0', inclusiveMax: false)) {
|
582 | return isWithinVersionRange(gradleV, min: '6.7.1', max: '7.1.1');
|
583 | }
|
584 | // Documented max is 1.7.10.
|
585 | if (isWithinVersionRange(kgpV, min: '1.7.0', max: '1.7.20', inclusiveMax: false)) {
|
586 | return isWithinVersionRange(gradleV, min: '6.7.1', max: '7.0.2');
|
587 | }
|
588 | // Documented max is 1.6.21.
|
589 | if (isWithinVersionRange(
|
590 | kgpV,
|
591 | min: oldestDocumentedKgpCompatabilityVersion,
|
592 | max: '1.7.0',
|
593 | inclusiveMax: false,
|
594 | )) {
|
595 | return isWithinVersionRange(gradleV, min: '6.1.1', max: '7.0.2');
|
596 | }
|
597 |
|
598 | logger.printTrace('Unknown KGP-Gradle compatibility, KGP: $kgpV, Gradle: $gradleV');
|
599 | return false;
|
600 | }
|
601 |
|
602 | // Validate that KGP and AGP are compatible with each other.
|
603 | //
|
604 | // Returns true if versions are compatible.
|
605 | // Null or empty KGP or AGP version returns false.
|
606 | // If compatibility cannot be evaluated returns false.
|
607 | // If versions are newer than the max known version a warning is logged and true
|
608 | // returned.
|
609 | //
|
610 | // Source of truth found here:
|
611 | // https://kotlinlang.org/docs/gradle-configure-project.html#apply-the-plugin
|
612 | bool validateAgpAndKgp(Logger logger, {required String? kgpV, required String? agpV}) {
|
613 | if (agpV == null || kgpV == null || agpV.isEmpty || kgpV.isEmpty) {
|
614 | logger.printTrace('KGP or AGP version unknown ($kgpV, $agpV).');
|
615 | return false;
|
616 | }
|
617 |
|
618 | if (isWithinVersionRange(agpV, min: '0.0', max: oldestConsideredAgpVersion)) {
|
619 | logger.printTrace(
|
620 | 'AGP version ($agpV) older than oldest supported $oldestConsideredAgpVersion.',
|
621 | );
|
622 | }
|
623 | const maxKnownAgpVersionWithFullKotinSupport = '8.7.2';
|
624 |
|
625 | if (isWithinVersionRange(
|
626 | kgpV,
|
627 | min: maxKnownAndSupportedKgpVersion,
|
628 | max: '100.100',
|
629 | inclusiveMin: false,
|
630 | ) ||
|
631 | isWithinVersionRange(
|
632 | agpV,
|
633 | min: maxKnownAgpVersionWithFullKotinSupport,
|
634 | max: '100.100',
|
635 | inclusiveMin: false,
|
636 | )) {
|
637 | logger.printTrace(
|
638 | 'Newer than known KGP version ($kgpV), AGP ($agpV).'
|
639 | '\n Treating as valid configuration.',
|
640 | );
|
641 | return true;
|
642 | }
|
643 |
|
644 | // https://kotlinlang.org/docs/gradle-configure-project.html#apply-the-plugin
|
645 | // Documenation is non continuous, past versions are known to the
|
646 | // publishers of KGP. When covering version ranges beyond what is documented
|
647 | // add a comment with the documented value.
|
648 | // Continuous KGP version handling is prefered in case an emergency patch to a
|
649 | // past release is shipped this code will assume the version range that is closest.
|
650 | if (isWithinVersionRange(kgpV, min: '2.1.0', max: '2.1.20')) {
|
651 | return isWithinVersionRange(agpV, min: '7.3.1', max: '8.7.2');
|
652 | }
|
653 | // Documented max is 2.0.21
|
654 | if (isWithinVersionRange(kgpV, min: '2.0.20', max: '2.1.0', inclusiveMax: false)) {
|
655 | // Documented max is 8.5.
|
656 | return isWithinVersionRange(agpV, min: '7.1.3', max: '8.6', inclusiveMax: false);
|
657 | }
|
658 | // Documented max is 2.0.0.
|
659 | if (isWithinVersionRange(kgpV, min: '2.0.0', max: '2.0.20', inclusiveMax: false)) {
|
660 | return isWithinVersionRange(agpV, min: '7.1.3', max: '8.3.1');
|
661 | }
|
662 | // Documented max is 1.9.25
|
663 | if (isWithinVersionRange(kgpV, min: '1.9.20', max: '2.0.0', inclusiveMax: false)) {
|
664 | return isWithinVersionRange(agpV, min: '4.2.2', max: '8.1.0');
|
665 | }
|
666 | // Documented max is 1.9.10
|
667 | if (isWithinVersionRange(kgpV, min: '1.9.0', max: '1.9.20', inclusiveMax: false)) {
|
668 | return isWithinVersionRange(agpV, min: '4.2.2', max: '7.4.0');
|
669 | }
|
670 | // Documented max is 1.8.22
|
671 | if (isWithinVersionRange(kgpV, min: '1.8.20', max: '1.9', inclusiveMax: false)) {
|
672 | return isWithinVersionRange(agpV, min: '4.1.3', max: '7.4.0');
|
673 | }
|
674 | // Documented max is 1.8.11
|
675 | if (isWithinVersionRange(kgpV, min: '1.8.0', max: '1.8.20', inclusiveMax: false)) {
|
676 | return isWithinVersionRange(agpV, min: '4.1.3', max: '7.2.1');
|
677 | }
|
678 | // Documented max is 1.7.22
|
679 | if (isWithinVersionRange(kgpV, min: '1.7.20', max: '1.8.0', inclusiveMax: false)) {
|
680 | return isWithinVersionRange(agpV, min: '3.6.4', max: '7.0.4');
|
681 | }
|
682 | // Documented max is 1.7.10
|
683 | // Documented gap between 1.6.21 and 1.7.0.
|
684 | if (isWithinVersionRange(kgpV, min: '1.6.20', max: '1.7.20', inclusiveMax: false)) {
|
685 | return isWithinVersionRange(agpV, min: '3.4.3', max: '7.0.2');
|
686 | }
|
687 |
|
688 | logger.printTrace('Unknown KGP-Gradle compatibility, KGP: $kgpV, AGP: $agpV');
|
689 | return false;
|
690 | }
|
691 |
|
692 | // Validate that Gradle version and AGP are compatible with each other.
|
693 | //
|
694 | // Returns true if versions are compatible.
|
695 | // Null Gradle version or AGP version returns false.
|
696 | // If compatibility cannot be evaluated returns false.
|
697 | // If versions are newer than the max known version a warning is logged and true
|
698 | // returned.
|
699 | //
|
700 | // Source of truth found here:
|
701 | // https://developer.android.com/studio/releases/gradle-plugin#updating-gradle
|
702 | // AGP has a minimum version of gradle required but no max starting at
|
703 | // AGP version 2.3.0+.
|
704 | bool validateGradleAndAgp(Logger logger, {required String? gradleV, required String? agpV}) {
|
705 | if (gradleV == null || agpV == null) {
|
706 | logger.printTrace('Gradle version or AGP version unknown ($gradleV, $agpV).');
|
707 | return false;
|
708 | }
|
709 |
|
710 | // First check if versions are too old.
|
711 | if (isWithinVersionRange(
|
712 | agpV,
|
713 | min: '0.0',
|
714 | max: oldestConsideredAgpVersion,
|
715 | inclusiveMax: false,
|
716 | )) {
|
717 | logger.printTrace('AGP Version: $agpV is too old.');
|
718 | return false;
|
719 | }
|
720 | if (isWithinVersionRange(
|
721 | gradleV,
|
722 | min: '0.0',
|
723 | max: oldestConsideredGradleVersion,
|
724 | inclusiveMax: false,
|
725 | )) {
|
726 | logger.printTrace('Gradle Version: $gradleV is too old.');
|
727 | return false;
|
728 | }
|
729 |
|
730 | // Check highest supported version before checking unknown versions.
|
731 | if (isWithinVersionRange(agpV, min: '8.0', max: maxKnownAndSupportedAgpVersion)) {
|
732 | return isWithinVersionRange(gradleV, min: '8.0', max: maxKnownAndSupportedGradleVersion);
|
733 | }
|
734 | // Check if versions are newer than the max known versions.
|
735 | if (isWithinVersionRange(agpV, min: maxKnownAndSupportedAgpVersion, max: '100.100')) {
|
736 | // Assume versions we do not know about are valid but log.
|
737 | final bool validGradle = isWithinVersionRange(gradleV, min: '8.0', max: '100.00');
|
738 | logger.printTrace(
|
739 | 'Newer than known AGP version ($agpV), gradle ($gradleV).'
|
740 | '\n Treating as valid configuration.',
|
741 | );
|
742 | return validGradle;
|
743 | }
|
744 |
|
745 | // Begin Known Gradle <-> AGP validation.
|
746 | if (isWithinVersionRange(agpV, min: '8.4.0', max: '8.4.99')) {
|
747 | return isWithinVersionRange(gradleV, min: '8.6', max: maxKnownAndSupportedGradleVersion);
|
748 | }
|
749 | if (isWithinVersionRange(agpV, min: '8.3.0', max: '8.3.99')) {
|
750 | return isWithinVersionRange(gradleV, min: '8.4', max: maxKnownAndSupportedGradleVersion);
|
751 | }
|
752 | if (isWithinVersionRange(agpV, min: '8.2.0', max: '8.2.99')) {
|
753 | return isWithinVersionRange(gradleV, min: '8.2', max: maxKnownAndSupportedGradleVersion);
|
754 | }
|
755 | if (isWithinVersionRange(agpV, min: '8.0.0', max: '8.1.99')) {
|
756 | return isWithinVersionRange(gradleV, min: '8.0', max: maxKnownAndSupportedGradleVersion);
|
757 | }
|
758 | // Max agp here is a made up version to contain all 7.4 changes.
|
759 | if (isWithinVersionRange(agpV, min: '7.4', max: '7.5')) {
|
760 | return isWithinVersionRange(gradleV, min: '7.5', max: maxKnownAndSupportedGradleVersion);
|
761 | }
|
762 | if (isWithinVersionRange(agpV, min: '7.3', max: '7.4', inclusiveMax: false)) {
|
763 | return isWithinVersionRange(gradleV, min: '7.4', max: maxKnownAndSupportedGradleVersion);
|
764 | }
|
765 | if (isWithinVersionRange(agpV, min: '7.2', max: '7.3', inclusiveMax: false)) {
|
766 | return isWithinVersionRange(gradleV, min: '7.3.3', max: maxKnownAndSupportedGradleVersion);
|
767 | }
|
768 | if (isWithinVersionRange(agpV, min: '7.1', max: '7.2', inclusiveMax: false)) {
|
769 | return isWithinVersionRange(gradleV, min: '7.2', max: maxKnownAndSupportedGradleVersion);
|
770 | }
|
771 | if (isWithinVersionRange(agpV, min: '7.0', max: '7.1', inclusiveMax: false)) {
|
772 | return isWithinVersionRange(gradleV, min: '7.0', max: maxKnownAndSupportedGradleVersion);
|
773 | }
|
774 | if (isWithinVersionRange(agpV, min: '4.2.0', max: '7.0', inclusiveMax: false)) {
|
775 | return isWithinVersionRange(gradleV, min: '6.7.1', max: maxKnownAndSupportedGradleVersion);
|
776 | }
|
777 | if (isWithinVersionRange(agpV, min: '4.1.0', max: '4.2.0', inclusiveMax: false)) {
|
778 | return isWithinVersionRange(gradleV, min: '6.5', max: maxKnownAndSupportedGradleVersion);
|
779 | }
|
780 | if (isWithinVersionRange(agpV, min: '4.0.0', max: '4.1.0', inclusiveMax: false)) {
|
781 | return isWithinVersionRange(gradleV, min: '6.1.1', max: maxKnownAndSupportedGradleVersion);
|
782 | }
|
783 | if (isWithinVersionRange(agpV, min: '3.6.0', max: '3.6.4')) {
|
784 | return isWithinVersionRange(gradleV, min: '5.6.4', max: maxKnownAndSupportedGradleVersion);
|
785 | }
|
786 | if (isWithinVersionRange(agpV, min: '3.5.0', max: '3.5.4')) {
|
787 | return isWithinVersionRange(gradleV, min: '5.4.1', max: maxKnownAndSupportedGradleVersion);
|
788 | }
|
789 | if (isWithinVersionRange(agpV, min: '3.4.0', max: '3.4.3')) {
|
790 | return isWithinVersionRange(gradleV, min: '5.1.1', max: maxKnownAndSupportedGradleVersion);
|
791 | }
|
792 | if (isWithinVersionRange(agpV, min: '3.3.0', max: '3.3.3')) {
|
793 | return isWithinVersionRange(gradleV, min: '4.10.1', max: maxKnownAndSupportedGradleVersion);
|
794 | }
|
795 |
|
796 | logger.printTrace('Unknown Gradle-AGP compatibility, Gradle: $gradleV, AGP: $agpV');
|
797 | return false;
|
798 | }
|
799 |
|
800 | /// Validate that the [javaVersion] and [gradleVersion] are compatible with
|
801 | /// each other.
|
802 | ///
|
803 | /// Source of truth:
|
804 | /// https://docs.gradle.org/current/userguide/compatibility.html#java
|
805 | bool validateJavaAndGradle(
|
806 | Logger logger, {
|
807 | required String? javaVersion,
|
808 | required String? gradleVersion,
|
809 | }) {
|
810 | // https://docs.gradle.org/current/userguide/compatibility.html#java
|
811 | const oldestConsideredJavaVersion = '1.8';
|
812 | const oldestDocumentedJavaGradleCompatibility = '2.0';
|
813 |
|
814 | // Begin Java <-> Gradle validation.
|
815 |
|
816 | if (javaVersion == null || gradleVersion == null) {
|
817 | logger.printTrace('Java version or Gradle version unknown ($javaVersion, $gradleVersion).');
|
818 | return false;
|
819 | }
|
820 |
|
821 | // First check if versions are too old.
|
822 | if (isWithinVersionRange(
|
823 | javaVersion,
|
824 | min: '1.1',
|
825 | max: oldestConsideredJavaVersion,
|
826 | inclusiveMax: false,
|
827 | )) {
|
828 | logger.printTrace('Java Version: $javaVersion is too old.');
|
829 | return false;
|
830 | }
|
831 | if (isWithinVersionRange(
|
832 | gradleVersion,
|
833 | min: '0.0',
|
834 | max: oldestDocumentedJavaGradleCompatibility,
|
835 | inclusiveMax: false,
|
836 | )) {
|
837 | logger.printTrace('Gradle Version: $gradleVersion is too old.');
|
838 | return false;
|
839 | }
|
840 |
|
841 | // Check if versions are newer than the max supported versions.
|
842 | if (isWithinVersionRange(javaVersion, min: oneMajorVersionHigherJavaVersion, max: '100.100')) {
|
843 | // Assume versions Java versions newer than [maxSupportedJavaVersion]
|
844 | // required a higher gradle version.
|
845 | final bool validGradle = isWithinVersionRange(
|
846 | gradleVersion,
|
847 | min: maxKnownAndSupportedGradleVersion,
|
848 | max: '100.00',
|
849 | );
|
850 | logger.printWarning(
|
851 | 'Newer than known valid Java version ($javaVersion), gradle ($gradleVersion).'
|
852 | '\n Treating as valid configuration.',
|
853 | );
|
854 | return validGradle;
|
855 | }
|
856 |
|
857 | // Begin known Java <-> Gradle evaluation.
|
858 | for (final JavaGradleCompat data in _javaGradleCompatList) {
|
859 | if (isWithinVersionRange(
|
860 | javaVersion,
|
861 | min: data.javaMin,
|
862 | max: data.javaMax,
|
863 | inclusiveMax: false,
|
864 | )) {
|
865 | return isWithinVersionRange(gradleVersion, min: data.gradleMin, max: data.gradleMax);
|
866 | }
|
867 | }
|
868 |
|
869 | logger.printTrace(
|
870 | 'Unknown Java-Gradle compatibility, Java: $javaVersion, Gradle: $gradleVersion',
|
871 | );
|
872 | return false;
|
873 | }
|
874 |
|
875 | /// Returns compatibility information for the valid range of Gradle versions for
|
876 | /// the specified Java version.
|
877 | ///
|
878 | /// Returns null when the tooling has not documented the compatible Gradle
|
879 | /// versions for the Java version (either the version is too old or too new). If
|
880 | /// this seems like a mistake, the caller may need to update the
|
881 | /// [_javaGradleCompatList] detailing Java/Gradle compatibility.
|
882 | JavaGradleCompat? getValidGradleVersionRangeForJavaVersion(Logger logger, {required String javaV}) {
|
883 | for (final JavaGradleCompat data in _javaGradleCompatList) {
|
884 | if (isWithinVersionRange(javaV, min: data.javaMin, max: data.javaMax, inclusiveMax: false)) {
|
885 | return data;
|
886 | }
|
887 | }
|
888 |
|
889 | logger.printTrace('Unable to determine valid Gradle version range for Java version $javaV.');
|
890 | return null;
|
891 | }
|
892 |
|
893 | /// Validate that the specified Java and Android Gradle Plugin (AGP) versions are
|
894 | /// compatible with each other.
|
895 | ///
|
896 | /// Returns true when the specified Java and AGP versions are
|
897 | /// definitely compatible; otherwise, false is assumed by default. In addition,
|
898 | /// this will return false when either a null Java or AGP version is provided.
|
899 | ///
|
900 | /// Source of truth are the AGP release notes:
|
901 | /// https://developer.android.com/build/releases/gradle-plugin
|
902 | bool validateJavaAndAgp(Logger logger, {required String? javaV, required String? agpV}) {
|
903 | if (javaV == null || agpV == null) {
|
904 | logger.printTrace('Java version or AGP version unknown ($javaV, $agpV).');
|
905 | return false;
|
906 | }
|
907 |
|
908 | // Check if AGP version is too old to perform validation.
|
909 | if (isWithinVersionRange(
|
910 | agpV,
|
911 | min: '1.0',
|
912 | max: oldestDocumentedJavaAgpCompatibilityVersion,
|
913 | inclusiveMax: false,
|
914 | )) {
|
915 | logger.printTrace('AGP Version: $agpV is too old to determine Java compatibility.');
|
916 | return false;
|
917 | }
|
918 |
|
919 | if (isWithinVersionRange(
|
920 | agpV,
|
921 | min: maxKnownAndSupportedAgpVersion,
|
922 | max: '100.100',
|
923 | inclusiveMin: false,
|
924 | )) {
|
925 | logger.printTrace('AGP Version: $agpV is too new to determine Java compatibility.');
|
926 | return false;
|
927 | }
|
928 |
|
929 | // Begin known Java <-> AGP evaluation.
|
930 | for (final JavaAgpCompat data in _javaAgpCompatList) {
|
931 | if (isWithinVersionRange(agpV, min: data.agpMin, max: data.agpMax)) {
|
932 | return isWithinVersionRange(javaV, min: data.javaMin, max: '100.100');
|
933 | }
|
934 | }
|
935 |
|
936 | logger.printTrace('Unknown Java-AGP compatibility $javaV, $agpV');
|
937 | return false;
|
938 | }
|
939 |
|
940 | /// Returns compatibility information concerning the minimum AGP
|
941 | /// version for the specified Java version.
|
942 | JavaAgpCompat? getMinimumAgpVersionForJavaVersion(Logger logger, {required String javaV}) {
|
943 | for (final JavaAgpCompat data in _javaAgpCompatList) {
|
944 | if (isWithinVersionRange(javaV, min: data.javaMin, max: '100.100')) {
|
945 | return data;
|
946 | }
|
947 | }
|
948 |
|
949 | logger.printTrace('Unable to determine minimum AGP version for specified Java version.');
|
950 | return null;
|
951 | }
|
952 |
|
953 | /// Returns valid Java range for specified Gradle and AGP versions.
|
954 | ///
|
955 | /// Assumes that gradleV and agpV are compatible versions.
|
956 | VersionRange getJavaVersionFor({required String gradleV, required String agpV}) {
|
957 | // Find minimum Java version based on AGP compatibility.
|
958 | String? minJavaVersion;
|
959 | for (final JavaAgpCompat data in _javaAgpCompatList) {
|
960 | if (isWithinVersionRange(agpV, min: data.agpMin, max: data.agpMax)) {
|
961 | minJavaVersion = data.javaMin;
|
962 | }
|
963 | }
|
964 |
|
965 | // Find maximum Java version based on Gradle compatibility.
|
966 | String? maxJavaVersion;
|
967 | for (final JavaGradleCompat data in _javaGradleCompatList.reversed) {
|
968 | if (isWithinVersionRange(
|
969 | gradleV,
|
970 | min: data.gradleMin,
|
971 | max: maxKnownAndSupportedGradleVersion,
|
972 | )) {
|
973 | maxJavaVersion = data.javaMax;
|
974 | }
|
975 | }
|
976 |
|
977 | return VersionRange(minJavaVersion, maxJavaVersion);
|
978 | }
|
979 |
|
980 | /// Returns the Gradle version that is required by the given Android Gradle plugin version
|
981 | /// by picking the largest compatible version from
|
982 | /// https://developer.android.com/studio/releases/gradle-plugin#updating-gradle
|
983 | String getGradleVersionFor(String androidPluginVersion) {
|
984 | final compatList = <GradleForAgp>[
|
985 | GradleForAgp(agpMin: '1.0.0', agpMax: '1.1.3', minRequiredGradle: '2.3'),
|
986 | GradleForAgp(agpMin: '1.2.0', agpMax: '1.3.1', minRequiredGradle: '2.9'),
|
987 | GradleForAgp(agpMin: '1.5.0', agpMax: '1.5.0', minRequiredGradle: '2.2.1'),
|
988 | GradleForAgp(agpMin: '2.0.0', agpMax: '2.1.2', minRequiredGradle: '2.13'),
|
989 | GradleForAgp(agpMin: '2.1.3', agpMax: '2.2.3', minRequiredGradle: '2.14.1'),
|
990 | GradleForAgp(agpMin: '2.3.0', agpMax: '2.9.9', minRequiredGradle: '3.3'),
|
991 | GradleForAgp(agpMin: '3.0.0', agpMax: '3.0.9', minRequiredGradle: '4.1'),
|
992 | GradleForAgp(agpMin: '3.1.0', agpMax: '3.1.9', minRequiredGradle: '4.4'),
|
993 | GradleForAgp(agpMin: '3.2.0', agpMax: '3.2.1', minRequiredGradle: '4.6'),
|
994 | GradleForAgp(agpMin: '3.3.0', agpMax: '3.3.2', minRequiredGradle: '4.10.2'),
|
995 | GradleForAgp(agpMin: '3.4.0', agpMax: '3.5.0', minRequiredGradle: '5.6.2'),
|
996 | GradleForAgp(agpMin: '4.0.0', agpMax: '4.1.0', minRequiredGradle: '6.7'),
|
997 | // 7.5 is a made up value to include everything through 7.4.*
|
998 | GradleForAgp(agpMin: '7.0.0', agpMax: '7.5', minRequiredGradle: '7.5'),
|
999 | // Use 0 and 99 as a patch values to signify every AGP patch version with
|
1000 | // that major and minor version.
|
1001 | GradleForAgp(agpMin: '8.0.0', agpMax: '8.1.99', minRequiredGradle: '8.0'),
|
1002 | GradleForAgp(agpMin: '8.2.0', agpMax: '8.2.99', minRequiredGradle: '8.2'),
|
1003 | GradleForAgp(agpMin: '8.3.0', agpMax: '8.3.99', minRequiredGradle: '8.4'),
|
1004 | GradleForAgp(agpMin: '8.4.0', agpMax: '8.4.99', minRequiredGradle: '8.6'),
|
1005 | GradleForAgp(agpMin: '8.5.0', agpMax: '8.6.99', minRequiredGradle: '8.7'),
|
1006 | GradleForAgp(agpMin: '8.7.0', agpMax: '8.7.99', minRequiredGradle: '8.9'),
|
1007 | GradleForAgp(agpMin: '8.8.0', agpMax: '8.8.99', minRequiredGradle: '8.10.2'),
|
1008 | GradleForAgp(agpMin: '8.9.0', agpMax: '8.9.99', minRequiredGradle: '8.11.1'),
|
1009 | // Assume if AGP is newer than this code know about return the highest gradle
|
1010 | // version we know about.
|
1011 | GradleForAgp(
|
1012 | agpMin: maxKnownAgpVersion,
|
1013 | agpMax: maxKnownAgpVersion,
|
1014 | minRequiredGradle: maxKnownAndSupportedGradleVersion,
|
1015 | ),
|
1016 | ];
|
1017 | for (final data in compatList) {
|
1018 | if (isWithinVersionRange(androidPluginVersion, min: data.agpMin, max: data.agpMax)) {
|
1019 | return data.minRequiredGradle;
|
1020 | }
|
1021 | }
|
1022 | if (isWithinVersionRange(androidPluginVersion, min: maxKnownAgpVersion, max: '100.100')) {
|
1023 | return maxKnownAndSupportedGradleVersion;
|
1024 | }
|
1025 | throwToolExit('Unsupported Android Plugin version: $androidPluginVersion.');
|
1026 | }
|
1027 |
|
1028 | /// Overwrite local.properties in the specified Flutter project's Android
|
1029 | /// sub-project, if needed.
|
1030 | ///
|
1031 | /// If [requireAndroidSdk] is true (the default) and no Android SDK is found,
|
1032 | /// this will fail with a [ToolExit].
|
1033 | void updateLocalProperties({
|
1034 | required FlutterProject project,
|
1035 | BuildInfo? buildInfo,
|
1036 | bool requireAndroidSdk = true,
|
1037 | }) {
|
1038 | if (requireAndroidSdk && globals.androidSdk == null) {
|
1039 | exitWithNoSdkMessage();
|
1040 | }
|
1041 | final File localProperties = project.android.localPropertiesFile;
|
1042 | var changed = false;
|
1043 |
|
1044 | SettingsFile settings;
|
1045 | if (localProperties.existsSync()) {
|
1046 | settings = SettingsFile.parseFromFile(localProperties);
|
1047 | } else {
|
1048 | settings = SettingsFile();
|
1049 | changed = true;
|
1050 | }
|
1051 |
|
1052 | void changeIfNecessary(String key, String? value) {
|
1053 | if (settings.values[key] == value) {
|
1054 | return;
|
1055 | }
|
1056 | if (value == null) {
|
1057 | settings.values.remove(key);
|
1058 | } else {
|
1059 | settings.values[key] = value;
|
1060 | }
|
1061 | changed = true;
|
1062 | }
|
1063 |
|
1064 | final AndroidSdk? androidSdk = globals.androidSdk;
|
1065 | if (androidSdk != null) {
|
1066 | changeIfNecessary('sdk.dir', globals.fsUtils.escapePath(androidSdk.directory.path));
|
1067 | }
|
1068 |
|
1069 | changeIfNecessary('flutter.sdk', globals.fsUtils.escapePath(Cache.flutterRoot!));
|
1070 | if (buildInfo != null) {
|
1071 | changeIfNecessary('flutter.buildMode', buildInfo.modeName);
|
1072 | final String? buildName = validatedBuildNameForPlatform(
|
1073 | TargetPlatform.android_arm,
|
1074 | buildInfo.buildName ?? project.manifest.buildName,
|
1075 | globals.logger,
|
1076 | );
|
1077 | changeIfNecessary('flutter.versionName', buildName);
|
1078 | final String? buildNumber = validatedBuildNumberForPlatform(
|
1079 | TargetPlatform.android_arm,
|
1080 | buildInfo.buildNumber ?? project.manifest.buildNumber,
|
1081 | globals.logger,
|
1082 | );
|
1083 | changeIfNecessary('flutter.versionCode', buildNumber);
|
1084 | }
|
1085 |
|
1086 | if (changed) {
|
1087 | settings.writeContents(localProperties);
|
1088 | }
|
1089 | }
|
1090 |
|
1091 | /// Writes standard Android local properties to the specified [properties] file.
|
1092 | ///
|
1093 | /// Writes the path to the Android SDK, if known.
|
1094 | void writeLocalProperties(File properties) {
|
1095 | final settings = SettingsFile();
|
1096 | final AndroidSdk? androidSdk = globals.androidSdk;
|
1097 | if (androidSdk != null) {
|
1098 | settings.values['sdk.dir'] = globals.fsUtils.escapePath(androidSdk.directory.path);
|
1099 | }
|
1100 | settings.writeContents(properties);
|
1101 | }
|
1102 |
|
1103 | void exitWithNoSdkMessage() {
|
1104 | globals.analytics.send(
|
1105 | Event.flutterBuildInfo(
|
1106 | label: 'unsupported-project',
|
1107 | buildType: 'gradle',
|
1108 | error: 'android-sdk-not-found',
|
1109 | ),
|
1110 | );
|
1111 | throwToolExit(
|
1112 | '${globals.logger.terminal.warningMark} No Android SDK found. '
|
1113 | 'Try setting the ANDROID_HOME environment variable.',
|
1114 | );
|
1115 | }
|
1116 |
|
1117 | // Data class to hold normal/defined Java <-> Gradle compatibility criteria.
|
1118 | //
|
1119 | // The [javaMax] is exclusive in terms of supporting the noted [gradleMin],
|
1120 | // whereas [javaMin] is inclusive.
|
1121 | @immutable
|
1122 | class JavaGradleCompat {
|
1123 | const JavaGradleCompat({
|
1124 | required this.javaMin,
|
1125 | required this.javaMax,
|
1126 | required this.gradleMin,
|
1127 | required this.gradleMax,
|
1128 | });
|
1129 |
|
1130 | final String javaMin;
|
1131 | final String javaMax;
|
1132 | final String gradleMin;
|
1133 | final String gradleMax;
|
1134 |
|
1135 | @override
|
1136 | bool operator ==(Object other) =>
|
1137 | other is JavaGradleCompat &&
|
1138 | other.javaMin == javaMin &&
|
1139 | other.javaMax == javaMax &&
|
1140 | other.gradleMin == gradleMin &&
|
1141 | other.gradleMax == gradleMax;
|
1142 |
|
1143 | @override
|
1144 | int get hashCode => Object.hash(javaMin, javaMax, gradleMin, gradleMax);
|
1145 | }
|
1146 |
|
1147 | // Data class to hold defined Java <-> AGP compatibility criteria.
|
1148 | //
|
1149 | // The [agpMin] and [agpMax] are inclusive in terms of having the
|
1150 | // noted [javaMin] and [javaDefault] versions.
|
1151 | @immutable
|
1152 | class JavaAgpCompat {
|
1153 | const JavaAgpCompat({
|
1154 | required this.javaMin,
|
1155 | required this.javaDefault,
|
1156 | required this.agpMin,
|
1157 | required this.agpMax,
|
1158 | });
|
1159 |
|
1160 | final String javaMin;
|
1161 | final String javaDefault;
|
1162 | final String agpMin;
|
1163 | final String agpMax;
|
1164 |
|
1165 | @override
|
1166 | bool operator ==(Object other) =>
|
1167 | other is JavaAgpCompat &&
|
1168 | other.javaMin == javaMin &&
|
1169 | other.javaDefault == javaDefault &&
|
1170 | other.agpMin == agpMin &&
|
1171 | other.agpMax == agpMax;
|
1172 |
|
1173 | @override
|
1174 | int get hashCode => Object.hash(javaMin, javaDefault, agpMin, agpMax);
|
1175 | }
|
1176 |
|
1177 | class GradleForAgp {
|
1178 | GradleForAgp({required this.agpMin, required this.agpMax, required this.minRequiredGradle});
|
1179 |
|
1180 | final String agpMin;
|
1181 | final String agpMax;
|
1182 | final String minRequiredGradle;
|
1183 | }
|
1184 |
|
1185 | // Returns gradlew file name based on the platform.
|
1186 | String getGradlewFileName(Platform platform) {
|
1187 | if (platform.isWindows) {
|
1188 | return 'gradlew.bat';
|
1189 | } else {
|
1190 | return 'gradlew';
|
1191 | }
|
1192 | }
|
1193 |
|
1194 | /// List of compatible Java/Gradle versions.
|
1195 | ///
|
1196 | /// Should be updated when a new version of Java is supported by a new version
|
1197 | /// of Gradle, as https://docs.gradle.org/current/userguide/compatibility.html
|
1198 | /// details.
|
1199 | var _javaGradleCompatList = const <JavaGradleCompat>[
|
1200 | JavaGradleCompat(
|
1201 | javaMin: '23',
|
1202 | javaMax: '24',
|
1203 | gradleMin: '8.10',
|
1204 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1205 | ),
|
1206 | JavaGradleCompat(
|
1207 | javaMin: '22',
|
1208 | javaMax: '23',
|
1209 | gradleMin: '8.7',
|
1210 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1211 | ),
|
1212 | JavaGradleCompat(
|
1213 | javaMin: '21',
|
1214 | javaMax: '22',
|
1215 | gradleMin: '8.4',
|
1216 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1217 | ),
|
1218 | JavaGradleCompat(
|
1219 | javaMin: '20',
|
1220 | javaMax: '21',
|
1221 | gradleMin: '8.1',
|
1222 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1223 | ),
|
1224 | JavaGradleCompat(
|
1225 | javaMin: '19',
|
1226 | javaMax: '20',
|
1227 | gradleMin: '7.6',
|
1228 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1229 | ),
|
1230 | JavaGradleCompat(
|
1231 | javaMin: '18',
|
1232 | javaMax: '19',
|
1233 | gradleMin: '7.5',
|
1234 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1235 | ),
|
1236 | JavaGradleCompat(
|
1237 | javaMin: '17',
|
1238 | javaMax: '18',
|
1239 | gradleMin: '7.3',
|
1240 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1241 | ),
|
1242 | JavaGradleCompat(
|
1243 | javaMin: '16',
|
1244 | javaMax: '17',
|
1245 | gradleMin: '7.0',
|
1246 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1247 | ),
|
1248 | JavaGradleCompat(
|
1249 | javaMin: '15',
|
1250 | javaMax: '16',
|
1251 | gradleMin: '6.7',
|
1252 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1253 | ),
|
1254 | JavaGradleCompat(
|
1255 | javaMin: '14',
|
1256 | javaMax: '15',
|
1257 | gradleMin: '6.3',
|
1258 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1259 | ),
|
1260 | JavaGradleCompat(
|
1261 | javaMin: '13',
|
1262 | javaMax: '14',
|
1263 | gradleMin: '6.0',
|
1264 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1265 | ),
|
1266 | JavaGradleCompat(
|
1267 | javaMin: '12',
|
1268 | javaMax: '13',
|
1269 | gradleMin: '5.4',
|
1270 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1271 | ),
|
1272 | JavaGradleCompat(
|
1273 | javaMin: '11',
|
1274 | javaMax: '12',
|
1275 | gradleMin: '5.0',
|
1276 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1277 | ),
|
1278 | // 1.11 is a made up java version to cover everything in 1.10.*
|
1279 | JavaGradleCompat(
|
1280 | javaMin: '1.10',
|
1281 | javaMax: '1.11',
|
1282 | gradleMin: '4.7',
|
1283 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1284 | ),
|
1285 | JavaGradleCompat(
|
1286 | javaMin: '1.9',
|
1287 | javaMax: '1.10',
|
1288 | gradleMin: '4.3',
|
1289 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1290 | ),
|
1291 | JavaGradleCompat(
|
1292 | javaMin: '1.8',
|
1293 | javaMax: '1.9',
|
1294 | gradleMin: '2.0',
|
1295 | gradleMax: maxKnownAndSupportedGradleVersion,
|
1296 | ),
|
1297 | ];
|
1298 |
|
1299 | // List of compatible Java/AGP versions, where agpMax versions are inclusive.
|
1300 | //
|
1301 | // Should be updated whenever a new version of AGP is released as
|
1302 | // https://developer.android.com/build/releases/gradle-plugin details.
|
1303 | var _javaAgpCompatList = const <JavaAgpCompat>[
|
1304 | JavaAgpCompat(
|
1305 | javaMin: '17',
|
1306 | javaDefault: '17',
|
1307 | agpMin: '8.0',
|
1308 | agpMax: maxKnownAndSupportedAgpVersion,
|
1309 | ),
|
1310 | JavaAgpCompat(javaMin: '11', javaDefault: '11', agpMin: '7.0', agpMax: '7.4'),
|
1311 | JavaAgpCompat(
|
1312 | // You may use JDK 1.7 with AGP 4.2, but we treat 1.8 as the default since
|
1313 | // it is used by default for this AGP version and lower versions of Java
|
1314 | // are deprecated for executing Gradle.
|
1315 | javaMin: '1.8',
|
1316 | javaDefault: '1.8',
|
1317 | agpMin: '4.2',
|
1318 | agpMax: '4.2',
|
1319 | ),
|
1320 | ];
|
1321 |
|