blob: 73d9ac2fb967f367b4392c40b691acf0df004c16 [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:mirrors';
import 'package:analyzer/error/error.dart';
import 'package:analyzer/file_system/file_system.dart';
import 'package:analyzer/file_system/memory_file_system.dart';
import 'package:analyzer/source/error_processor.dart';
import 'package:analyzer/src/analysis_options/analysis_options_provider.dart';
import 'package:analyzer/src/analysis_options/error/option_codes.dart';
import 'package:analyzer/src/dart/error/hint_codes.dart';
import 'package:analyzer/src/error/codes.dart';
import 'package:analyzer/src/file_system/file_system.dart';
import 'package:analyzer/src/generated/engine.dart';
import 'package:analyzer/src/generated/source.dart';
import 'package:analyzer/src/lint/linter.dart';
import 'package:analyzer/src/lint/registry.dart';
import 'package:analyzer/src/task/options.dart';
import 'package:test/test.dart';
import 'package:test_reflective_loader/test_reflective_loader.dart';
import 'package:yaml/yaml.dart';
import '../../generated/test_support.dart';
import '../../resource_utils.dart';
main() {
defineReflectiveSuite(() {
defineReflectiveTests(ContextConfigurationTest);
defineReflectiveTests(ErrorCodeValuesTest);
defineReflectiveTests(OptionsFileValidatorTest);
defineReflectiveTests(OptionsProviderTest);
});
}
@reflectiveTest
class ContextConfigurationTest {
final AnalysisOptions analysisOptions = AnalysisOptionsImpl();
final AnalysisOptionsProvider optionsProvider = AnalysisOptionsProvider();
void configureContext(String optionsSource) =>
applyToAnalysisOptions(analysisOptions, parseOptions(optionsSource));
YamlMap parseOptions(String source) =>
optionsProvider.getOptionsFromString(source);
test_configure_chromeos_checks() {
configureContext('''
analyzer:
optional-checks:
chrome-os-manifest-checks
''');
expect(true, analysisOptions.chromeOsManifestChecks);
}
test_configure_chromeos_checks_map() {
configureContext('''
analyzer:
optional-checks:
chrome-os-manifest-checks : true
''');
expect(true, analysisOptions.chromeOsManifestChecks);
}
test_configure_error_processors() {
configureContext('''
analyzer:
errors:
invalid_assignment: ignore
unused_local_variable: error
''');
List<ErrorProcessor> processors = analysisOptions.errorProcessors;
expect(processors, hasLength(2));
var unused_local =
AnalysisError(TestSource(), 0, 1, HintCode.UNUSED_LOCAL_VARIABLE, [
['x']
]);
var invalid_assignment = AnalysisError(
TestSource(), 0, 1, CompileTimeErrorCode.INVALID_ASSIGNMENT, [
['x'],
['y']
]);
// ignore
var invalidAssignment =
processors.firstWhere((p) => p.appliesTo(invalid_assignment));
expect(invalidAssignment.severity, isNull);
// error
var unusedLocal = processors.firstWhere((p) => p.appliesTo(unused_local));
expect(unusedLocal.severity, ErrorSeverity.ERROR);
}
test_configure_excludes() {
configureContext('''
analyzer:
exclude:
- foo/bar.dart
- 'test/**'
''');
List<String> excludes = analysisOptions.excludePatterns;
expect(excludes, unorderedEquals(['foo/bar.dart', 'test/**']));
}
test_configure_plugins_list() {
configureContext('''
analyzer:
plugins:
- angular2
- intl
''');
List<String> names = analysisOptions.enabledPluginNames;
expect(names, ['angular2', 'intl']);
}
test_configure_plugins_map() {
configureContext('''
analyzer:
plugins:
angular2:
enabled: true
''');
List<String> names = analysisOptions.enabledPluginNames;
expect(names, ['angular2']);
}
test_configure_plugins_string() {
configureContext('''
analyzer:
plugins:
angular2
''');
List<String> names = analysisOptions.enabledPluginNames;
expect(names, ['angular2']);
}
}
@reflectiveTest
class ErrorCodeValuesTest {
test_errorCodes() {
// Now that we're using unique names for comparison, the only reason to
// split the codes by class is to find all of the classes that need to be
// checked against `errorCodeValues`.
var errorTypeMap = <Type, List<ErrorCode>>{};
for (ErrorCode code in errorCodeValues) {
Type type = code.runtimeType;
errorTypeMap.putIfAbsent(type, () => <ErrorCode>[]).add(code);
}
StringBuffer missingCodes = StringBuffer();
errorTypeMap.forEach((Type errorType, List<ErrorCode> codes) {
var listedNames = codes.map((ErrorCode code) => code.uniqueName).toSet();
var declaredNames = reflectClass(errorType)
.declarations
.values
.map((DeclarationMirror declarationMirror) {
String name = declarationMirror.simpleName.toString();
//TODO(danrubel): find a better way to extract the text from the symbol
assert(name.startsWith('Symbol("') && name.endsWith('")'));
return errorType.toString() + '.' + name.substring(8, name.length - 2);
}).where((String name) {
return name == name.toUpperCase();
}).toList();
// Assert that all declared names are in errorCodeValues
for (String declaredName in declaredNames) {
if (!listedNames.contains(declaredName)) {
missingCodes.writeln();
missingCodes.write(' $declaredName');
}
}
});
if (missingCodes.isNotEmpty) {
fail('Missing error codes:$missingCodes');
}
}
}
class ErrorProcessorMatcher extends Matcher {
final ErrorProcessor required;
ErrorProcessorMatcher(this.required);
@override
Description describe(Description desc) => desc
..add("an ErrorProcessor setting ${required.code} to ${required.severity}");
@override
bool matches(dynamic o, Map<dynamic, dynamic> options) {
return o is ErrorProcessor &&
o.code.toUpperCase() == required.code.toUpperCase() &&
o.severity == required.severity;
}
}
@reflectiveTest
class OptionsFileValidatorTest {
final OptionsFileValidator validator = OptionsFileValidator(TestSource());
final AnalysisOptionsProvider optionsProvider = AnalysisOptionsProvider();
test_analyzer_enableExperiment_badValue() {
validate('''
analyzer:
enable-experiment:
- not-an-experiment
''', [AnalysisOptionsWarningCode.UNSUPPORTED_OPTION_WITHOUT_VALUES]);
}
test_analyzer_enableExperiment_notAList() {
validate('''
analyzer:
enable-experiment:
experiment: true
''', [AnalysisOptionsWarningCode.INVALID_SECTION_FORMAT]);
}
test_analyzer_error_code_supported() {
validate('''
analyzer:
errors:
unused_local_variable: ignore
invalid_assignment: warning
missing_return: error
dead_code: info
''', []);
}
test_analyzer_error_code_supported_bad_value() {
validate('''
analyzer:
errors:
unused_local_variable: ftw
''', [AnalysisOptionsWarningCode.UNSUPPORTED_OPTION_WITH_LEGAL_VALUES]);
}
test_analyzer_error_code_unsupported() {
validate('''
analyzer:
errors:
not_supported: ignore
''', [AnalysisOptionsWarningCode.UNRECOGNIZED_ERROR_CODE]);
}
test_analyzer_language_bad_format_list() {
validate('''
analyzer:
language:
- enableSuperMixins: true
''', [AnalysisOptionsWarningCode.INVALID_SECTION_FORMAT]);
}
test_analyzer_language_bad_format_scalar() {
validate('''
analyzer:
language: true
''', [AnalysisOptionsWarningCode.INVALID_SECTION_FORMAT]);
}
test_analyzer_language_supports_empty() {
validate('''
analyzer:
language:
''', []);
}
test_analyzer_language_unsupported_key() {
validate('''
analyzer:
language:
unsupported: true
''', [AnalysisOptionsWarningCode.UNSUPPORTED_OPTION_WITH_LEGAL_VALUES]);
}
test_analyzer_lint_codes_recognized() {
Registry.ruleRegistry.register(TestRule());
validate('''
analyzer:
errors:
fantastic_test_rule: ignore
''', []);
}
test_analyzer_strong_mode_deprecated() {
validate('''
analyzer:
strong-mode: true
''', [AnalysisOptionsHintCode.STRONG_MODE_SETTING_DEPRECATED]);
}
test_analyzer_strong_mode_deprecated_key() {
validate('''
analyzer:
strong-mode:
declaration-casts: false
''', [AnalysisOptionsWarningCode.ANALYSIS_OPTION_DEPRECATED]);
}
test_analyzer_strong_mode_error_code_supported() {
validate('''
analyzer:
errors:
invalid_cast_method: ignore
''', []);
}
test_analyzer_strong_mode_false_removed() {
validate('''
analyzer:
strong-mode: false
''', [AnalysisOptionsWarningCode.SPEC_MODE_REMOVED]);
}
test_analyzer_strong_mode_unsupported_key() {
validate('''
analyzer:
strong-mode:
unsupported: true
''', [AnalysisOptionsWarningCode.UNSUPPORTED_OPTION_WITH_LEGAL_VALUES]);
}
test_analyzer_strong_mode_unsupported_value() {
validate('''
analyzer:
strong-mode:
implicit-dynamic: foo
''', [AnalysisOptionsWarningCode.UNSUPPORTED_VALUE]);
}
test_analyzer_supported_exclude() {
validate('''
analyzer:
exclude:
- test/_data/p4/lib/lib1.dart
''', []);
}
test_analyzer_supported_strong_mode_supported_bad_value() {
validate('''
analyzer:
strong-mode: w00t
''', [AnalysisOptionsWarningCode.UNSUPPORTED_VALUE]);
}
test_analyzer_unsupported_option() {
validate('''
analyzer:
not_supported: true
''', [AnalysisOptionsWarningCode.UNSUPPORTED_OPTION_WITH_LEGAL_VALUES]);
}
test_chromeos_manifest_checks() {
validate('''
analyzer:
optional-checks:
chrome-os-manifest-checks
''', []);
}
test_chromeos_manifest_checks_invalid() {
validate('''
analyzer:
optional-checks:
chromeos-manifest
''', [AnalysisOptionsWarningCode.UNSUPPORTED_OPTION_WITH_LEGAL_VALUE]);
}
test_linter_supported_rules() {
Registry.ruleRegistry.register(TestRule());
validate('''
linter:
rules:
- fantastic_test_rule
''', []);
}
test_linter_unsupported_option() {
validate('''
linter:
unsupported: true
''', [AnalysisOptionsWarningCode.UNSUPPORTED_OPTION_WITH_LEGAL_VALUE]);
}
void validate(String source, List<ErrorCode> expected) {
var options = optionsProvider.getOptionsFromString(source);
var errors = validator.validate(options);
expect(errors.map((AnalysisError e) => e.errorCode),
unorderedEquals(expected));
}
}
@reflectiveTest
class OptionsProviderTest {
TestPathTranslator pathTranslator;
ResourceProvider resourceProvider;
AnalysisOptionsProvider provider;
String get optionsFilePath => '/analysis_options.yaml';
void setUp() {
var rawProvider = MemoryResourceProvider();
resourceProvider = TestResourceProvider(rawProvider);
pathTranslator = TestPathTranslator(rawProvider);
provider = AnalysisOptionsProvider(SourceFactory([
ResourceUriResolver(rawProvider),
]));
}
test_perform_include_merge() {
pathTranslator.newFile('/other_options.yaml', '''
analyzer:
exclude:
- toplevelexclude.dart
plugins:
toplevelplugin:
enabled: true
errors:
toplevelerror: warning
linter:
rules:
- toplevellint
''');
String code = r'''
include: other_options.yaml
analyzer:
exclude:
- lowlevelexclude.dart
plugins:
lowlevelplugin:
enabled: true
errors:
lowlevelerror: warning
linter:
rules:
- lowlevellint
''';
pathTranslator.newFile(optionsFilePath, code);
final lowlevellint = TestRule.withName('lowlevellint');
final toplevellint = TestRule.withName('toplevellint');
Registry.ruleRegistry.register(lowlevellint);
Registry.ruleRegistry.register(toplevellint);
final options = _getOptionsObject('/');
expect(options.lintRules, unorderedEquals([toplevellint, lowlevellint]));
expect(options.enabledPluginNames,
unorderedEquals(['toplevelplugin', 'lowlevelplugin']));
expect(options.excludePatterns,
unorderedEquals(['toplevelexclude.dart', 'lowlevelexclude.dart']));
expect(
options.errorProcessors,
unorderedMatches([
ErrorProcessorMatcher(
ErrorProcessor('toplevelerror', ErrorSeverity.WARNING)),
ErrorProcessorMatcher(
ErrorProcessor('lowlevelerror', ErrorSeverity.WARNING))
]));
}
YamlMap _getOptions(String posixPath, {bool crawlUp = false}) {
Resource resource = pathTranslator.getResource(posixPath);
return provider.getOptions(resource, crawlUp: crawlUp);
}
AnalysisOptions _getOptionsObject(String posixPath, {bool crawlUp = false}) {
final map = _getOptions(posixPath, crawlUp: crawlUp);
final options = AnalysisOptionsImpl();
applyToAnalysisOptions(options, map);
return options;
}
}
class TestRule extends LintRule {
TestRule() : super(name: 'fantastic_test_rule', description: '');
TestRule.withName(String name) : super(name: name, description: '');
}