Unverified Commit 3215a013 authored by includecmath's avatar includecmath Committed by GitHub

[flutter_tools] Unified analyze error log format (#61146)

* Unified analyze error log format

* Fix review issues
parent 0a03d643
...@@ -17,6 +17,7 @@ import '../base/platform.dart'; ...@@ -17,6 +17,7 @@ import '../base/platform.dart';
import '../base/terminal.dart'; import '../base/terminal.dart';
import '../base/utils.dart'; import '../base/utils.dart';
import '../cache.dart'; import '../cache.dart';
import '../dart/analysis.dart';
import '../globals.dart' as globals; import '../globals.dart' as globals;
/// Common behavior for `flutter analyze` and `flutter analyze --watch` /// Common behavior for `flutter analyze` and `flutter analyze --watch`
...@@ -84,7 +85,16 @@ abstract class AnalyzeBase { ...@@ -84,7 +85,16 @@ abstract class AnalyzeBase {
logger.printStatus('Analysis benchmark written to $benchmarkOut ($data).'); logger.printStatus('Analysis benchmark written to $benchmarkOut ($data).');
} }
bool get isFlutterRepo => argResults['flutter-repo'] as bool;
String get sdkPath => argResults['dart-sdk'] as String ?? artifacts.getArtifactPath(Artifact.engineDartSdkPath);
bool get isBenchmarking => argResults['benchmark'] as bool; bool get isBenchmarking => argResults['benchmark'] as bool;
bool get isDartDocs => argResults['dartdocs'] as bool;
static int countMissingDartDocs(List<AnalysisError> errors) {
return errors.where((AnalysisError error) {
return error.code == 'public_member_api_docs';
}).length;
}
static String generateDartDocMessage(int undocumentedMembers) { static String generateDartDocMessage(int undocumentedMembers) {
String dartDocMessage; String dartDocMessage;
...@@ -103,6 +113,41 @@ abstract class AnalyzeBase { ...@@ -103,6 +113,41 @@ abstract class AnalyzeBase {
return dartDocMessage; return dartDocMessage;
} }
/// Generate an analysis summary for both [AnalyzeOnce], [AnalyzeContinuously].
static String generateErrorsMessage({
@required int issueCount,
int issueDiff,
int files,
@required String seconds,
int undocumentedMembers = 0,
String dartDocMessage = '',
}) {
final StringBuffer errorsMessage = StringBuffer(issueCount > 0
? '$issueCount ${pluralize('issue', issueCount)} found.'
: 'No issues found!');
// Only [AnalyzeContinuously] has issueDiff message.
if (issueDiff != null) {
if (issueDiff > 0) {
errorsMessage.write(' ($issueDiff new)');
} else if (issueDiff < 0) {
errorsMessage.write(' (${-issueDiff} fixed)');
}
}
// Only [AnalyzeContinuously] has files message.
if (files != null) {
errorsMessage.write(' • analyzed $files ${pluralize('file', files)}');
}
if (undocumentedMembers > 0) {
errorsMessage.write(' (ran in ${seconds}s; $dartDocMessage)');
} else {
errorsMessage.write(' (ran in ${seconds}s;)');
}
return errorsMessage.toString();
}
} }
class PackageDependency { class PackageDependency {
......
...@@ -15,7 +15,6 @@ import '../base/io.dart'; ...@@ -15,7 +15,6 @@ import '../base/io.dart';
import '../base/logger.dart'; import '../base/logger.dart';
import '../base/platform.dart'; import '../base/platform.dart';
import '../base/terminal.dart'; import '../base/terminal.dart';
import '../base/utils.dart';
import '../dart/analysis.dart'; import '../dart/analysis.dart';
import 'analyze_base.dart'; import 'analyze_base.dart';
...@@ -56,7 +55,7 @@ class AnalyzeContinuously extends AnalyzeBase { ...@@ -56,7 +55,7 @@ class AnalyzeContinuously extends AnalyzeBase {
Future<void> analyze() async { Future<void> analyze() async {
List<String> directories; List<String> directories;
if (argResults['flutter-repo'] as bool) { if (isFlutterRepo) {
final PackageDependencyTracker dependencies = PackageDependencyTracker(); final PackageDependencyTracker dependencies = PackageDependencyTracker();
dependencies.checkForConflictingDependencies(repoPackages, dependencies); dependencies.checkForConflictingDependencies(repoPackages, dependencies);
...@@ -72,9 +71,6 @@ class AnalyzeContinuously extends AnalyzeBase { ...@@ -72,9 +71,6 @@ class AnalyzeContinuously extends AnalyzeBase {
analysisTarget = fileSystem.currentDirectory.path; analysisTarget = fileSystem.currentDirectory.path;
} }
final String sdkPath = argResults['dart-sdk'] as String ??
artifacts.getArtifactPath(Artifact.engineDartSdkPath);
final AnalysisServer server = AnalysisServer( final AnalysisServer server = AnalysisServer(
sdkPath, sdkPath,
directories, directories,
...@@ -131,10 +127,8 @@ class AnalyzeContinuously extends AnalyzeBase { ...@@ -131,10 +127,8 @@ class AnalyzeContinuously extends AnalyzeBase {
int issueCount = errors.length; int issueCount = errors.length;
// count missing dartdocs // count missing dartdocs
final int undocumentedMembers = errors.where((AnalysisError error) { final int undocumentedMembers = AnalyzeBase.countMissingDartDocs(errors);
return error.code == 'public_member_api_docs'; if (!isDartDocs) {
}).length;
if (!(argResults['dartdocs'] as bool)) {
errors.removeWhere((AnalysisError error) => error.code == 'public_member_api_docs'); errors.removeWhere((AnalysisError error) => error.code == 'public_member_api_docs');
issueCount -= undocumentedMembers; issueCount -= undocumentedMembers;
} }
...@@ -150,32 +144,20 @@ class AnalyzeContinuously extends AnalyzeBase { ...@@ -150,32 +144,20 @@ class AnalyzeContinuously extends AnalyzeBase {
dumpErrors(errors.map<String>((AnalysisError error) => error.toLegacyString())); dumpErrors(errors.map<String>((AnalysisError error) => error.toLegacyString()));
// Print an analysis summary.
String errorsMessage;
final int issueDiff = issueCount - lastErrorCount; final int issueDiff = issueCount - lastErrorCount;
lastErrorCount = issueCount; lastErrorCount = issueCount;
final String seconds = (analysisTimer.elapsedMilliseconds / 1000.0).toStringAsFixed(2);
if (firstAnalysis) {
errorsMessage = '$issueCount ${pluralize('issue', issueCount)} found';
} else if (issueDiff > 0) {
errorsMessage = '$issueCount ${pluralize('issue', issueCount)} found ($issueDiff new)';
} else if (issueDiff < 0) {
errorsMessage = '$issueCount ${pluralize('issue', issueCount)} found (${-issueDiff} fixed)';
} else if (issueCount != 0) {
errorsMessage = '$issueCount ${pluralize('issue', issueCount)} found';
} else {
errorsMessage = 'No issues found!';
}
final String dartDocMessage = AnalyzeBase.generateDartDocMessage(undocumentedMembers); final String dartDocMessage = AnalyzeBase.generateDartDocMessage(undocumentedMembers);
final String errorsMessage = AnalyzeBase.generateErrorsMessage(
issueCount: issueCount,
issueDiff: issueDiff,
files: analyzedPaths.length,
seconds: seconds,
undocumentedMembers: undocumentedMembers,
dartDocMessage: dartDocMessage,
);
final String files = '${analyzedPaths.length} ${pluralize('file', analyzedPaths.length)}'; logger.printStatus(errorsMessage);
final String seconds = (analysisTimer.elapsedMilliseconds / 1000.0).toStringAsFixed(2);
if (undocumentedMembers > 0) {
logger.printStatus('$errorsMessage$dartDocMessage • analyzed $files in $seconds seconds');
} else {
logger.printStatus('$errorsMessage • analyzed $files in $seconds seconds');
}
if (firstAnalysis && isBenchmarking) { if (firstAnalysis && isBenchmarking) {
writeBenchmark(analysisTimer, issueCount, undocumentedMembers); writeBenchmark(analysisTimer, issueCount, undocumentedMembers);
......
...@@ -14,7 +14,6 @@ import '../base/file_system.dart'; ...@@ -14,7 +14,6 @@ import '../base/file_system.dart';
import '../base/logger.dart'; import '../base/logger.dart';
import '../base/platform.dart'; import '../base/platform.dart';
import '../base/terminal.dart'; import '../base/terminal.dart';
import '../base/utils.dart';
import '../dart/analysis.dart'; import '../dart/analysis.dart';
import 'analyze_base.dart'; import 'analyze_base.dart';
...@@ -67,7 +66,7 @@ class AnalyzeOnce extends AnalyzeBase { ...@@ -67,7 +66,7 @@ class AnalyzeOnce extends AnalyzeBase {
} }
} }
if (argResults['flutter-repo'] as bool) { if (isFlutterRepo) {
// check for conflicting dependencies // check for conflicting dependencies
final PackageDependencyTracker dependencies = PackageDependencyTracker(); final PackageDependencyTracker dependencies = PackageDependencyTracker();
dependencies.checkForConflictingDependencies(repoPackages, dependencies); dependencies.checkForConflictingDependencies(repoPackages, dependencies);
...@@ -88,9 +87,6 @@ class AnalyzeOnce extends AnalyzeBase { ...@@ -88,9 +87,6 @@ class AnalyzeOnce extends AnalyzeBase {
final Completer<void> analysisCompleter = Completer<void>(); final Completer<void> analysisCompleter = Completer<void>();
final List<AnalysisError> errors = <AnalysisError>[]; final List<AnalysisError> errors = <AnalysisError>[];
final String sdkPath = argResults['dart-sdk'] as String ??
artifacts.getArtifactPath(Artifact.engineDartSdkPath);
final AnalysisServer server = AnalysisServer( final AnalysisServer server = AnalysisServer(
sdkPath, sdkPath,
directories.toList(), directories.toList(),
...@@ -152,11 +148,8 @@ class AnalyzeOnce extends AnalyzeBase { ...@@ -152,11 +148,8 @@ class AnalyzeOnce extends AnalyzeBase {
timer?.stop(); timer?.stop();
} }
// count missing dartdocs final int undocumentedMembers = AnalyzeBase.countMissingDartDocs(errors);
final int undocumentedMembers = errors.where((AnalysisError error) { if (!isDartDocs) {
return error.code == 'public_member_api_docs';
}).length;
if (!(argResults['dartdocs'] as bool)) {
errors.removeWhere((AnalysisError error) => error.code == 'public_member_api_docs'); errors.removeWhere((AnalysisError error) => error.code == 'public_member_api_docs');
} }
...@@ -177,31 +170,28 @@ class AnalyzeOnce extends AnalyzeBase { ...@@ -177,31 +170,28 @@ class AnalyzeOnce extends AnalyzeBase {
logger.printStatus(error.toString(), hangingIndent: 7); logger.printStatus(error.toString(), hangingIndent: 7);
} }
final int errorCount = errors.length;
final String seconds = (timer.elapsedMilliseconds / 1000.0).toStringAsFixed(1); final String seconds = (timer.elapsedMilliseconds / 1000.0).toStringAsFixed(1);
final String dartDocMessage = AnalyzeBase.generateDartDocMessage(undocumentedMembers); final String dartDocMessage = AnalyzeBase.generateDartDocMessage(undocumentedMembers);
final String errorsMessage = AnalyzeBase.generateErrorsMessage(
issueCount: errorCount,
seconds: seconds,
undocumentedMembers: undocumentedMembers,
dartDocMessage: dartDocMessage,
);
// We consider any level of error to be an error exit (we don't report different levels). if (errorCount > 0) {
if (errors.isNotEmpty) {
final int errorCount = errors.length;
logger.printStatus(''); logger.printStatus('');
if (undocumentedMembers > 0) { // We consider any level of error to be an error exit (we don't report different levels).
throwToolExit('$errorCount ${pluralize('issue', errorCount)} found. (ran in ${seconds}s; $dartDocMessage)'); throwToolExit(errorsMessage);
} else {
throwToolExit('$errorCount ${pluralize('issue', errorCount)} found. (ran in ${seconds}s)');
}
} }
if (server.didServerErrorOccur) { if (argResults['congratulate'] as bool) {
throwToolExit('Server error(s) occurred. (ran in ${seconds}s)'); logger.printStatus(errorsMessage);
} }
if (argResults['congratulate'] as bool) { if (server.didServerErrorOccur) {
if (undocumentedMembers > 0) { throwToolExit('Server error(s) occurred. (ran in ${seconds}s)');
logger.printStatus('No issues found! (ran in ${seconds}s; $dartDocMessage)');
} else {
logger.printStatus('No issues found! (ran in ${seconds}s)');
}
} }
} }
} }
...@@ -19,6 +19,30 @@ void main() { ...@@ -19,6 +19,30 @@ void main() {
expect(AnalyzeBase.generateDartDocMessage(2), '2 public members lack documentation'); expect(AnalyzeBase.generateDartDocMessage(2), '2 public members lack documentation');
}); });
testWithoutContext('analyze generate correct errors message', () async {
expect(
AnalyzeBase.generateErrorsMessage(
issueCount: 0,
seconds: '0.1',
undocumentedMembers: 1,
dartDocMessage: 'one public member lacks documentation',
),
'No issues found! (ran in 0.1s; one public member lacks documentation)',
);
expect(
AnalyzeBase.generateErrorsMessage(
issueCount: 3,
issueDiff: 2,
files: 1,
seconds: '0.1',
undocumentedMembers: 1,
dartDocMessage: 'one public member lacks documentation',
),
'3 issues found. (2 new) • analyzed 1 file (ran in 0.1s; one public member lacks documentation)',
);
});
testWithoutContext('analyze inRepo', () { testWithoutContext('analyze inRepo', () {
final FileSystem fileSystem = MemoryFileSystem.test(); final FileSystem fileSystem = MemoryFileSystem.test();
fileSystem.directory(_kFlutterRoot).createSync(recursive: true); fileSystem.directory(_kFlutterRoot).createSync(recursive: true);
...@@ -54,10 +78,10 @@ void main() { ...@@ -54,10 +78,10 @@ void main() {
'offset': 362, 'offset': 362,
'length': 72, 'length': 72,
'startLine': 15, 'startLine': 15,
'startColumn': 4 'startColumn': 4,
}, },
'message': 'Prefer final for variable declarations if they are not reassigned.', 'message': 'Prefer final for variable declarations if they are not reassigned.',
'hasFix': false 'hasFix': false,
}; };
expect(WrittenError.fromJson(json).toString(), expect(WrittenError.fromJson(json).toString(),
'[info] Prefer final for variable declarations if they are not reassigned (/Users/.../lib/test.dart:15:4)'); '[info] Prefer final for variable declarations if they are not reassigned (/Users/.../lib/test.dart:15:4)');
......
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