packages.dart 12.1 KB
Newer Older
Ian Hickson's avatar
Ian Hickson committed
1
// Copyright 2014 The Flutter Authors. All rights reserved.
2 3 4
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5 6
// @dart = 2.8

7 8
import 'package:args/args.dart';

9
import '../base/common.dart';
10
import '../base/os.dart';
11 12 13 14
import '../build_info.dart';
import '../build_system/build_system.dart';
import '../cache.dart';
import '../dart/generate_synthetic_packages.dart';
15
import '../dart/pub.dart';
16
import '../flutter_plugins.dart';
17
import '../globals_null_migrated.dart' as globals;
18
import '../plugins.dart';
19
import '../project.dart';
20
import '../reporting/reporting.dart';
21 22 23 24
import '../runner/flutter_command.dart';

class PackagesCommand extends FlutterCommand {
  PackagesCommand() {
25
    addSubcommand(PackagesGetCommand('get', false));
26
    addSubcommand(PackagesInteractiveGetCommand('upgrade', "Upgrade the current package's dependencies to latest versions."));
27 28
    addSubcommand(PackagesInteractiveGetCommand('add', 'Add a dependency to pubspec.yaml.'));
    addSubcommand(PackagesInteractiveGetCommand('remove', 'Removes a dependency from the current package.'));
29
    addSubcommand(PackagesTestCommand());
30
    addSubcommand(PackagesForwardCommand('publish', 'Publish the current package to pub.dartlang.org', requiresPubspec: true));
31 32 33 34 35
    addSubcommand(PackagesForwardCommand('downgrade', 'Downgrade packages in a Flutter project', requiresPubspec: true));
    addSubcommand(PackagesForwardCommand('deps', 'Print package dependencies', requiresPubspec: true));
    addSubcommand(PackagesForwardCommand('run', 'Run an executable from a package', requiresPubspec: true));
    addSubcommand(PackagesForwardCommand('cache', 'Work with the Pub system cache'));
    addSubcommand(PackagesForwardCommand('version', 'Print Pub version'));
36
    addSubcommand(PackagesForwardCommand('uploader', 'Manage uploaders for a package on pub.dev'));
37 38
    addSubcommand(PackagesForwardCommand('login', 'Log into pub.dev.'));
    addSubcommand(PackagesForwardCommand('logout', 'Log out of pub.dev.'));
39
    addSubcommand(PackagesForwardCommand('global', 'Work with Pub global packages'));
40
    addSubcommand(PackagesForwardCommand('outdated', 'Analyze dependencies to find which ones can be upgraded', requiresPubspec: true));
41
    addSubcommand(PackagesPassthroughCommand());
42 43 44
  }

  @override
45
  final String name = 'pub';
46 47

  @override
48
  List<String> get aliases => const <String>['packages'];
49 50 51 52

  @override
  final String description = 'Commands for managing Flutter packages.';

53
  @override
54
  Future<FlutterCommandResult> runCommand() async => null;
55 56 57
}

class PackagesGetCommand extends FlutterCommand {
58
  PackagesGetCommand(this.name, this.upgrade) {
59
    requiresPubspecYaml();
60 61
    argParser.addFlag('offline',
      negatable: false,
62
      help: 'Use cached packages instead of accessing the network.',
63 64
    );
  }
65

66 67 68 69 70
  @override
  final String name;

  final bool upgrade;

71
  @override
72 73 74
  String get description {
    return '${ upgrade ? "Upgrade" : "Get" } packages in a Flutter project.';
  }
75 76

