Commit 25f332d8 authored by Devon Carew's avatar Devon Carew

re-work the doctor command

parent 1db09ffe
...@@ -102,20 +102,16 @@ class AndroidSdk { ...@@ -102,20 +102,16 @@ class AndroidSdk {
String get adbPath => getPlatformToolsPath('adb'); String get adbPath => getPlatformToolsPath('adb');
bool validateSdkWellFormed({ bool complain: false }) { /// Validate the Android SDK. This returns an empty list if there are no
if (!FileSystemEntity.isFileSync(adbPath)) { /// issues; otherwise, it returns a list of issues found.
if (complain) List<String> validateSdkWellFormed() {
printError('Android SDK file not found: $adbPath.'); if (!FileSystemEntity.isFileSync(adbPath))
return false; return <String>['Android SDK file not found: $adbPath.'];
}
if (sdkVersions.isEmpty) { if (sdkVersions.isEmpty || latestVersion == null)
if (complain) return <String>['Android SDK does not have the proper build-tools.'];
printError('Android SDK does not have the proper build-tools.');
return false;
}
return latestVersion.validateSdkWellFormed(complain: complain); return latestVersion.validateSdkWellFormed();
} }
String getPlatformToolsPath(String binaryName) { String getPlatformToolsPath(String binaryName) {
...@@ -215,12 +211,20 @@ class AndroidSdkVersion implements Comparable<AndroidSdkVersion> { ...@@ -215,12 +211,20 @@ class AndroidSdkVersion implements Comparable<AndroidSdkVersion> {
String get zipalignPath => getBuildToolsPath('zipalign'); String get zipalignPath => getBuildToolsPath('zipalign');
bool validateSdkWellFormed({ bool complain: false }) { List<String> validateSdkWellFormed() {
return if (_exists(androidJarPath) != null)
_exists(androidJarPath, complain: complain) && return <String>[_exists(androidJarPath)];
_exists(aaptPath, complain: complain) &&
_exists(dxPath, complain: complain) && if (_exists(aaptPath) != null)
_exists(zipalignPath, complain: complain); return <String>[_exists(aaptPath)];
if (_exists(dxPath) != null)
return <String>[_exists(dxPath)];
if (_exists(zipalignPath) != null)
return <String>[_exists(zipalignPath)];
return <String>[];
} }
String getPlatformsPath(String itemName) { String getPlatformsPath(String itemName) {
...@@ -237,13 +241,9 @@ class AndroidSdkVersion implements Comparable<AndroidSdkVersion> { ...@@ -237,13 +241,9 @@ class AndroidSdkVersion implements Comparable<AndroidSdkVersion> {
@override @override
String toString() => '[${sdk.directory}, SDK version $sdkLevel, build-tools $buildToolsVersionName]'; String toString() => '[${sdk.directory}, SDK version $sdkLevel, build-tools $buildToolsVersionName]';
bool _exists(String path, { bool complain: false }) { String _exists(String path) {
if (!FileSystemEntity.isFileSync(path)) { if (!FileSystemEntity.isFileSync(path))
if (complain) return 'Android SDK file not found: $path.';
printError('Android SDK file not found: $path.'); return null;
return false;
}
return true;
} }
} }
...@@ -6,9 +6,8 @@ import '../doctor.dart'; ...@@ -6,9 +6,8 @@ import '../doctor.dart';
import '../globals.dart'; import '../globals.dart';
import 'android_sdk.dart'; import 'android_sdk.dart';
class AndroidWorkflow extends Workflow { class AndroidWorkflow extends DoctorValidator implements Workflow {
@override AndroidWorkflow() : super('Android toolchain - develop for Android devices');
String get label => 'Android toolchain';
@override @override
bool get appliesToHostPlatform => true; bool get appliesToHostPlatform => true;
...@@ -17,30 +16,43 @@ class AndroidWorkflow extends Workflow { ...@@ -17,30 +16,43 @@ class AndroidWorkflow extends Workflow {
bool get canListDevices => getAdbPath(androidSdk) != null; bool get canListDevices => getAdbPath(androidSdk) != null;
@override @override
bool get canLaunchDevices => androidSdk != null && androidSdk.validateSdkWellFormed(complain: false); bool get canLaunchDevices => androidSdk != null && androidSdk.validateSdkWellFormed().isEmpty;
@override @override
ValidationResult validate() { ValidationResult validate() {
Validator androidValidator = new Validator( List<ValidationMessage> messages = <ValidationMessage>[];
label, ValidationType type = ValidationType.missing;
description: 'develop for Android devices' String sdkVersionText;
);
if (androidSdk == null) {
ValidationType sdkExists() { messages.add(new ValidationMessage.error(
return androidSdk == null ? ValidationType.missing : ValidationType.installed; 'Android Studio / Android SDK not found. Download from https://developer.android.com/sdk/\n'
}; '(or visit https://flutter.io/setup/#android-setup for detailed instructions).'
));
androidValidator.addValidator(new Validator( } else {
'Android Studio / Android SDK', type = ValidationType.partial;
description: 'enable development for Android devices',
resolution: 'Download from https://developer.android.com/sdk/ (or visit ' messages.add(new ValidationMessage('Android SDK at ${androidSdk.directory}'));
'https://flutter.io/setup/#android-setup for detailed instructions)',
validatorFunction: sdkExists if (androidSdk.latestVersion != null) {
)); sdkVersionText = 'Android SDK ${androidSdk.latestVersion.buildToolsVersionName}';
return androidValidator.validate(); messages.add(new ValidationMessage('Platform ${androidSdk.latestVersion.platformVersionName}'));
messages.add(new ValidationMessage('Build-tools ${androidSdk.latestVersion.buildToolsVersionName}'));
}
List<String> validationResult = androidSdk.validateSdkWellFormed();
if (validationResult.isEmpty) {
type = ValidationType.installed;
} else {
messages.addAll(validationResult.map((String message) {
return new ValidationMessage.error(message);
}));
messages.add(new ValidationMessage('Try re-installing or updating your Android SDK.'));
}
}
return new ValidationResult(type, messages, statusInfo: sdkVersionText);
} }
@override
void diagnose() => validate().print();
} }
...@@ -178,7 +178,9 @@ class BuildApkCommand extends FlutterCommand { ...@@ -178,7 +178,9 @@ class BuildApkCommand extends FlutterCommand {
return 1; return 1;
} }
if (!androidSdk.validateSdkWellFormed(complain: true)) { List<String> validationResult = androidSdk.validateSdkWellFormed();
if (validationResult.isNotEmpty) {
validationResult.forEach(printError);
printError('Try re-installing or updating your Android SDK.'); printError('Try re-installing or updating your Android SDK.');
return 1; return 1;
} }
...@@ -375,7 +377,9 @@ Future<int> buildAndroid({ ...@@ -375,7 +377,9 @@ Future<int> buildAndroid({
return 1; return 1;
} }
if (!androidSdk.validateSdkWellFormed(complain: true)) { List<String> validationResult = androidSdk.validateSdkWellFormed();
if (validationResult.isNotEmpty) {
validationResult.forEach(printError);
printError('Try re-installing or updating your Android SDK.'); printError('Try re-installing or updating your Android SDK.');
return 1; return 1;
} }
......
...@@ -4,10 +4,8 @@ ...@@ -4,10 +4,8 @@
import 'dart:async'; import 'dart:async';
import '../artifacts.dart';
import '../globals.dart'; import '../globals.dart';
import '../runner/flutter_command.dart'; import '../runner/flutter_command.dart';
import '../runner/version.dart';
class DoctorCommand extends FlutterCommand { class DoctorCommand extends FlutterCommand {
@override @override
...@@ -21,18 +19,8 @@ class DoctorCommand extends FlutterCommand { ...@@ -21,18 +19,8 @@ class DoctorCommand extends FlutterCommand {
@override @override
Future<int> runInProject() async { Future<int> runInProject() async {
// general info
String flutterRoot = ArtifactStore.flutterRoot;
printStatus('Flutter root: $flutterRoot.');
printStatus('');
// doctor // doctor
doctor.diagnose(); doctor.diagnose();
printStatus('');
// version
printStatus(getVersion(flutterRoot));
return 0; return 0;
} }
} }
...@@ -22,7 +22,7 @@ class UpgradeCommand extends FlutterCommand { ...@@ -22,7 +22,7 @@ class UpgradeCommand extends FlutterCommand {
@override @override
Future<int> runInProject() async { Future<int> runInProject() async {
printStatus(getVersion(ArtifactStore.flutterRoot)); printStatus(FlutterVersion.getVersion(ArtifactStore.flutterRoot).toString());
try { try {
runCheckedSync(<String>[ runCheckedSync(<String>[
...@@ -50,7 +50,7 @@ class UpgradeCommand extends FlutterCommand { ...@@ -50,7 +50,7 @@ class UpgradeCommand extends FlutterCommand {
return code; return code;
printStatus(''); printStatus('');
printStatus(getVersion(ArtifactStore.flutterRoot)); printStatus(FlutterVersion.getVersion(ArtifactStore.flutterRoot).toString());
return 0; return 0;
} }
......
This diff is collapsed.
...@@ -79,7 +79,7 @@ class IOSDevice extends Device { ...@@ -79,7 +79,7 @@ class IOSDevice extends Device {
bool get supportsStartPaused => false; bool get supportsStartPaused => false;
static List<IOSDevice> getAttachedDevices([IOSDevice mockIOS]) { static List<IOSDevice> getAttachedDevices([IOSDevice mockIOS]) {
if (!doctor.iosWorkflow.hasIdeviceId) if (!doctor.iosWorkflow.hasIDeviceId)
return <IOSDevice>[]; return <IOSDevice>[];
List<IOSDevice> devices = []; List<IOSDevice> devices = [];
......
...@@ -8,112 +8,96 @@ import '../base/process.dart'; ...@@ -8,112 +8,96 @@ import '../base/process.dart';
import '../doctor.dart'; import '../doctor.dart';
import 'mac.dart'; import 'mac.dart';
class IOSWorkflow extends Workflow { XCode get xcode => XCode.instance;
@override
String get label => 'iOS toolchain'; class IOSWorkflow extends DoctorValidator implements Workflow {
IOSWorkflow() : super('iOS toolchain - develop for iOS devices');
@override @override
bool get appliesToHostPlatform => Platform.isMacOS; bool get appliesToHostPlatform => Platform.isMacOS;
// We need xcode (+simctl) to list simulator devices, and idevice_id to list real devices. // We need xcode (+simctl) to list simulator devices, and idevice_id to list real devices.
@override @override
bool get canListDevices => XCode.instance.isInstalledAndMeetsVersionCheck; bool get canListDevices => xcode.isInstalledAndMeetsVersionCheck;
// We need xcode to launch simulator devices, and ideviceinstaller and ios-deploy // We need xcode to launch simulator devices, and ideviceinstaller and ios-deploy
// for real devices. // for real devices.
@override @override
bool get canLaunchDevices => XCode.instance.isInstalledAndMeetsVersionCheck; bool get canLaunchDevices => xcode.isInstalledAndMeetsVersionCheck;
bool get hasIDeviceId => exitsHappy(<String>['idevice_id', '-h']);
@override @override
ValidationResult validate() { ValidationResult validate() {
Validator iosValidator = new Validator( List<ValidationMessage> messages = <ValidationMessage>[];
label, int installCount = 0;
description: 'develop for iOS devices' String xcodeVersionInfo;
);
if (xcode.isInstalled) {
ValidationType xcodeExists() { installCount++;
return XCode.instance.isInstalled ? ValidationType.installed : ValidationType.missing;
}; xcodeVersionInfo = xcode.xcodeVersionText;
if (xcodeVersionInfo.contains(','))
ValidationType xcodeVersionSatisfactory() { xcodeVersionInfo = xcodeVersionInfo.substring(0, xcodeVersionInfo.indexOf(','));
return XCode.instance.isInstalledAndMeetsVersionCheck ? ValidationType.installed : ValidationType.missing;
}; messages.add(new ValidationMessage(xcode.xcodeVersionText));
ValidationType xcodeEulaSigned() { if (!xcode.isInstalledAndMeetsVersionCheck) {
return XCode.instance.eulaSigned ? ValidationType.installed : ValidationType.missing; messages.add(new ValidationMessage.error(
}; 'Flutter requires a minimum XCode version of $kXcodeRequiredVersionMajor.$kXcodeRequiredVersionMinor.0.\n'
'Download the latest version or update via the Mac App Store.'
ValidationType brewExists() { ));
return exitsHappy(<String>['brew', '-v']) }
? ValidationType.installed : ValidationType.missing;
}; if (!xcode.eulaSigned) {
messages.add(new ValidationMessage.error(
ValidationType ideviceinstallerExists() { 'XCode end user license agreement not signed; open XCode or run the command \'sudo xcodebuild -license\'.'
return exitsHappy(<String>['ideviceinstaller', '-h']) ));
? ValidationType.installed : ValidationType.missing; }
}; } else {
messages.add(new ValidationMessage.error(
ValidationType iosdeployExists() { 'XCode not installed; this is necessary for iOS development.\n'
return hasIdeviceId ? ValidationType.installed : ValidationType.missing; 'Download at https://developer.apple.com/xcode/download/.'
}; ));
}
Validator xcodeValidator = new Validator(
'XCode', // brew installed
description: 'enable development for iOS devices', if (exitsHappy(<String>['brew', '-v'])) {
resolution: 'Download at https://developer.apple.com/xcode/download/', installCount++;
validatorFunction: xcodeExists
); List<String> installed = <String>[];
iosValidator.addValidator(xcodeValidator); if (!exitsHappy(<String>['ideviceinstaller', '-h'])) {
messages.add(new ValidationMessage.error(
xcodeValidator.addValidator(new Validator( 'ideviceinstaller not available; this is used to discover connected iOS devices.\n'
'version', 'Install via \'brew install ideviceinstaller\'.'
description: 'Xcode minimum version of $kXcodeRequiredVersionMajor.$kXcodeRequiredVersionMinor.0', ));
resolution: 'Download the latest version or update via the Mac App Store', } else {
validatorFunction: xcodeVersionSatisfactory installed.add('ideviceinstaller');
)); }
xcodeValidator.addValidator(new Validator( if (!hasIDeviceId) {
'EULA', messages.add(new ValidationMessage.error(
description: 'XCode end user license agreement', 'ios-deploy not available; this is used to deploy to connected iOS devices.\n'
resolution: "Open XCode or run the command 'sudo xcodebuild -license'", 'Install via \'brew install ios-deploy\'.'
validatorFunction: xcodeEulaSigned ));
)); } else {
installed.add('ios-deploy');
Validator brewValidator = new Validator( }
'brew',
description: 'install additional development packages', if (installed.isNotEmpty)
resolution: 'Download at http://brew.sh/', messages.add(new ValidationMessage(installed.join(', ') + ' installed'));
validatorFunction: brewExists } else {
messages.add(new ValidationMessage.error(
'Brew not installed; use this to install tools for iOS device development.\n'
'Download brew at http://brew.sh/.'
));
}
return new ValidationResult(
installCount == 2 ? ValidationType.installed : installCount == 1 ? ValidationType.partial : ValidationType.missing,
messages,
statusInfo: xcodeVersionInfo
); );
iosValidator.addValidator(brewValidator);
brewValidator.addValidator(new Validator(
'ideviceinstaller',
description: 'discover connected iOS devices',
resolution: "Install via 'brew install ideviceinstaller'",
validatorFunction: ideviceinstallerExists
));
brewValidator.addValidator(new Validator(
'ios-deploy',
description: 'deploy to connected iOS devices',
resolution: "Install via 'brew install ios-deploy'",
validatorFunction: iosdeployExists
));
return iosValidator.validate();
}
@override
void diagnose() => validate().print();
bool get hasIdeviceId => exitsHappy(<String>['idevice_id', '-h']);
/// Return whether the tooling to list and deploy to real iOS devices (not the
/// simulator) is installed on the user's machine.
bool get canWorkWithIOSDevices {
return exitsHappy(<String>['ideviceinstaller', '-h']) && hasIdeviceId;
} }
} }
...@@ -54,27 +54,24 @@ class XCode { ...@@ -54,27 +54,24 @@ class XCode {
} }
} }
bool _xcodeVersionSatisfactory; String _xcodeVersionText;
bool get xcodeVersionSatisfactory {
if (_xcodeVersionSatisfactory != null)
return _xcodeVersionSatisfactory;
try { String get xcodeVersionText {
String output = runSync(<String>['xcodebuild', '-version']); if (_xcodeVersionText == null)
RegExp regex = new RegExp(r'Xcode ([0-9.]+)'); _xcodeVersionText = runSync(<String>['xcodebuild', '-version']).replaceAll('\n', ', ');
return _xcodeVersionText;
}
String version = regex.firstMatch(output).group(1); bool get xcodeVersionSatisfactory {
List<String> components = version.split('.'); RegExp regex = new RegExp(r'Xcode ([0-9.]+)');
int major = int.parse(components[0]); String version = regex.firstMatch(xcodeVersionText).group(1);
int minor = components.length == 1 ? 0 : int.parse(components[1]); List<String> components = version.split('.');
_xcodeVersionSatisfactory = _xcodeVersionCheckValid(major, minor); int major = int.parse(components[0]);
} catch (error) { int minor = components.length == 1 ? 0 : int.parse(components[1]);
_xcodeVersionSatisfactory = false;
}
return _xcodeVersionSatisfactory; return _xcodeVersionCheckValid(major, minor);
} }
} }
......
...@@ -200,16 +200,8 @@ class FlutterCommandRunner extends CommandRunner { ...@@ -200,16 +200,8 @@ class FlutterCommandRunner extends CommandRunner {
} }
} }
if (androidSdk != null) {
printTrace('Using Android SDK at ${androidSdk.directory}.');
if (androidSdk.latestVersion != null)
printTrace('${androidSdk.latestVersion}');
}
if (globalResults['version']) { if (globalResults['version']) {
printStatus(getVersion(ArtifactStore.flutterRoot)); printStatus(FlutterVersion.getVersion(ArtifactStore.flutterRoot).toString());
printStatus('');
doctor.summary();
return new Future<int>.value(0); return new Future<int>.value(0);
} }
......
...@@ -5,26 +5,54 @@ ...@@ -5,26 +5,54 @@
import '../artifacts.dart'; import '../artifacts.dart';
import '../base/process.dart'; import '../base/process.dart';
String getVersion(String flutterRoot) { class FlutterVersion {
String upstream = runSync([ FlutterVersion(this.flutterRoot) {
'git', 'rev-parse', '--abbrev-ref', '--symbolic', '@{u}' _channel = _runGit('git rev-parse --abbrev-ref --symbolic @{u}');
], workingDirectory: flutterRoot).trim();
String repository; int slash = _channel.indexOf('/');
int slash = upstream.indexOf('/'); if (slash != -1) {
if (slash != -1) { String remote = _channel.substring(0, slash);
String remote = upstream.substring(0, slash); _repositoryUrl = _runGit('git ls-remote --get-url $remote');
repository = runSync([ _channel = _channel.substring(slash + 1);
'git', 'ls-remote', '--get-url', remote } else if (_channel.isEmpty) {
], workingDirectory: flutterRoot).trim(); _channel = 'unknown';
upstream = upstream.substring(slash + 1); }
_frameworkRevision = _runGit('git log -n 1 --pretty=format:%H');
_frameworkAge = _runGit('git log -n 1 --pretty=format:%ar');
} }
String revision = runSync([
'git', 'log', '-n', '1', '--pretty=format:%H (%ar)'
], workingDirectory: flutterRoot).trim();
String from = repository == null ? 'Flutter from unknown source' : 'Flutter from $repository (on $upstream)'; final String flutterRoot;
String flutterVersion = 'Framework: $revision';
String engineRevision = 'Engine: ${ArtifactStore.engineRevision}'; String _repositoryUrl;
String get repositoryUrl => _repositoryUrl;
String _channel;
/// `master`, `alpha`, `hackathon`, ...
String get channel => _channel;
String _frameworkRevision;
String get frameworkRevision => _frameworkRevision;
String get frameworkRevisionShort => _runGit('git rev-parse --short $frameworkRevision');
String _frameworkAge;
String get frameworkAge => _frameworkAge;
return '$from\n$flutterVersion\n$engineRevision'; String get engineRevision => ArtifactStore.engineRevision;
String get engineRevisionShort => _runGit('git rev-parse --short $engineRevision');
String _runGit(String command) => runSync(command.split(' '), workingDirectory: flutterRoot);
@override
String toString() {
String from = repositoryUrl == null ? 'Flutter from unknown source' : 'Flutter from $repositoryUrl (on channel $channel)';
String flutterText = 'Framework: $frameworkRevisionShort ($frameworkAge)';
String engineText = 'Engine: $engineRevisionShort';
return '$from\n\n$flutterText\n$engineText';
}
static FlutterVersion getVersion([String flutterRoot]) {
return new FlutterVersion(flutterRoot != null ? flutterRoot : ArtifactStore.flutterRoot);
}
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment