blob: c5e1bc2c731a3d00756ca2e20085a64aea40685f [file] [log] [blame]
// Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'dart:async';
import 'dart:io';
import 'package:analyzer/dart/analysis/results.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/error/error.dart';
import 'package:analyzer/exception/exception.dart';
import 'package:analyzer/src/dart/analysis/driver.dart';
import 'package:analyzer/src/dart/analysis/file_state.dart';
import 'package:analyzer/src/generated/engine.dart';
import 'package:analyzer/src/generated/source.dart';
import 'package:analyzer/src/generated/source_io.dart';
import 'package:analyzer/src/generated/utilities_general.dart';
import 'package:analyzer_cli/src/driver.dart';
import 'package:analyzer_cli/src/error_formatter.dart';
import 'package:analyzer_cli/src/error_severity.dart';
import 'package:analyzer_cli/src/options.dart';
import 'package:path/path.dart' as path;
int get currentTimeMillis => new DateTime.now().millisecondsSinceEpoch;
/// Analyzes single library [File].
class AnalyzerImpl {
static final PerformanceTag _prepareErrorsTag =
new PerformanceTag("AnalyzerImpl.prepareErrors");
static final PerformanceTag _resolveLibraryTag =
new PerformanceTag("AnalyzerImpl._resolveLibrary");
final CommandLineOptions options;
final int startTime;
final AnalysisOptions analysisOptions;
final AnalysisDriver analysisDriver;
/// Accumulated analysis statistics.
final AnalysisStats stats;
/// The library file to analyze.
final FileState libraryFile;
/// All files references by the analyzed library.
final Set<String> files = new Set<String>();
/// All [AnalysisErrorInfo]s in the analyzed library.
final List<ErrorsResult> errorsResults = [];
/// If the file specified on the command line is part of a package, the name
/// of that package. Otherwise `null`. This allows us to analyze the file
/// specified on the command line as though it is reached via a "package:"
/// URI, but avoid suppressing its output in the event that the user has not
/// specified the "--package-warnings" option.
String _selfPackageName;
AnalyzerImpl(this.analysisOptions, this.analysisDriver, this.libraryFile,
this.options, this.stats, this.startTime);
void addCompilationUnitSource(
CompilationUnitElement unit, Set<CompilationUnitElement> units) {
if (unit == null || !units.add(unit)) {
return;
}
Source source = unit.source;
if (source != null) {
files.add(source.fullName);
}
}
void addLibrarySources(LibraryElement library, Set<LibraryElement> libraries,
Set<CompilationUnitElement> units) {
if (library == null || !libraries.add(library)) {
return;
}
// Maybe skip library.
if (!_isAnalyzedLibrary(library)) {
return;
}
// Add compilation units.
addCompilationUnitSource(library.definingCompilationUnit, units);
for (CompilationUnitElement child in library.parts) {
addCompilationUnitSource(child, units);
}
// Add referenced libraries.
for (LibraryElement child in library.importedLibraries) {
addLibrarySources(child, libraries, units);
}
for (LibraryElement child in library.exportedLibraries) {
addLibrarySources(child, libraries, units);
}
}
/// Treats the [sourcePath] as the top level library and analyzes it using
/// the analysis engine. If [printMode] is `0`, then no error or performance
/// information is printed. If [printMode] is `1`, then errors will be printed.
/// If [printMode] is `2`, then performance information will be printed, and
/// it will be marked as being for a cold VM.
Future<ErrorSeverity> analyze(ErrorFormatter formatter,
{int printMode: 1}) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
setupForAnalysis();
return await _analyze(printMode, formatter);
}
/// Returns the maximal [ErrorSeverity] of the recorded errors.
ErrorSeverity computeMaxErrorSeverity() {
ErrorSeverity status = ErrorSeverity.NONE;
for (ErrorsResult result in errorsResults) {
for (AnalysisError error in result.errors) {
if (_defaultSeverityProcessor(error) == null) {
continue;
}
status = status.max(computeSeverity(error, options, analysisOptions));
}
}
return status;
}
/// Fills [errorsResults] using [files].
Future<void> prepareErrors() async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
PerformanceTag previous = _prepareErrorsTag.makeCurrent();
try {
for (String path in files) {
ErrorsResult errorsResult = await analysisDriver.getErrors(path);
errorsResults.add(errorsResult);
}
} finally {
previous.makeCurrent();
}
}
/// Fills [files].
void prepareSources(LibraryElement library) {
var units = new Set<CompilationUnitElement>();
var libraries = new Set<LibraryElement>();
addLibrarySources(library, libraries, units);
}
/// Setup local fields such as the analysis context for analysis.
void setupForAnalysis() {
files.clear();
errorsResults.clear();
Uri libraryUri = libraryFile.uri;
if (libraryUri.scheme == 'package' && libraryUri.pathSegments.length > 0) {
_selfPackageName = libraryUri.pathSegments[0];
}
}
Future<ErrorSeverity> _analyze(
int printMode, ErrorFormatter formatter) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
// Don't try to analyze parts.
if (libraryFile.isPart) {
String libraryPath = libraryFile.path;
stderr.writeln("Only libraries can be analyzed.");
stderr.writeln("$libraryPath is a part and can not be analyzed.");
return ErrorSeverity.ERROR;
}
LibraryElement libraryElement = await _resolveLibrary();
prepareSources(libraryElement);
await prepareErrors();
// Print errors and performance numbers.
if (printMode == 1) {
formatter.formatErrors(errorsResults);
} else if (printMode == 2) {
_printColdPerf();
}
// Compute and return max severity.
return computeMaxErrorSeverity();
}
ErrorSeverity _defaultSeverityProcessor(AnalysisError error) =>
determineProcessedSeverity(error, options, analysisOptions);
/// Returns true if we want to report diagnostics for this library.
bool _isAnalyzedLibrary(LibraryElement library) {
Source source = library.source;
switch (source.uriKind) {
case UriKind.DART_URI:
return options.showSdkWarnings;
case UriKind.PACKAGE_URI:
if (_isPathInPubCache(source.fullName)) {
return false;
}
return _isAnalyzedPackage(source.uri);
default:
return true;
}
}
/// Determine whether the given URI refers to a package being analyzed.
bool _isAnalyzedPackage(Uri uri) {
if (uri.scheme != 'package' || uri.pathSegments.isEmpty) {
return false;
}
String packageName = uri.pathSegments.first;
if (packageName == _selfPackageName) {
return true;
} else if (!options.showPackageWarnings) {
return false;
} else if (options.showPackageWarningsPrefix == null) {
return true;
} else {
return packageName.startsWith(options.showPackageWarningsPrefix);
}
}
// TODO(devoncarew): This is never called.
void _printColdPerf() {
// Print cold VM performance numbers.
int totalTime = currentTimeMillis - startTime;
int otherTime = totalTime;
for (PerformanceTag tag in PerformanceTag.all) {
if (tag != PerformanceTag.unknown) {
int tagTime = tag.elapsedMs;
outSink.writeln('${tag.label}-cold:$tagTime');
otherTime -= tagTime;
}
}
outSink.writeln('other-cold:$otherTime');
outSink.writeln("total-cold:$totalTime");
}
Future<LibraryElement> _resolveLibrary() async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
PerformanceTag previous = _resolveLibraryTag.makeCurrent();
try {
String libraryPath = libraryFile.path;
analysisDriver.priorityFiles = [libraryPath];
UnitElementResult elementResult =
await analysisDriver.getUnitElement(libraryPath);
return elementResult.element.library;
} finally {
previous.makeCurrent();
}
}
/// Return `true` if the given [pathName] is in the Pub cache.
static bool _isPathInPubCache(String pathName) {
List<String> parts = path.split(pathName);
if (parts.contains('.pub-cache')) {
return true;
}
for (int i = 0; i < parts.length - 2; i++) {
if (parts[i] == 'Pub' && parts[i + 1] == 'Cache') {
return true;
}
}
return false;
}
}
/// This [Logger] prints out information comments to [outSink] and error messages
/// to [errorSink].
class StdLogger extends Logger {
StdLogger();
@override
void logError(String message, [CaughtException exception]) {
errorSink.writeln(message);
if (exception != null) {
errorSink.writeln(exception);
}
}
@override
void logInformation(String message, [CaughtException exception]) {
outSink.writeln(message);
if (exception != null) {
outSink.writeln(exception);
}
}
}