blob: 45636afe29c60e25c466f3e164378daa18f0c2a5 [file] [log] [blame]
// 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 '../base/time.dart';
import '../base/utils.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 _flutterInstallDocs = 'https://flutter.dev/setup';
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,
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(
'continue-started-at',
hide: !verboseHelp,
help:
'If "--continue" is provided, an ISO 8601 timestamp of the time that the '
'initial upgrade command was started. This should not be invoked manually.',
)
..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 name = 'upgrade';
@override
final description = 'Upgrade your copy of Flutter.';
@override
final String category = FlutterCommandCategory.sdk;
@override
bool get shouldUpdateCache => false;
UpgradePhase _parsePhaseFromContinueArg() {
if (!boolArg('continue')) {
return const UpgradePhase.firstHalf();
} else {
final DateTime? upgradeStartedAt;
if (stringArg('continue-started-at') case final String iso8601String) {
upgradeStartedAt = DateTime.parse(iso8601String);
} else {
upgradeStartedAt = null;
}
return UpgradePhase.secondHalf(upgradeStartedAt: upgradeStartedAt);
}
}
@override
Future<FlutterCommandResult> runCommand() {
_commandRunner.workingDirectory = stringArg('working-directory') ?? Cache.flutterRoot!;
return _commandRunner.runCommand(
_parsePhaseFromContinueArg(),
force: boolArg('force'),
testFlow: stringArg('working-directory') != null,
gitTagVersion: GitTagVersion.determine(
globals.platform,
git: globals.git,
workingDirectory: _commandRunner.workingDirectory,
),
flutterVersion: stringArg('working-directory') == null
? globals.flutterVersion
: FlutterVersion(
flutterRoot: _commandRunner.workingDirectory!,
fs: globals.fs,
git: globals.git,
),
verifyOnly: boolArg('verify-only'),
);
}
}
@immutable
sealed class UpgradePhase {
const factory UpgradePhase.firstHalf() = _FirstHalf;
const factory UpgradePhase.secondHalf({required DateTime? upgradeStartedAt}) = _SecondHalf;
}
final class _FirstHalf implements UpgradePhase {
const _FirstHalf();
}
final class _SecondHalf implements UpgradePhase {
const _SecondHalf({required this.upgradeStartedAt});
/// What time the original `flutter upgrade` command started at.
///
/// If omitted, the initiating client was too old to know to pass this value.
final DateTime? upgradeStartedAt;
}
@visibleForTesting
class UpgradeCommandRunner {
String? workingDirectory; // set in runCommand() above
@visibleForTesting
var clock = const SystemClock();
Future<FlutterCommandResult> runCommand(
UpgradePhase phase, {
required bool force,
required bool testFlow,
required GitTagVersion gitTagVersion,
required FlutterVersion flutterVersion,
required bool verifyOnly,
}) async {
switch (phase) {
case _FirstHalf():
await _runCommandFirstHalf(
startedAt: clock.now(),
force: force,
gitTagVersion: gitTagVersion,
flutterVersion: flutterVersion,
testFlow: testFlow,
verifyOnly: verifyOnly,
);
case _SecondHalf(:final DateTime? upgradeStartedAt):
await _runCommandSecondHalf(flutterVersion);
if (upgradeStartedAt != null) {
final Duration execution = clock.now().difference(upgradeStartedAt);
globals.printStatus('Took ${getElapsedAsMinutesOrSeconds(execution)}');
}
}
return FlutterCommandResult.success();
}
Future<void> _runCommandFirstHalf({
required DateTime startedAt,
required bool force,
required GitTagVersion gitTagVersion,
required FlutterVersion flutterVersion,
required bool testFlow,
required bool verifyOnly,
}) async {
final FlutterVersion upstreamVersion = await fetchLatestVersion(localVersion: flutterVersion);
// It's possible for a given framework revision to have multiple tags (i.e., due to a release
// rollback). Verify the upstream version tag isn't newer than the current tag.
if (flutterVersion.frameworkRevision == upstreamVersion.frameworkRevision &&
flutterVersion.gitTagVersion.gitTag.compareTo(upstreamVersion.gitTagVersion.gitTag) >= 0) {
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://docs.flutter.dev/release/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(startedAt: startedAt);
}
}
void recordState(FlutterVersion flutterVersion) {
final Channel? channel = getChannelForName(flutterVersion.channel);
if (channel == null) {
return;
}
globals.persistentToolState!.updateLastActiveVersion(flutterVersion.frameworkRevision, channel);
}
@visibleForTesting
Future<void> flutterUpgradeContinue({required DateTime startedAt}) async {
final int code = await globals.processUtils.stream(
[
globals.fs.path.join('bin', 'flutter'),
'upgrade',
'--continue',
'--continue-started-at',
startedAt.toIso8601String(),
'--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/main/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.',
);
}
}
@protected
Future<bool> hasUncommittedChanges() async {
try {
final RunResult result = await globals.git.run(
['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.
@visibleForTesting
Future<FlutterVersion> fetchLatestVersion({required FlutterVersion localVersion}) async {
String revision;
try {
// Fetch upstream branch's commits and tags
await globals.git.run(
['fetch', '--tags'],
throwOnError: true,
workingDirectory: workingDirectory,
);
// Get the latest commit revision of the upstream
final RunResult result = await globals.git.run(
['rev-parse', '--verify', kGitTrackingUpstream],
throwOnError: true,
workingDirectory: workingDirectory,
);
revision = result.stdout.trim();
} on Exception catch (e) {
final 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,
git: globals.git,
);
}
/// 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.
@visibleForTesting
Future<void> attemptReset(String newRevision) async {
try {
await globals.git.run(
['reset', '--hard', newRevision],
throwOnError: true,
workingDirectory: workingDirectory,
);
} on ProcessException catch (e) {
throwToolExit(e.message, exitCode: e.errorCode);
}
}
/// Update the user's packages.
@protected
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.
@protected
Future<void> runDoctor() async {
globals.printStatus('');
globals.printStatus('Running flutter doctor...');
await globals.processUtils.stream(
[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(
[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);
}
}