  @override
77
  String get invocation {
78
    return '${runner.executableName} pub $name [<target directory>]';
79
  }
80

81 82
  /// The pub packages usage values are incorrect since these are calculated/sent
  /// before pub get completes. This needs to be performed after dependency resolution.
83
  @override
84
  Future<CustomDimensions> get usageValues async {
85
    final String workingDirectory = argResults.rest.length == 1 ? argResults.rest[0] : null;
86
    final String target = findProjectRoot(globals.fs, workingDirectory);
87
    if (target == null) {
88
      return const CustomDimensions();
89
    }
90 91 92

    int numberPlugins;

93
    final FlutterProject rootProject = FlutterProject.fromDirectory(globals.fs.directory(target));
94 95 96 97
    // Do not send plugin analytics if pub has not run before.
    final bool hasPlugins = rootProject.flutterPluginsDependenciesFile.existsSync()
      && rootProject.packagesFile.existsSync()
      && rootProject.packageConfigFile.existsSync();
98
    if (hasPlugins) {
99 100 101
      // Do not fail pub get if package config files are invalid before pub has
      // had a chance to run.
      final List<Plugin> plugins = await findPlugins(rootProject, throwOnError: false);
102
      numberPlugins = plugins.length;
103
    } else {
104
      numberPlugins = 0;
105
    }
106 107 108 109 110 111

    return CustomDimensions(
      commandPackagesNumberPlugins: numberPlugins,
      commandPackagesProjectModule: rootProject.isModule,
      commandPackagesAndroidEmbeddingVersion: rootProject.android.getEmbeddingVersion().toString().split('.').last,
    );
112 113
  }

114
  Future<void> _runPubGet(String directory, FlutterProject flutterProject) async {
115 116 117 118 119 120 121 122 123 124
    if (flutterProject.manifest.generateSyntheticPackage) {
      final Environment environment = Environment(
        artifacts: globals.artifacts,
        logger: globals.logger,
        cacheDir: globals.cache.getRoot(),
        engineVersion: globals.flutterVersion.engineRevision,
        fileSystem: globals.fs,
        flutterRootDir: globals.fs.directory(Cache.flutterRoot),
        outputDir: globals.fs.directory(getBuildDirectory()),
        processManager: globals.processManager,
125
        platform: globals.platform,
126
        projectDir: flutterProject.directory,
127
        generateDartPluginRegistry: true,
128 129 130 131 132 133 134 135
      );

      await generateLocalizationsSyntheticPackage(
        environment: environment,
        buildSystem: globals.buildSystem,
      );
    }

136 137
    final Stopwatch pubGetTimer = Stopwatch()..start();
    try {
138 139
      await pub.get(
        context: PubContext.pubGet,
140
        directory: directory,
141
        upgrade: upgrade,
142
        offline: boolArg('offline'),
143
        generateSyntheticPackage: flutterProject.manifest.generateSyntheticPackage,
144 145
      );
      pubGetTimer.stop();
146
      globals.flutterUsage.sendTiming('pub', 'get', pubGetTimer.elapsed, label: 'success');
147 148
    // Not limiting to catching Exception because the exception is rethrown.
    } catch (_) { // ignore: avoid_catches_without_on_clauses
149
      pubGetTimer.stop();
150
      globals.flutterUsage.sendTiming('pub', 'get', pubGetTimer.elapsed, label: 'failure');
151 152
      rethrow;
    }
153 154
  }

155
  @override
156
  Future<FlutterCommandResult> runCommand() async {
157
    if (argResults.rest.length > 1) {
158
      throwToolExit('Too many arguments.\n$usage');
159
    }
160

161
    final String workingDirectory = argResults.rest.length == 1 ? argResults.rest[0] : null;
162
    final String target = findProjectRoot(globals.fs, workingDirectory);
163
    if (target == null) {
164
      throwToolExit(
165
       'Expected to find project root in '
166
       '${ workingDirectory ?? "current working directory" }.'
167 168
      );
    }
169
    final FlutterProject rootProject = FlutterProject.fromDirectory(globals.fs.directory(target));
170 171

    await _runPubGet(target, rootProject);
172
    await rootProject.regeneratePlatformSpecificTooling();
173 174

    // Get/upgrade packages in example app as well
175
    if (rootProject.hasExampleApp && rootProject.example.pubspecFile.existsSync()) {
176
      final FlutterProject exampleProject = rootProject.example;
177
      await _runPubGet(exampleProject.directory.path, exampleProject);
178
      await exampleProject.regeneratePlatformSpecificTooling();
179
    }
180

181
    return FlutterCommandResult.success();
182 183
  }
}
184 185

class PackagesTestCommand extends FlutterCommand {
186 187 188 189
  PackagesTestCommand() {
    requiresPubspecYaml();
  }

190 191 192 193 194 195
  @override
  String get name => 'test';

  @override
  String get description {
    return 'Run the "test" package.\n'
196 197 198 199
           'This is similar to "flutter test", but instead of hosting the tests in the '
           'flutter environment it hosts the tests in a pure Dart environment. The main '
           'differences are that the "dart:ui" library is not available and that tests '
           'run faster. This is helpful for testing libraries that do not depend on any '
200 201 202 203 204
           'packages from the Flutter SDK. It is equivalent to "pub run test".';
  }

  @override
  String get invocation {
205
    return '${runner.executableName} pub test [<tests...>]';
206 207 208
  }

  @override
209
  Future<FlutterCommandResult> runCommand() async {
210
    await pub.batch(<String>['run', 'test', ...argResults.rest], context: PubContext.runTest, retry: false);
211
    return FlutterCommandResult.success();
212
  }
213 214
}

