// Copyright 2014 The Flutter Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. import 'dart:collection'; import 'package:meta/meta.dart'; import 'package:xml/xml.dart'; import 'package:yaml/yaml.dart'; import '../src/convert.dart'; import 'android/android_builder.dart'; import 'android/gradle_utils.dart' as gradle; import 'base/common.dart'; import 'base/error_handling_io.dart'; import 'base/file_system.dart'; import 'base/logger.dart'; import 'base/utils.dart'; import 'base/version.dart'; import 'base/yaml.dart'; import 'bundle.dart' as bundle; import 'cmake_project.dart'; import 'dart/package_map.dart'; import 'features.dart'; import 'flutter_manifest.dart'; import 'flutter_plugins.dart'; import 'globals.dart' as globals; import 'platform_plugins.dart'; import 'project_validator_result.dart'; import 'template.dart'; import 'xcode_project.dart'; export 'cmake_project.dart'; export 'xcode_project.dart'; /// Enum for each officially supported platform. enum SupportedPlatform { android, ios, linux, macos, web, windows, fuchsia, root, // Special platform to represent the root project directory } class FlutterProjectFactory { FlutterProjectFactory({required Logger logger, required FileSystem fileSystem}) : _logger = logger, _fileSystem = fileSystem; final Logger _logger; final FileSystem _fileSystem; @visibleForTesting final projects = {}; /// Returns a [FlutterProject] view of the given directory or a ToolExit error, /// if `pubspec.yaml` or `example/pubspec.yaml` is invalid. FlutterProject fromDirectory(Directory directory) { return projects.putIfAbsent(directory.path, () { final FlutterManifest manifest = FlutterProject._readManifest( directory.childFile(bundle.defaultManifestPath).path, logger: _logger, fileSystem: _fileSystem, ); final FlutterManifest exampleManifest = FlutterProject._readManifest( FlutterProject._exampleDirectory(directory).childFile(bundle.defaultManifestPath).path, logger: _logger, fileSystem: _fileSystem, ); return FlutterProject(directory, manifest, exampleManifest); }); } } /// Represents the contents of a Flutter project at the specified [directory]. /// /// [FlutterManifest] information is read from `pubspec.yaml` and /// `example/pubspec.yaml` files on construction of a [FlutterProject] instance. /// The constructed instance carries an immutable snapshot representation of the /// presence and content of those files. Accordingly, [FlutterProject] instances /// should be discarded upon changes to the `pubspec.yaml` files, but can be /// used across changes to other files, as no other file-level information is /// cached. class FlutterProject { @visibleForTesting FlutterProject(this.directory, this._manifest, this._exampleManifest); /// Returns a [FlutterProject] view of the given directory or a ToolExit error, /// if `pubspec.yaml` or `example/pubspec.yaml` is invalid. static FlutterProject fromDirectory(Directory directory) => globals.projectFactory.fromDirectory(directory); /// Returns a [FlutterProject] view of the current directory or a ToolExit error, /// if `pubspec.yaml` or `example/pubspec.yaml` is invalid. static FlutterProject current() => globals.projectFactory.fromDirectory(globals.fs.currentDirectory); /// Create a [FlutterProject] and bypass the project caching. @visibleForTesting static FlutterProject fromDirectoryTest(Directory directory, [Logger? logger]) { final FileSystem fileSystem = directory.fileSystem; logger ??= BufferLogger.test(); final FlutterManifest manifest = FlutterProject._readManifest( directory.childFile(bundle.defaultManifestPath).path, logger: logger, fileSystem: fileSystem, ); final FlutterManifest exampleManifest = FlutterProject._readManifest( FlutterProject._exampleDirectory(directory).childFile(bundle.defaultManifestPath).path, logger: logger, fileSystem: fileSystem, ); return FlutterProject(directory, manifest, exampleManifest); } /// The location of this project. final Directory directory; /// The location of the build folder. Directory get buildDirectory => directory.childDirectory('build'); /// The manifest of this project. FlutterManifest get manifest => _manifest; late FlutterManifest _manifest; /// The manifest of the example sub-project of this project. final FlutterManifest _exampleManifest; /// List of [FlutterProject]s corresponding to the workspace entries. List get workspaceProjects => manifest.workspace .map( (String entry) => FlutterProject.fromDirectory( directory.childDirectory(directory.fileSystem.path.normalize(entry)), ), ) .toList(); /// The set of organization names found in this project as /// part of iOS product bundle identifier, Android application ID, or /// Gradle group ID. Future> get organizationNames async { final candidates = []; if (ios.existsSync()) { // Don't require iOS build info, this method is only // used during create as best-effort, use the // default target bundle identifier. try { final String? bundleIdentifier = await ios.productBundleIdentifier(null); if (bundleIdentifier != null) { candidates.add(bundleIdentifier); } } on ToolExit { // It's possible that while parsing the build info for the ios project // that the bundleIdentifier can't be resolve. However, we would like // skip parsing that id in favor of searching in other place. We can // consider a tool exit in this case to be non fatal for the program. } } if (android.existsSync()) { final String? applicationId = android.applicationId; final String? group = android.group; candidates.addAll([?applicationId, ?group]); } if (example.android.existsSync()) { final String? applicationId = example.android.applicationId; if (applicationId != null) { candidates.add(applicationId); } } if (example.ios.existsSync()) { final String? bundleIdentifier = await example.ios.productBundleIdentifier(null); if (bundleIdentifier != null) { candidates.add(bundleIdentifier); } } return Set.of( candidates.map(_organizationNameFromPackageName).whereType(), ); } String? _organizationNameFromPackageName(String packageName) { if (0 <= packageName.lastIndexOf('.')) { return packageName.substring(0, packageName.lastIndexOf('.')); } return null; } /// The iOS sub project of this project. late final ios = IosProject.fromFlutter(this); /// The Android sub project of this project. late final android = AndroidProject._(this); /// The web sub project of this project. late final web = WebProject._(this); /// The MacOS sub project of this project. late final macos = MacOSProject.fromFlutter(this); /// The Linux sub project of this project. late final linux = LinuxProject.fromFlutter(this); /// The Windows sub project of this project. late final windows = WindowsProject.fromFlutter(this); /// The Fuchsia sub project of this project. late final fuchsia = FuchsiaProject._(this); /// The `pubspec.yaml` file of this project. File get pubspecFile => directory.childFile('pubspec.yaml'); /// The `.metadata` file of this project. File get metadataFile => directory.childFile('.metadata'); /// The `.flutter-plugins-dependencies` file of this project. /// /// Contains the dependencies each plugin depends on. File get flutterPluginsDependenciesFile => directory.childFile('.flutter-plugins-dependencies'); /// The `.gitignore` file of this project. File get gitignoreFile => directory.childFile('.gitignore'); File get packageConfig => findPackageConfigFileOrDefault(directory); /// The `.dart-tool` directory of this project. Directory get dartTool => directory.childDirectory('.dart_tool'); /// The location of the generated scaffolding project for hosting widget /// previews from this project. // TODO(bkonyi): don't create this project in $TMP. // See https://github.com/flutter/flutter/issues/179036 late final Directory widgetPreviewScaffold = directory.fileSystem.systemTempDirectory .createTempSync('widget_preview_scaffold'); /// The directory containing the generated code for this project. Directory get generated => directory.absolute .childDirectory('.dart_tool') .childDirectory('build') .childDirectory('generated') .childDirectory(manifest.appName); /// The set of directories created by the tool containing ephemeral state. // TODO(bkonyi): provide getters for each project type that returns the set // of known ephemeral files / directories. Set get ephemeralDirectories => UnmodifiableSetView(_ephemeralDirectories); late final _ephemeralDirectories = { buildDirectory, android.ephemeralDirectory, ios.ephemeralDirectory, ios.ephemeralModuleDirectory, ios.symlinks, linux.ephemeralDirectory, macos.ephemeralDirectory, windows.ephemeralDirectory, }; /// The generated Dart plugin registrant for non-web platforms. File get dartPluginRegistrant => dartTool.childDirectory('flutter_build').childFile('dart_plugin_registrant.dart'); /// The example sub-project of this project. FlutterProject get example => FlutterProject( _exampleDirectory(directory), _exampleManifest, FlutterManifest.empty(logger: globals.logger), ); /// The generated scaffolding project for hosting widget previews from this /// project. late final FlutterProject widgetPreviewScaffoldProject = FlutterProject.fromDirectory( widgetPreviewScaffold, ); /// True if this project is a Flutter module project. bool get isModule => manifest.isModule; /// True if this project is a Flutter plugin project. bool get isPlugin => manifest.isPlugin; /// True if the Flutter project is using the AndroidX support library. bool get usesAndroidX => manifest.usesAndroidX; /// True if this project has an example application. bool get hasExampleApp => _exampleDirectory(directory).existsSync(); /// Returns a list of platform names that are supported by the project. List getSupportedPlatforms({bool includeRoot = false}) { return [ if (includeRoot) SupportedPlatform.root, if (android.existsSync()) SupportedPlatform.android, if (ios.exists) SupportedPlatform.ios, if (web.existsSync()) SupportedPlatform.web, if (macos.existsSync()) SupportedPlatform.macos, if (linux.existsSync()) SupportedPlatform.linux, if (windows.existsSync()) SupportedPlatform.windows, if (fuchsia.existsSync()) SupportedPlatform.fuchsia, ]; } /// The directory that will contain the example if an example exists. static Directory _exampleDirectory(Directory directory) => directory.childDirectory('example'); /// Reads and validates the `pubspec.yaml` file at [path], asynchronously /// returning a [FlutterManifest] representation of the contents. /// /// Completes with an empty [FlutterManifest], if the file does not exist. /// Completes with a ToolExit on validation error. static FlutterManifest _readManifest( String path, { required Logger logger, required FileSystem fileSystem, }) { FlutterManifest? manifest; try { manifest = FlutterManifest.createFromPath(path, logger: logger, fileSystem: fileSystem); } on YamlException catch (e) { logger.printStatus('Error detected in pubspec.yaml:', emphasis: true); logger.printError('$e'); } on FormatException catch (e) { logger.printError('Error detected while parsing pubspec.yaml:', emphasis: true); logger.printError('$e'); } on FileSystemException catch (e) { logger.printError('Error detected while reading pubspec.yaml:', emphasis: true); logger.printError('$e'); } if (manifest == null) { throwToolExit('Please correct the pubspec.yaml file at $path'); } return manifest; } /// Reloads the content of [pubspecFile] and updates the contents of [manifest]. void reloadManifest({required Logger logger, required FileSystem fs}) { _manifest = _readManifest(pubspecFile.path, logger: logger, fileSystem: fs); } /// Returns the MD5 hash of the contents of [manifest], ensuring [manifest] is up to date before /// calculating the hash. String computeManifestMD5Hash({required Logger logger, required FileSystem fs}) { reloadManifest(logger: logger, fs: fs); return _manifest.computeMD5Hash(); } /// Replaces the content of [pubspecFile] with the contents of [updated] and /// sets [manifest] to the [updated] manifest. void replacePubspec(FlutterManifest updated) { final YamlMap updatedPubspecContents = updated.toYaml(); pubspecFile.writeAsStringSync(encodeYamlAsString(updatedPubspecContents)); _manifest = updated; } /// Reapplies template files and regenerates project files and plugin /// registrants for app and module projects only. /// /// Will not create project platform directories if they do not already exist. /// /// If [releaseMode] is `true`, platform-specific tooling and metadata generated /// may apply optimizations or changes that are only specific to release builds, /// such as not including dev-only dependencies. Future regeneratePlatformSpecificTooling({ DeprecationBehavior deprecationBehavior = DeprecationBehavior.none, required bool releaseMode, }) async { return ensureReadyForPlatformSpecificTooling( androidPlatform: android.existsSync(), iosPlatform: ios.existsSync(), // TODO(stuartmorgan): Revisit the conditions here once the plans for handling // desktop in existing projects are in place. linuxPlatform: featureFlags.isLinuxEnabled && linux.existsSync(), macOSPlatform: featureFlags.isMacOSEnabled && macos.existsSync(), windowsPlatform: featureFlags.isWindowsEnabled && windows.existsSync(), webPlatform: featureFlags.isWebEnabled && web.existsSync(), deprecationBehavior: deprecationBehavior, releaseMode: releaseMode, ); } /// Applies template files and generates project files and plugin /// registrants for app and module projects only for the specified platforms. /// /// If [releaseMode] is `true`, platform-specific tooling and metadata generated /// may apply optimizations or changes that are only specific to release builds, /// such as not including dev-only dependencies. Future ensureReadyForPlatformSpecificTooling({ required bool releaseMode, bool androidPlatform = false, bool iosPlatform = false, bool linuxPlatform = false, bool macOSPlatform = false, bool windowsPlatform = false, bool webPlatform = false, DeprecationBehavior deprecationBehavior = DeprecationBehavior.none, }) async { if (!directory.existsSync() || isPlugin) { return; } await refreshPluginsList(this, iosPlatform: iosPlatform, macOSPlatform: macOSPlatform); if (androidPlatform) { await android.ensureReadyForPlatformSpecificTooling(deprecationBehavior: deprecationBehavior); } if (iosPlatform) { await ios.ensureReadyForPlatformSpecificTooling(); } if (linuxPlatform) { await linux.ensureReadyForPlatformSpecificTooling(); } if (macOSPlatform) { await macos.ensureReadyForPlatformSpecificTooling(); } if (windowsPlatform) { await windows.ensureReadyForPlatformSpecificTooling(); } if (webPlatform) { await web.ensureReadyForPlatformSpecificTooling(); } await injectPlugins( this, androidPlatform: androidPlatform, iosPlatform: iosPlatform, linuxPlatform: linuxPlatform, macOSPlatform: macOSPlatform, windowsPlatform: windowsPlatform, releaseMode: releaseMode, ); } void checkForDeprecation({DeprecationBehavior deprecationBehavior = DeprecationBehavior.none}) { if (android.existsSync() && pubspecFile.existsSync()) { android.checkForDeprecation(deprecationBehavior: deprecationBehavior); } } /// A JSON encoded string containing the [FlutterManifest.appName], /// [FlutterManifest.buildName] (version), and [FlutterManifest.buildNumber] /// that are used to generate `version.json`. String getVersionInfo() { final String? buildName = manifest.buildName; final String? buildNumber = manifest.buildNumber; final versionFileJson = { 'app_name': manifest.appName, 'version': ?buildName, 'build_number': ?buildNumber, 'package_name': manifest.appName, }; return jsonEncode(versionFileJson); } } /// Base class for projects per platform. abstract class FlutterProjectPlatform { /// Plugin's platform config key, e.g., "macos", "ios". String get pluginConfigKey; /// Whether the platform exists in the project. bool existsSync(); } /// Represents the Android sub-project of a Flutter project. /// /// Instances will reflect the contents of the `android/` sub-folder of /// Flutter applications and the `.android/` sub-folder of Flutter module projects. class AndroidProject extends FlutterProjectPlatform { AndroidProject._(this.parent); // User facing string when java/gradle/agp/kgp versions are compatible. @visibleForTesting static const validJavaGradleAgpKgpString = 'compatible java/gradle/agp/kgp'; // User facing link that describes compatibility between gradle and // android gradle plugin. static const gradleAgpCompatUrl = 'https://developer.android.com/studio/releases/gradle-plugin#updating-gradle'; // User facing link that describes compatibility between java and the first // version of gradle to support it. static const javaGradleCompatUrl = 'https://docs.gradle.org/current/userguide/compatibility.html#java'; // User facing link that describes compatibility between KGP and Gradle // and AGP. static const kgpCompatUrl = 'https://kotlinlang.org/docs/gradle-configure-project.html#apply-the-plugin'; // User facing link that describes instructions for downloading // the latest version of Android Studio. static const installAndroidStudioUrl = 'https://developer.android.com/studio/install'; /// The parent of this project. final FlutterProject parent; @override String get pluginConfigKey => AndroidPlugin.kConfigKey; static final _androidNamespacePattern = RegExp( 'android {[\\S\\s]+namespace\\s*=?\\s*[\'"](.+)[\'"]', ); static final _applicationIdPattern = RegExp('^\\s*applicationId\\s*=?\\s*[\'"](.*)[\'"]\\s*\$'); static final _imperativeKotlinPluginPattern = RegExp( '^\\s*apply plugin\\:\\s+[\'"]kotlin-android[\'"]\\s*\$', ); /// Examples of strings that this regex matches: /// - `id "kotlin-android"` /// - `id("kotlin-android")` /// - `id ( "kotlin-android" ) ` /// - `id "org.jetbrains.kotlin.android"` /// - `id("org.jetbrains.kotlin.android")` /// - `id ( "org.jetbrains.kotlin.android" )` static final _declarativeKotlinPluginPatterns = [ RegExp('^\\s*id\\s*\\(?\\s*[\'"]kotlin-android[\'"]\\s*\\)?\\s*\$'), RegExp('^\\s*id\\s*\\(?\\s*[\'"]org.jetbrains.kotlin.android[\'"]\\s*\\)?\\s*\$'), ]; /// Pattern used to find the assignment of the "group" property in Gradle. /// Expected example: `group "dev.flutter.plugin"` /// Regex is used in both Groovy and Kotlin Gradle files. static final _groupPattern = RegExp('^\\s*group\\s*=?\\s*[\'"](.*)[\'"]\\s*\$'); /// The Gradle root directory of the Android host app. This is the directory /// containing the `app/` subdirectory and the `settings.gradle` file that /// includes it in the overall Gradle project. Directory get hostAppGradleRoot { if (!isModule || _editableHostAppDirectory.existsSync()) { return _editableHostAppDirectory; } return ephemeralDirectory; } /// The Gradle root directory of the Android wrapping of Flutter and plugins. /// This is the same as [hostAppGradleRoot] except when the project is /// a Flutter module with an editable host app. Directory get _flutterLibGradleRoot => isModule ? ephemeralDirectory : _editableHostAppDirectory; Directory get ephemeralDirectory => parent.directory.childDirectory('.android'); Directory get _editableHostAppDirectory => parent.directory.childDirectory('android'); /// True if the parent Flutter project is a module. bool get isModule => parent.isModule; /// True if the parent Flutter project is a plugin. bool get isPlugin => parent.isPlugin; /// True if the Flutter project is using the AndroidX support library. bool get usesAndroidX => parent.usesAndroidX; /// Returns true if the current version of the Gradle plugin is supported. late final bool isSupportedVersion = _computeSupportedVersion(); /// Gets all build variants of this project. Future> getBuildVariants() async { if (!existsSync() || androidBuilder == null) { return const []; } return androidBuilder!.getBuildVariants(project: parent); } /// Outputs app link related settings into a json file. /// /// The return future resolves to the path of the json file. /// /// The future resolves to null if it fails to retrieve app link settings. Future outputsAppLinkSettings({required String variant}) async { if (!existsSync() || androidBuilder == null) { throwToolExit('Target directory $hostAppGradleRoot is not an Android project'); } return androidBuilder!.outputsAppLinkSettings(variant, project: parent); } bool _computeSupportedVersion() { final FileSystem fileSystem = hostAppGradleRoot.fileSystem; final File plugin = hostAppGradleRoot.childFile( fileSystem.path.join('buildSrc', 'src', 'main', 'groovy', 'FlutterPlugin.groovy'), ); if (plugin.existsSync()) { return false; } try { for (final String line in appGradleFile.readAsLinesSync()) { // This syntax corresponds to applying the Flutter Gradle Plugin with a // script. // See https://docs.gradle.org/current/userguide/plugins.html#sec:script_plugins. final bool fileBasedApply = line.contains(RegExp(r'apply from: .*/flutter.gradle')); // This syntax corresponds to applying the Flutter Gradle Plugin using // the declarative "plugins {}" block after including it in the // pluginManagement block of the settings.gradle file. // See https://docs.gradle.org/current/userguide/composite_builds.html#included_plugin_builds, // as well as the settings.gradle and build.gradle templates. final bool declarativeApply = line.contains( RegExp(r'dev\.flutter\.(?:(?:flutter-gradle-plugin)|(?:`flutter-gradle-plugin`))'), ); // This case allows for flutter run/build to work for modules. It does // not guarantee the Flutter Gradle Plugin is applied. final bool managed = line.contains(RegExp('def flutterPluginVersion = [\'"]managed[\'"]')); if (fileBasedApply || declarativeApply || managed) { return true; } } } on FileSystemException { return false; } return false; } /// True, if the app project is using Kotlin. bool get isKotlin { final imperativeMatch = firstMatchInFile(appGradleFile, _imperativeKotlinPluginPattern) != null; final bool declarativeMatch = _declarativeKotlinPluginPatterns.any((RegExp pattern) { return (firstMatchInFile(appGradleFile, pattern) != null); }); return imperativeMatch || declarativeMatch; } /// Gets top-level Gradle build file. /// See https://developer.android.com/build#top-level. /// /// The file must exist and it must be written in either Groovy (build.gradle) /// or Kotlin (build.gradle.kts). File get hostAppGradleFile { return getGroovyOrKotlin(hostAppGradleRoot, 'build.gradle'); } /// Gets the project root level Gradle settings file. /// /// The file must exist and it must be written in either Groovy (settings.gradle) /// or Kotlin (settings.gradle.kts). File get settingsGradleFile { return getGroovyOrKotlin(hostAppGradleRoot, 'settings.gradle'); } File getGroovyOrKotlin(Directory directory, String baseFilename) { final File groovyFile = directory.childFile(baseFilename); final File kotlinFile = directory.childFile('$baseFilename.kts'); if (groovyFile.existsSync()) { // We mimic Gradle's behavior of preferring Groovy over Kotlin when both files exist. return groovyFile; } if (kotlinFile.existsSync()) { return kotlinFile; } // TODO(bartekpacia): An exception should be thrown when neither // the Groovy or Kotlin file exists, instead of falling back to the // Groovy file. See #141180. return groovyFile; } /// Gets the module-level build.gradle file. /// See https://developer.android.com/build#module-level. /// /// The file must exist and it must be written in either Groovy (build.gradle) /// or Kotlin (build.gradle.kts). File get appGradleFile { final Directory appDir = hostAppGradleRoot.childDirectory('app'); return getGroovyOrKotlin(appDir, 'build.gradle'); } File get appManifestFile { if (isUsingGradle) { return hostAppGradleRoot .childDirectory('app') .childDirectory('src') .childDirectory('main') .childFile('AndroidManifest.xml'); } return hostAppGradleRoot.childFile('AndroidManifest.xml'); } /// Gets the Gradle wrapper properties file. /// /// This file is located under `gradle/wrapper/gradle-wrapper.properties` /// in the host app's Gradle root directory. It defines the distribution /// settings for the Gradle wrapper. File get gradleWrapperPropertiesFile { return hostAppGradleRoot .childDirectory('gradle') .childDirectory('wrapper') .childFile('gradle-wrapper.properties'); } File get generatedPluginRegistrantFile { return hostAppGradleRoot .childDirectory('app') .childDirectory('src') .childDirectory('main') .childDirectory('java') .childDirectory('io') .childDirectory('flutter') .childDirectory('plugins') .childFile('GeneratedPluginRegistrant.java'); } File get gradleAppOutV1File => gradleAppOutV1Directory.childFile('app-debug.apk'); Directory get gradleAppOutV1Directory { return globals.fs.directory( globals.fs.path.join(hostAppGradleRoot.path, 'app', 'build', 'outputs', 'apk'), ); } /// Whether the current flutter project has an Android sub-project. @override bool existsSync() { return parent.isModule || _editableHostAppDirectory.existsSync(); } /// Check if the versions of Java, Gradle and AGP are compatible. /// /// This is expected to be called from /// flutter_tools/lib/src/project_validator.dart. Future validateJavaAndGradleAgpVersions() async { // Constructing ProjectValidatorResult happens here and not in // flutter_tools/lib/src/project_validator.dart because of the additional // Complexity of variable status values and error string formatting. const visibleName = 'Java/Gradle/KGP/Android Gradle Plugin'; final CompatibilityResult validJavaGradleAgpVersions = await hasValidJavaGradleAgpVersions(); return ProjectValidatorResult( name: visibleName, value: validJavaGradleAgpVersions.description, status: validJavaGradleAgpVersions.success ? StatusProjectValidator.success : StatusProjectValidator.error, ); } /// Ensures Java SDK is compatible with the project's Gradle version and /// the project's Gradle version is compatible with the AGP version and /// kotlin version used in build.gradle. Future hasValidJavaGradleAgpVersions() async { final String? gradleVersion = await gradle.getGradleVersion( hostAppGradleRoot, globals.logger, globals.processManager, ); final String? agpVersion = gradle.getAgpVersion(hostAppGradleRoot, globals.logger); final String? javaVersion = versionToParsableString(globals.java?.version); final String? kgpVersion = await gradle.getKgpVersion( hostAppGradleRoot, globals.logger, globals.processManager, ); // Assume valid configuration. String description = validJavaGradleAgpKgpString; final bool compatibleGradleAgp = gradle.validateGradleAndAgp( globals.logger, gradleV: gradleVersion, agpV: agpVersion, ); final bool compatibleJavaGradle = gradle.validateJavaAndGradle( globals.logger, javaVersion: javaVersion, gradleVersion: gradleVersion, ); final bool compatibleKgpGradle = gradle.validateGradleAndKGP( globals.logger, gradleV: gradleVersion, kgpV: kgpVersion, ); final bool compatibleAgpKgp = gradle.validateAgpAndKgp( globals.logger, agpV: agpVersion, kgpV: kgpVersion, ); // Begin description formatting. if (!compatibleGradleAgp) { final gradleDescription = agpVersion != null ? 'Update Gradle to at least "${gradle.getGradleVersionFor(agpVersion)}".' : ''; description = ''' Incompatible Gradle/AGP versions. \n Gradle Version: $gradleVersion, AGP Version: $agpVersion $gradleDescription\n See the link below for more information: $gradleAgpCompatUrl '''; } if (!compatibleJavaGradle) { // Should contain the agp error (if present) but not the valid String. description = ''' ${compatibleGradleAgp ? '' : description} Incompatible Java/Gradle versions. Java Version: $javaVersion, Gradle Version: $gradleVersion\n See the link below for more information: $javaGradleCompatUrl '''; } if (!compatibleKgpGradle) { description = ''' ${compatibleGradleAgp ? '' : description} Incompatible KGP/Gradle versions. Gradle Version: $gradleVersion, Kotlin Version: $kgpVersion\n See the link below for more information: $kgpCompatUrl '''; } if (!compatibleAgpKgp) { description = ''' ${compatibleGradleAgp ? '' : description} Incompatible AGP/KGP versions. AGP Version: $agpVersion, KGP Version: $kgpVersion\n See the link below for more information: $kgpCompatUrl '''; } return CompatibilityResult( compatibleJavaGradle && compatibleGradleAgp && compatibleKgpGradle && compatibleAgpKgp, description, ); } bool get isUsingGradle { return hostAppGradleFile.existsSync(); } String? get applicationId { return firstMatchInFile(appGradleFile, _applicationIdPattern)?.group(1); } /// Get the namespace for newer Android projects, /// which replaces the `package` attribute in the Manifest.xml. String? get namespace { try { // firstMatchInFile() reads per line but `_androidNamespacePattern` matches a multiline pattern. return _androidNamespacePattern.firstMatch(appGradleFile.readAsStringSync())?.group(1); } on FileSystemException { return null; } } String? get group { return firstMatchInFile(hostAppGradleFile, _groupPattern)?.group(1); } /// The build directory where the Android artifacts are placed. Directory get buildDirectory { return parent.buildDirectory; } Future ensureReadyForPlatformSpecificTooling({ DeprecationBehavior deprecationBehavior = DeprecationBehavior.none, }) async { if (isModule && _shouldRegenerateFromTemplate()) { await _regenerateLibrary(); // Add ephemeral host app, if an editable host app does not already exist. if (!_editableHostAppDirectory.existsSync()) { await _overwriteFromTemplate( globals.fs.path.join('module', 'android', 'host_app_common'), ephemeralDirectory, ); await _overwriteFromTemplate( globals.fs.path.join('module', 'android', 'host_app_ephemeral'), ephemeralDirectory, ); } } if (!hostAppGradleRoot.existsSync()) { return; } gradle.updateLocalProperties(project: parent, requireAndroidSdk: false); } bool _shouldRegenerateFromTemplate() { return globals.fsUtils.isOlderThanReference( entity: ephemeralDirectory, referenceFile: parent.pubspecFile, ) || globals.cache.isOlderThanToolsStamp(ephemeralDirectory); } File get localPropertiesFile => _flutterLibGradleRoot.childFile('local.properties'); Directory get pluginRegistrantHost => _flutterLibGradleRoot.childDirectory(isModule ? 'Flutter' : 'app'); Future _regenerateLibrary() async { ErrorHandlingFileSystem.deleteIfExists(ephemeralDirectory, recursive: true); await _overwriteFromTemplate( globals.fs.path.join('module', 'android', 'library_new_embedding'), ephemeralDirectory, ); await _overwriteFromTemplate( globals.fs.path.join('module', 'android', 'gradle'), ephemeralDirectory, ); globals.gradleUtils?.injectGradleWrapperIfNeeded(ephemeralDirectory); } Future _overwriteFromTemplate(String path, Directory target) async { final Template template = await Template.fromName( path, fileSystem: globals.fs, templateManifest: null, logger: globals.logger, templateRenderer: globals.templateRenderer, ); final String androidIdentifier = parent.manifest.androidPackage ?? 'com.example.${parent.manifest.appName}'; template.render(target, { 'android': true, 'projectName': parent.manifest.appName, 'androidIdentifier': androidIdentifier, 'androidX': usesAndroidX, 'agpVersion': gradle.templateAndroidGradlePluginVersion, 'agpVersionForModule': gradle.templateAndroidGradlePluginVersionForModule, 'kotlinVersion': gradle.templateKotlinGradlePluginVersion, 'gradleVersion': gradle.templateDefaultGradleVersion, 'compileSdkVersion': gradle.compileSdkVersion, 'minSdkVersion': gradle.minSdkVersion, 'ndkVersion': gradle.ndkVersion, 'targetSdkVersion': gradle.targetSdkVersion, }, printStatusWhenWriting: false); } void checkForDeprecation({DeprecationBehavior deprecationBehavior = DeprecationBehavior.none}) { if (deprecationBehavior == DeprecationBehavior.none) { return; } final AndroidEmbeddingVersionResult result = computeEmbeddingVersion(); if (result.version != AndroidEmbeddingVersion.v1) { return; } // The v1 android embedding has been deleted. throwToolExit('Build failed due to use of deleted Android v1 embedding.', exitCode: 1); } AndroidEmbeddingVersion getEmbeddingVersion() { final AndroidEmbeddingVersion androidEmbeddingVersion = computeEmbeddingVersion().version; if (androidEmbeddingVersion == AndroidEmbeddingVersion.v1) { throwToolExit('Build failed due to use of deleted Android v1 embedding.', exitCode: 1); } return androidEmbeddingVersion; } AndroidEmbeddingVersionResult computeEmbeddingVersion() { if (isModule) { // A module type's Android project is used in add-to-app scenarios and // only supports the V2 embedding. return AndroidEmbeddingVersionResult(AndroidEmbeddingVersion.v2, 'Is add-to-app module'); } if (isPlugin) { // Plugins do not use an appManifest, so we stop here. // // TODO(garyq): This method does not currently check for code references to // the v1 embedding, we should check for this once removal is further along. return AndroidEmbeddingVersionResult(AndroidEmbeddingVersion.v2, 'Is plugin'); } if (!appManifestFile.existsSync()) { return AndroidEmbeddingVersionResult( AndroidEmbeddingVersion.v1, 'No `${appManifestFile.absolute.path}` file', ); } XmlDocument document; try { document = XmlDocument.parse(appManifestFile.readAsStringSync()); } on XmlException { throwToolExit( 'Error parsing $appManifestFile ' 'Please ensure that the android manifest is a valid XML document and try again.', ); } on FileSystemException { throwToolExit( 'Error reading $appManifestFile even though it exists. ' 'Please ensure that you have read permission to this file and try again.', ); } for (final XmlElement application in document.findAllElements('application')) { final String? applicationName = application.getAttribute('android:name'); if (applicationName == 'io.flutter.app.FlutterApplication') { return AndroidEmbeddingVersionResult( AndroidEmbeddingVersion.v1, '${appManifestFile.absolute.path} uses `android:name="io.flutter.app.FlutterApplication"`', ); } } for (final XmlElement metaData in document.findAllElements('meta-data')) { final String? name = metaData.getAttribute('android:name'); // External code checks for this string to identify flutter android apps. // See cl/667760684 as an example. if (name == 'flutterEmbedding') { final String? embeddingVersionString = metaData.getAttribute('android:value'); if (embeddingVersionString == '1') { return AndroidEmbeddingVersionResult( AndroidEmbeddingVersion.v1, '${appManifestFile.absolute.path} `` in ${appManifestFile.absolute.path}', ); } static const _impellerEnabledByDefault = true; /// Returns the `io.flutter.embedding.android.EnableImpeller` manifest value. /// /// If there is no manifest file, or the key is not present, returns `false`. bool computeImpellerEnabled() { if (!appManifestFile.existsSync()) { return _impellerEnabledByDefault; } final XmlDocument document; try { document = XmlDocument.parse(appManifestFile.readAsStringSync()); } on XmlException { throwToolExit( 'Error parsing $appManifestFile ' 'Please ensure that the android manifest is a valid XML document and try again.', ); } on FileSystemException { throwToolExit( 'Error reading $appManifestFile even though it exists. ' 'Please ensure that you have read permission to this file and try again.', ); } for (final XmlElement metaData in document.findAllElements('meta-data')) { final String? name = metaData.getAttribute('android:name'); if (name == 'io.flutter.embedding.android.EnableImpeller') { final String? value = metaData.getAttribute('android:value'); if (value == 'true') { return true; } if (value == 'false') { return false; } } } return _impellerEnabledByDefault; } } /// Iteration of the embedding Java API in the engine used by the Android project. enum AndroidEmbeddingVersion { /// V1 APIs based on io.flutter.app.FlutterActivity. v1, /// V2 APIs based on io.flutter.embedding.android.FlutterActivity. v2, } /// Data class that holds the results of checking for embedding version. /// /// This class includes the reason why a particular embedding was selected. class AndroidEmbeddingVersionResult { AndroidEmbeddingVersionResult(this.version, this.reason); /// The embedding version. AndroidEmbeddingVersion version; /// The reason why the embedding version was selected. String reason; } // What the tool should do when encountering deprecated API in applications. enum DeprecationBehavior { // The command being run does not care about deprecation status. none, // The command should continue and ignore the deprecation warning. ignore, // The command should exit the tool. exit, } /// Represents the web sub-project of a Flutter project. class WebProject extends FlutterProjectPlatform { WebProject._(this.parent); final FlutterProject parent; @override String get pluginConfigKey => WebPlugin.kConfigKey; /// Whether this flutter project has a web sub-project. @override bool existsSync() { return parent.directory.childDirectory('web').existsSync() && indexFile.existsSync(); } /// The 'lib' directory for the application. Directory get libDirectory => parent.directory.childDirectory('lib'); /// The directory containing additional files for the application. Directory get directory => parent.directory.childDirectory('web'); /// The html file used to host the flutter web application. File get indexFile => parent.directory.childDirectory('web').childFile('index.html'); /// The .dart_tool/dartpad directory Directory get dartpadToolDirectory => parent.directory.childDirectory('.dart_tool').childDirectory('dartpad'); Future ensureReadyForPlatformSpecificTooling() async { /// Create .dart_tool/dartpad/web_plugin_registrant.dart. /// See: https://github.com/dart-lang/dart-services/pull/874 await injectBuildTimePluginFilesForWebPlatform(parent, destination: dartpadToolDirectory); } } /// The Fuchsia sub project. class FuchsiaProject { FuchsiaProject._(this.project); final FlutterProject project; Directory? _editableHostAppDirectory; Directory get editableHostAppDirectory => _editableHostAppDirectory ??= project.directory.childDirectory('fuchsia'); bool existsSync() => editableHostAppDirectory.existsSync(); Directory? _meta; Directory get meta => _meta ??= editableHostAppDirectory.childDirectory('meta'); } // Combines success and a description into one object that can be returned // together. @visibleForTesting class CompatibilityResult { CompatibilityResult(this.success, this.description); final bool success; final String description; } /// Converts a [Version] to a string that can be parsed by [Version.parse]. String? versionToParsableString(Version? version) { if (version == null) { return null; } return '${version.major}.${version.minor}.${version.patch}'; }