| // 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 'package:meta/meta.dart'; |
| |
| import '../base/common.dart'; |
| import '../base/io.dart'; |
| import '../base/os.dart'; |
| import '../base/process.dart'; |
| import '../cache.dart'; |
| import '../dart/pub.dart'; |
| import '../globals.dart' as globals; |
| import '../persistent_tool_state.dart'; |
| import '../project.dart'; |
| import '../runner/flutter_command.dart'; |
| import '../version.dart'; |
| import 'channel.dart'; |
| |
| // The official docs to install Flutter. |
| const String _flutterInstallDocs = 'https://flutter.dev/docs/get-started/install'; |
| |
| class UpgradeCommand extends FlutterCommand { |
| UpgradeCommand({ |
| required bool verboseHelp, |
| UpgradeCommandRunner? commandRunner, |
| }) |
| : _commandRunner = commandRunner ?? UpgradeCommandRunner() { |
| argParser |
| ..addFlag( |
| 'force', |
| abbr: 'f', |
| help: 'Force upgrade the flutter branch, potentially discarding local changes.', |
| negatable: false, |
| ) |
| ..addFlag( |
| 'continue', |
| hide: !verboseHelp, |
| negatable: false, |
| help: 'Trigger the second half of the upgrade flow. This should not be invoked ' |
| 'manually. It is used re-entrantly by the standard upgrade command after ' |
| 'the new version of Flutter is available, to hand off the upgrade process ' |
| 'from the old version to the new version.', |
| ) |
| ..addOption( |
| 'working-directory', |
| hide: !verboseHelp, |
| help: 'Override the upgrade working directory. ' |
| 'This is only intended to enable integration testing of the tool itself.' |
| // Also notably, this will override the FakeFlutterVersion if any is set! |
| ) |
| ..addFlag( |
| 'verify-only', |
| help: 'Checks for any new Flutter updates, without actually fetching them.', |
| negatable: false, |
| ); |
| } |
| |
| final UpgradeCommandRunner _commandRunner; |
| |
| @override |
| final String name = 'upgrade'; |
| |
| @override |
| final String description = 'Upgrade your copy of Flutter.'; |
| |
| @override |
| final String category = FlutterCommandCategory.sdk; |
| |
| @override |
| bool get shouldUpdateCache => false; |
| |
| @override |
| Future<FlutterCommandResult> runCommand() { |
| _commandRunner.workingDirectory = stringArg('working-directory') ?? Cache.flutterRoot!; |
| return _commandRunner.runCommand( |
| force: boolArg('force'), |
| continueFlow: boolArg('continue'), |
| testFlow: stringArg('working-directory') != null, |
| gitTagVersion: GitTagVersion.determine( |
| globals.processUtils, |
| globals.platform, |
| workingDirectory: _commandRunner.workingDirectory, |
| ), |
| flutterVersion: stringArg('working-directory') == null |
| ? globals.flutterVersion |
| : FlutterVersion(flutterRoot: _commandRunner.workingDirectory!, fs: globals.fs), |
| verifyOnly: boolArg('verify-only'), |
| ); |
| } |
| } |
| |
| @visibleForTesting |
| class UpgradeCommandRunner { |
| String? workingDirectory; // set in runCommand() above |
| |
| Future<FlutterCommandResult> runCommand({ |
| required bool force, |
| required bool continueFlow, |
| required bool testFlow, |
| required GitTagVersion gitTagVersion, |
| required FlutterVersion flutterVersion, |
| required bool verifyOnly, |
| }) async { |
| if (!continueFlow) { |
| await runCommandFirstHalf( |
| force: force, |
| gitTagVersion: gitTagVersion, |
| flutterVersion: flutterVersion, |
| testFlow: testFlow, |
| verifyOnly: verifyOnly, |
| ); |
| } else { |
| await runCommandSecondHalf(flutterVersion); |
| } |
| return FlutterCommandResult.success(); |
| } |
| |
| Future<void> runCommandFirstHalf({ |
| required bool force, |
| required GitTagVersion gitTagVersion, |
| required FlutterVersion flutterVersion, |
| required bool testFlow, |
| required bool verifyOnly, |
| }) async { |
| final FlutterVersion upstreamVersion = await fetchLatestVersion(localVersion: flutterVersion); |
| if (flutterVersion.frameworkRevision == upstreamVersion.frameworkRevision) { |
| globals.printStatus('Flutter is already up to date on channel ${flutterVersion.channel}'); |
| globals.printStatus('$flutterVersion'); |
| return; |
| } else if (verifyOnly) { |
| globals.printStatus('A new version of Flutter is available on channel ${flutterVersion.channel}\n'); |
| globals.printStatus('The latest version: ${upstreamVersion.frameworkVersion} (revision ${upstreamVersion.frameworkRevisionShort})', emphasis: true); |
| globals.printStatus('Your current version: ${flutterVersion.frameworkVersion} (revision ${flutterVersion.frameworkRevisionShort})\n'); |
| globals.printStatus('To upgrade now, run "flutter upgrade".'); |
| if (flutterVersion.channel == 'stable') { |
| globals.printStatus('\nSee the announcement and release notes:'); |
| globals.printStatus('https://flutter.dev/docs/development/tools/sdk/release-notes'); |
| } |
| return; |
| } |
| if (!force && gitTagVersion == const GitTagVersion.unknown()) { |
| // If the commit is a recognized branch and not master, |
| // explain that we are avoiding potential damage. |
| if (flutterVersion.channel != 'master' && kOfficialChannels.contains(flutterVersion.channel)) { |
| throwToolExit( |
| 'Unknown flutter tag. Abandoning upgrade to avoid destroying local ' |
| 'changes. It is recommended to use git directly if not working on ' |
| 'an official channel.' |
| ); |
| // Otherwise explain that local changes can be lost. |
| } else { |
| throwToolExit( |
| 'Unknown flutter tag. Abandoning upgrade to avoid destroying local ' |
| 'changes. If it is okay to remove local changes, then re-run this ' |
| 'command with "--force".' |
| ); |
| } |
| } |
| // If there are uncommitted changes we might be on the right commit but |
| // we should still warn. |
| if (!force && await hasUncommittedChanges()) { |
| throwToolExit( |
| 'Your flutter checkout has local changes that would be erased by ' |
| 'upgrading. If you want to keep these changes, it is recommended that ' |
| 'you stash them via "git stash" or else commit the changes to a local ' |
| 'branch. If it is okay to remove local changes, then re-run this ' |
| 'command with "--force".' |
| ); |
| } |
| recordState(flutterVersion); |
| await ChannelCommand.upgradeChannel(flutterVersion); |
| globals.printStatus('Upgrading Flutter to ${upstreamVersion.frameworkVersion} from ${flutterVersion.frameworkVersion} in $workingDirectory...'); |
| await attemptReset(upstreamVersion.frameworkRevision); |
| if (!testFlow) { |
| await flutterUpgradeContinue(); |
| } |
| } |
| |
| void recordState(FlutterVersion flutterVersion) { |
| final Channel? channel = getChannelForName(flutterVersion.channel); |
| if (channel == null) { |
| return; |
| } |
| globals.persistentToolState!.updateLastActiveVersion(flutterVersion.frameworkRevision, channel); |
| } |
| |
| Future<void> flutterUpgradeContinue() async { |
| final int code = await globals.processUtils.stream( |
| <String>[ |
| globals.fs.path.join('bin', 'flutter'), |
| 'upgrade', |
| '--continue', |
| '--no-version-check', |
| ], |
| workingDirectory: workingDirectory, |
| allowReentrantFlutter: true, |
| environment: Map<String, String>.of(globals.platform.environment), |
| ); |
| if (code != 0) { |
| throwToolExit(null, exitCode: code); |
| } |
| } |
| |
| // This method should only be called if the upgrade command is invoked |
| // re-entrantly with the `--continue` flag |
| Future<void> runCommandSecondHalf(FlutterVersion flutterVersion) async { |
| // Make sure the welcome message re-display is delayed until the end. |
| final PersistentToolState persistentToolState = globals.persistentToolState!; |
| persistentToolState.setShouldRedisplayWelcomeMessage(false); |
| await precacheArtifacts(workingDirectory); |
| await updatePackages(flutterVersion); |
| await runDoctor(); |
| // Force the welcome message to re-display following the upgrade. |
| persistentToolState.setShouldRedisplayWelcomeMessage(true); |
| if (globals.flutterVersion.channel == 'master' || globals.flutterVersion.channel == 'main') { |
| globals.printStatus( |
| '\n' |
| 'This channel is intended for Flutter contributors. ' |
| 'This channel is not as thoroughly tested as the "beta" and "stable" channels. ' |
| 'We do not recommend using this channel for normal use as it more likely to contain serious regressions.\n' |
| '\n' |
| 'For information on contributing to Flutter, see our contributing guide:\n' |
| ' https://github.com/flutter/flutter/blob/master/CONTRIBUTING.md\n' |
| '\n' |
| 'For the most up to date stable version of flutter, consider using the "beta" channel instead. ' |
| 'The Flutter "beta" channel enjoys all the same automated testing as the "stable" channel, ' |
| 'but is updated roughly once a month instead of once a quarter.\n' |
| 'To change channel, run the "flutter channel beta" command.', |
| ); |
| } |
| } |
| |
| Future<bool> hasUncommittedChanges() async { |
| try { |
| final RunResult result = await globals.processUtils.run( |
| <String>['git', 'status', '-s'], |
| throwOnError: true, |
| workingDirectory: workingDirectory, |
| ); |
| return result.stdout.trim().isNotEmpty; |
| } on ProcessException catch (error) { |
| throwToolExit( |
| 'The tool could not verify the status of the current flutter checkout. ' |
| 'This might be due to git not being installed or an internal error. ' |
| 'If it is okay to ignore potential local changes, then re-run this ' |
| 'command with "--force".\n' |
| 'Error: $error.' |
| ); |
| } |
| } |
| |
| /// Returns the remote HEAD flutter version. |
| /// |
| /// Exits tool if HEAD isn't pointing to a branch, or there is no upstream. |
| Future<FlutterVersion> fetchLatestVersion({ |
| required FlutterVersion localVersion, |
| }) async { |
| String revision; |
| try { |
| // Fetch upstream branch's commits and tags |
| await globals.processUtils.run( |
| <String>['git', 'fetch', '--tags'], |
| throwOnError: true, |
| workingDirectory: workingDirectory, |
| ); |
| // Get the latest commit revision of the upstream |
| final RunResult result = await globals.processUtils.run( |
| <String>['git', 'rev-parse', '--verify', kGitTrackingUpstream], |
| throwOnError: true, |
| workingDirectory: workingDirectory, |
| ); |
| revision = result.stdout.trim(); |
| } on Exception catch (e) { |
| final String errorString = e.toString(); |
| if (errorString.contains('fatal: HEAD does not point to a branch')) { |
| throwToolExit( |
| 'Unable to upgrade Flutter: Your Flutter checkout is currently not ' |
| 'on a release branch.\n' |
| 'Use "flutter channel" to switch to an official channel, and retry. ' |
| 'Alternatively, re-install Flutter by going to $_flutterInstallDocs.' |
| ); |
| } else if (errorString.contains('fatal: no upstream configured for branch')) { |
| throwToolExit( |
| 'Unable to upgrade Flutter: The current Flutter branch/channel is ' |
| 'not tracking any remote repository.\n' |
| 'Re-install Flutter by going to $_flutterInstallDocs.' |
| ); |
| } else { |
| throwToolExit(errorString); |
| } |
| } |
| // At this point the current checkout should be on HEAD of a branch having |
| // an upstream. Check whether this upstream is "standard". |
| final VersionCheckError? error = VersionUpstreamValidator(version: localVersion, platform: globals.platform).run(); |
| if (error != null) { |
| throwToolExit( |
| 'Unable to upgrade Flutter: ' |
| '${error.message}\n' |
| 'Reinstalling Flutter may fix this issue. Visit $_flutterInstallDocs ' |
| 'for instructions.' |
| ); |
| } |
| return FlutterVersion.fromRevision( |
| flutterRoot: workingDirectory!, |
| frameworkRevision: revision, |
| fs: globals.fs, |
| ); |
| } |
| |
| /// Attempts a hard reset to the given revision. |
| /// |
| /// This is a reset instead of fast forward because if we are on a release |
| /// branch with cherry picks, there may not be a direct fast-forward route |
| /// to the next release. |
| Future<void> attemptReset(String newRevision) async { |
| try { |
| await globals.processUtils.run( |
| <String>['git', 'reset', '--hard', newRevision], |
| throwOnError: true, |
| workingDirectory: workingDirectory, |
| ); |
| } on ProcessException catch (e) { |
| throwToolExit(e.message, exitCode: e.errorCode); |
| } |
| } |
| |
| /// Update the user's packages. |
| Future<void> updatePackages(FlutterVersion flutterVersion) async { |
| globals.printStatus(''); |
| globals.printStatus(flutterVersion.toString()); |
| final String? projectRoot = findProjectRoot(globals.fs); |
| if (projectRoot != null) { |
| globals.printStatus(''); |
| await pub.get( |
| context: PubContext.pubUpgrade, |
| project: FlutterProject.fromDirectory(globals.fs.directory(projectRoot)), |
| upgrade: true, |
| ); |
| } |
| } |
| |
| /// Run flutter doctor in case requirements have changed. |
| Future<void> runDoctor() async { |
| globals.printStatus(''); |
| globals.printStatus('Running flutter doctor...'); |
| await globals.processUtils.stream( |
| <String>[ |
| globals.fs.path.join('bin', 'flutter'), '--no-version-check', 'doctor', |
| ], |
| workingDirectory: workingDirectory, |
| allowReentrantFlutter: true, |
| ); |
| } |
| } |
| |
| /// Update the engine repository and precache all artifacts. |
| /// |
| /// Check for and download any engine and pkg/ updates. We run the 'flutter' |
| /// shell script reentrantly here so that it will download the updated |
| /// Dart and so forth if necessary. |
| Future<void> precacheArtifacts([String? workingDirectory]) async { |
| globals.printStatus(''); |
| globals.printStatus('Upgrading engine...'); |
| final int code = await globals.processUtils.stream( |
| <String>[ |
| globals.fs.path.join('bin', 'flutter'), '--no-color', '--no-version-check', 'precache', |
| ], |
| allowReentrantFlutter: true, |
| environment: Map<String, String>.of(globals.platform.environment), |
| workingDirectory: workingDirectory, |
| ); |
| if (code != 0) { |
| throwToolExit(null, exitCode: code); |
| } |
| } |