215 216 217 218 219 220
class PackagesForwardCommand extends FlutterCommand {
  PackagesForwardCommand(this._commandName, this._description, {bool requiresPubspec = false}) {
    if (requiresPubspec) {
      requiresPubspecYaml();
    }
  }
221 222 223 224

  @override
  ArgParser argParser = ArgParser.allowAnything();

225 226 227 228 229 230 231 232 233 234 235 236 237 238
  final String _commandName;
  final String _description;

  @override
  String get name => _commandName;

  @override
  String get description {
    return '$_description.\n'
           'This runs the "pub" tool in a Flutter context.';
  }

  @override
  String get invocation {
239
    return '${runner.executableName} pub $_commandName [<arguments...>]';
240 241 242 243
  }

  @override
  Future<FlutterCommandResult> runCommand() async {
244 245 246
    final List<String> subArgs = argResults.rest.toList()
      ..removeWhere((String arg) => arg == '--');
    await pub.interactively(<String>[_commandName, ...subArgs], stdio: globals.stdio);
247
    return FlutterCommandResult.success();
248 249 250
  }
}

251
class PackagesPassthroughCommand extends FlutterCommand {
252 253 254
  PackagesPassthroughCommand() {
    requiresPubspecYaml();
  }
255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270

  @override
  String get name => 'pub';

  @override
  String get description {
    return 'Pass the remaining arguments to Dart\'s "pub" tool.\n'
           'This runs the "pub" tool in a Flutter context.';
  }

  @override
  String get invocation {
    return '${runner.executableName} packages pub [<arguments...>]';
  }

  @override
271
  Future<FlutterCommandResult> runCommand() async {
272
    await pub.interactively(argResults.rest, stdio: globals.stdio);
273
    return FlutterCommandResult.success();
274
  }
275
}
276 277

class PackagesInteractiveGetCommand extends FlutterCommand {
278
  PackagesInteractiveGetCommand(this._commandName, this._description);
279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302

  @override
  ArgParser argParser = ArgParser.allowAnything();

  final String _commandName;
  final String _description;

  @override
  String get name => _commandName;

  @override
  String get description {
    return '$_description.\n'
           'This runs the "pub" tool in a Flutter context.';
  }

  @override
  String get invocation {
    return '${runner.executableName} pub $_commandName [<arguments...>]';
  }

  @override
  Future<FlutterCommandResult> runCommand() async {
    List<String> rest = argResults.rest;
303
    final bool isHelp = rest.contains('-h') || rest.contains('--help');
304
    String target;
305
    if (rest.length == 1 && (rest[0].contains('/') || rest[0].contains(r'\'))) {
306 307
      // HACK: Supporting flutter specific behavior where you can pass a
      //       folder to the command.
308
      target = findProjectRoot(globals.fs, rest[0]);
309 310
      rest = <String>[];
    } else {
311
      target = findProjectRoot(globals.fs);
312 313
    }

314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332
    FlutterProject flutterProject;
    if (!isHelp) {
      if (target == null) {
        throwToolExit('Expected to find project root in current working directory.');
      }
      flutterProject = FlutterProject.fromDirectory(globals.fs.directory(target));

      if (flutterProject.manifest.generateSyntheticPackage) {
        final Environment environment = Environment(
          artifacts: globals.artifacts,
          logger: globals.logger,
          cacheDir: globals.cache.getRoot(),
          engineVersion: globals.flutterVersion.engineRevision,
          fileSystem: globals.fs,
          flutterRootDir: globals.fs.directory(Cache.flutterRoot),
          outputDir: globals.fs.directory(getBuildDirectory()),
          processManager: globals.processManager,
          platform: globals.platform,
          projectDir: flutterProject.directory,
333
          generateDartPluginRegistry: true,
334 335 336 337 338 339 340
        );

        await generateLocalizationsSyntheticPackage(
          environment: environment,
          buildSystem: globals.buildSystem,
        );
      }
341 342
    }

343
    final List<String> subArgs = rest.toList()..removeWhere((String arg) => arg == '--');
344 345 346 347
    await pub.interactively(
      <String>[name, ...subArgs],
      directory: target,
      stdio: globals.stdio,
348 349
      touchesPackageConfig: !isHelp,
      generateSyntheticPackage: flutterProject?.manifest?.generateSyntheticPackage ?? false,
350 351
    );

352
    await flutterProject?.regeneratePlatformSpecificTooling();
353 354 355
    return FlutterCommandResult.success();
  }
}