| // Copyright 2013 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 'package:engine_build_configs/engine_build_configs.dart'; |
| |
| import '../build_plan.dart'; |
| import '../build_utils.dart'; |
| import '../gn.dart'; |
| import '../label.dart'; |
| import 'command.dart'; |
| import 'flags.dart'; |
| |
| // ignore: public_member_api_docs |
| final class QueryCommand extends CommandBase { |
| // ignore: public_member_api_docs |
| QueryCommand({ |
| required super.environment, |
| required this.configs, |
| super.help = false, |
| super.usageLineLength, |
| }) { |
| // Add options here that are common to all queries. |
| argParser |
| ..addFlag( |
| allFlag, |
| abbr: 'a', |
| help: 'List all results, even when not relevant on this platform', |
| negatable: false, |
| ) |
| ..addOption( |
| builderFlag, |
| abbr: 'b', |
| help: 'Restrict the query to a single builder.', |
| allowed: <String>[ |
| for (final MapEntry<String, BuilderConfig> entry in configs.entries) |
| if (entry.value.canRunOn(environment.platform)) entry.key, |
| ], |
| allowedHelp: <String, String>{ |
| for (final MapEntry<String, BuilderConfig> entry in configs.entries) |
| if (entry.value.canRunOn(environment.platform)) |
| entry.key: entry.value.path, |
| }, |
| ); |
| |
| addSubcommand(QueryBuildersCommand( |
| environment: environment, |
| configs: configs, |
| help: help, |
| )); |
| addSubcommand(QueryTargetsCommand( |
| environment: environment, |
| configs: configs, |
| help: help, |
| )); |
| } |
| |
| /// Build configurations loaded from the engine from under ci/builders. |
| final Map<String, BuilderConfig> configs; |
| |
| @override |
| String get name => 'query'; |
| |
| @override |
| String get description => 'Provides information about build configurations ' |
| 'and tests.'; |
| } |
| |
| /// The 'query builders' command. |
| final class QueryBuildersCommand extends CommandBase { |
| /// Constructs the 'query builders' command. |
| QueryBuildersCommand({ |
| required super.environment, |
| required this.configs, |
| super.help = false, |
| }); |
| |
| /// Build configurations loaded from the engine from under ci/builders. |
| final Map<String, BuilderConfig> configs; |
| |
| @override |
| String get name => 'builders'; |
| |
| @override |
| String get description => 'Provides information about CI builder ' |
| 'configurations'; |
| |
| @override |
| Future<int> run() async { |
| // Loop through all configs, and log those that are compatible with the |
| // current platform. |
| final bool all = parent!.argResults![allFlag]! as bool; |
| final String? builderName = parent!.argResults![builderFlag] as String?; |
| if (!environment.verbose) { |
| environment.logger.status( |
| 'Add --verbose to see detailed information about each builder', |
| ); |
| environment.logger.status(''); |
| } |
| for (final String key in configs.keys) { |
| if (builderName != null && key != builderName) { |
| continue; |
| } |
| |
| final BuilderConfig config = configs[key]!; |
| if (!config.canRunOn(environment.platform) && !all) { |
| continue; |
| } |
| |
| environment.logger.status('"$key" builder:'); |
| for (final Build build in config.builds) { |
| if (!build.canRunOn(environment.platform) && !all) { |
| continue; |
| } |
| environment.logger.status('"${build.name}" config', indent: 3); |
| if (!environment.verbose) { |
| continue; |
| } |
| environment.logger.status('gn flags:', indent: 6); |
| for (final String flag in build.gn) { |
| environment.logger.status(flag, indent: 9); |
| } |
| if (build.ninja.targets.isNotEmpty) { |
| environment.logger.status('ninja targets:', indent: 6); |
| for (final String target in build.ninja.targets) { |
| environment.logger.status(target, indent: 9); |
| } |
| } |
| } |
| } |
| return 0; |
| } |
| } |
| |
| /// The query targets command. |
| final class QueryTargetsCommand extends CommandBase { |
| /// Constructs the 'query targets' command. |
| QueryTargetsCommand({ |
| required super.environment, |
| required this.configs, |
| super.help = false, |
| }) { |
| builds = BuildPlan.configureArgParser( |
| argParser, |
| environment, |
| configs: configs, |
| help: help, |
| ); |
| argParser.addFlag( |
| testOnlyFlag, |
| abbr: 't', |
| help: 'Filter build targets to only include tests', |
| negatable: false, |
| ); |
| } |
| |
| /// Build configurations loaded from the engine from under ci/builders. |
| final Map<String, BuilderConfig> configs; |
| |
| /// List of compatible builds. |
| late final List<Build> builds; |
| |
| @override |
| String get name => 'targets'; |
| |
| @override |
| String get description => ''' |
| Provides information about build targets |
| et query targets --testonly # List only test targets |
| et query targets //flutter/fml/... # List all targets under `//flutter/fml` |
| '''; |
| |
| @override |
| Future<int> run() async { |
| final plan = BuildPlan.fromArgResults( |
| argResults!, |
| environment, |
| builds: builds, |
| ); |
| |
| if (!await ensureBuildDir( |
| environment, |
| plan.build, |
| enableRbe: plan.useRbe, |
| )) { |
| return 1; |
| } |
| |
| // Builds only accept labels as arguments, so convert patterns to labels. |
| // TODO(matanlurey): Can be optimized in cases where wildcards are not used. |
| final gn = Gn.fromEnvironment(environment); |
| |
| // TODO(matanlurey): Discuss if we want to just require '//...'. |
| // For now this retains the existing behavior. |
| var patterns = argResults!.rest; |
| if (patterns.isEmpty) { |
| patterns = ['//...']; |
| } |
| |
| final allTargets = <BuildTarget>{}; |
| for (final pattern in patterns) { |
| final target = TargetPattern.parse(pattern); |
| final targets = await gn.desc( |
| 'out/${plan.build.ninja.config}', |
| target, |
| ); |
| allTargets.addAll(targets); |
| } |
| |
| if (allTargets.isEmpty) { |
| environment.logger.error('No targets found, nothing to query.'); |
| return 1; |
| } |
| |
| final testOnly = argResults!.flag(testOnlyFlag); |
| for (final target in allTargets) { |
| if (testOnly && (!target.testOnly || target is! ExecutableBuildTarget)) { |
| continue; |
| } |
| environment.logger.status(target.label); |
| } |
| return 0; |
| } |
| } |