OLD | NEW |
1 // Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 /// Types needed to implement "strong" checking in the Dart analyzer. | 5 /// Types needed to implement "strong" checking in the Dart analyzer. |
6 /// This is intended to be used by analyzer_cli and analysis_server packages. | 6 /// This is intended to be used by analyzer_cli and analysis_server packages. |
7 library dev_compiler.strong_mode; | 7 library dev_compiler.strong_mode; |
8 | 8 |
9 import 'package:analyzer/src/generated/engine.dart' | 9 import 'package:analyzer/src/generated/engine.dart' |
10 show AnalysisContextImpl, AnalysisErrorInfo, AnalysisErrorInfoImpl; | 10 show AnalysisContextImpl, AnalysisErrorInfo, AnalysisErrorInfoImpl; |
11 import 'package:analyzer/src/generated/error.dart' | 11 import 'package:analyzer/src/generated/error.dart' |
12 show | 12 show |
13 AnalysisError, | 13 AnalysisError, |
| 14 AnalysisErrorListener, |
| 15 CompileTimeErrorCode, |
14 ErrorCode, | 16 ErrorCode, |
15 CompileTimeErrorCode, | 17 ErrorSeverity, |
16 StaticTypeWarningCode, | 18 HintCode, |
17 HintCode; | 19 StaticTypeWarningCode; |
18 import 'package:analyzer/src/generated/source.dart' show Source; | 20 import 'package:analyzer/src/generated/source.dart' show Source; |
19 import 'package:args/args.dart'; | 21 import 'package:args/args.dart'; |
20 import 'package:logging/logging.dart' show Level; | |
21 | 22 |
22 import 'src/checker/checker.dart' show CodeChecker; | 23 import 'src/checker/checker.dart' show CodeChecker; |
23 import 'src/checker/resolver.dart' show LibraryResolverWithInference; | 24 import 'src/checker/resolver.dart' show LibraryResolverWithInference; |
24 import 'src/checker/rules.dart' show RestrictedRules; | 25 import 'src/checker/rules.dart' show RestrictedRules; |
25 import 'src/report.dart' show CheckerReporter, Message; | |
26 | 26 |
27 /// A type checker for Dart code that operates under stronger rules, and has | 27 /// A type checker for Dart code that operates under stronger rules, and has |
28 /// the ability to do local type inference in some situations. | 28 /// the ability to do local type inference in some situations. |
29 class StrongChecker { | 29 class StrongChecker { |
30 final AnalysisContextImpl _context; | 30 final AnalysisContextImpl _context; |
31 final CodeChecker _checker; | 31 final CodeChecker _checker; |
32 final _ErrorReporter _reporter; | 32 final _ErrorCollector _reporter; |
33 final StrongModeOptions _options; | |
34 | 33 |
35 StrongChecker._(this._context, this._options, this._checker, this._reporter); | 34 StrongChecker._(this._context, this._checker, this._reporter); |
36 | 35 |
37 factory StrongChecker( | 36 factory StrongChecker( |
38 AnalysisContextImpl context, StrongModeOptions options) { | 37 AnalysisContextImpl context, StrongModeOptions options) { |
39 // TODO(jmesserly): is there a cleaner way to plug this in? | 38 // TODO(jmesserly): is there a cleaner way to plug this in? |
40 if (context.libraryResolverFactory != null) { | 39 if (context.libraryResolverFactory != null) { |
41 throw new ArgumentError.value(context, 'context', | 40 throw new ArgumentError.value(context, 'context', |
42 'Analysis context must not have libraryResolverFactory already set.'); | 41 'Analysis context must not have libraryResolverFactory already set.'); |
43 } | 42 } |
44 context.libraryResolverFactory = | 43 context.libraryResolverFactory = |
45 (c) => new LibraryResolverWithInference(c, options); | 44 (c) => new LibraryResolverWithInference(c, options); |
46 | 45 |
47 var rules = new RestrictedRules(context.typeProvider, options: options); | 46 var rules = new RestrictedRules(context.typeProvider, options: options); |
48 var reporter = new _ErrorReporter(); | 47 var reporter = new _ErrorCollector(options.hints); |
49 var checker = new CodeChecker(rules, reporter, options); | 48 var checker = new CodeChecker(rules, reporter, options); |
50 return new StrongChecker._(context, options, checker, reporter); | 49 return new StrongChecker._(context, checker, reporter); |
51 } | 50 } |
52 | 51 |
53 /// Computes and returns DDC errors for the [source]. | 52 /// Computes and returns DDC errors for the [source]. |
54 AnalysisErrorInfo computeErrors(Source source) { | 53 AnalysisErrorInfo computeErrors(Source source) { |
55 var errors = new List<AnalysisError>(); | 54 var errors = new List<AnalysisError>(); |
56 | 55 _reporter.errors = errors; |
57 // TODO(jmesserly): change DDC to emit ErrorCodes directly. | |
58 _reporter._log = (Message msg) { | |
59 // Skip hints unless requested. | |
60 if (msg.level < Level.WARNING && !_options.hints) return; | |
61 | |
62 var errorCodeFactory = _levelToErrorCode[msg.level]; | |
63 var category = '${msg.runtimeType}'; | |
64 var errorCode = errorCodeFactory(category, msg.message); | |
65 var len = msg.end - msg.begin; | |
66 errors.add(new AnalysisError(source, msg.begin, len, errorCode)); | |
67 }; | |
68 | 56 |
69 for (Source librarySource in _context.getLibrariesContaining(source)) { | 57 for (Source librarySource in _context.getLibrariesContaining(source)) { |
70 var resolved = _context.resolveCompilationUnit2(source, librarySource); | 58 var resolved = _context.resolveCompilationUnit2(source, librarySource); |
71 _checker.visitCompilationUnit(resolved); | 59 _checker.visitCompilationUnit(resolved); |
72 } | 60 } |
73 _reporter._log = null; | 61 _reporter.errors = null; |
| 62 |
74 return new AnalysisErrorInfoImpl(errors, _context.getLineInfo(source)); | 63 return new AnalysisErrorInfoImpl(errors, _context.getLineInfo(source)); |
75 } | 64 } |
76 } | 65 } |
77 | 66 |
78 /// Maps a DDC log level to an analyzer ErrorCode subclass. | 67 class _ErrorCollector implements AnalysisErrorListener { |
79 final _levelToErrorCode = <Level, _ErrorCodeFactory>{ | 68 List<AnalysisError> errors; |
80 Level.SEVERE: (n, m) => new CompileTimeErrorCode(n, m), | 69 final bool hints; |
81 Level.WARNING: (n, m) => new StaticTypeWarningCode(n, m), | 70 _ErrorCollector(this.hints); |
82 Level.INFO: (n, m) => new HintCode(n, m) | |
83 }; | |
84 | 71 |
85 class _ErrorReporter implements CheckerReporter { | 72 void onError(AnalysisError error) { |
86 _CheckerReporterLog _log; | 73 // Unless DDC hints are requested, filter them out. |
87 void log(Message message) => _log(message); | 74 var HINT = ErrorSeverity.INFO.ordinal; |
| 75 if (hints || error.errorCode.errorSeverity.ordinal > HINT) { |
| 76 errors.add(error); |
| 77 } |
| 78 } |
88 } | 79 } |
89 | 80 |
90 class StrongModeOptions { | 81 class StrongModeOptions { |
91 | 82 |
92 /// Whether to infer return types and field types from overridden members. | 83 /// Whether to infer return types and field types from overridden members. |
93 final bool inferFromOverrides; | 84 final bool inferFromOverrides; |
94 static const inferFromOverridesDefault = true; | 85 static const inferFromOverridesDefault = true; |
95 | 86 |
96 /// Whether to infer types for consts and fields by looking at initializers on | 87 /// Whether to infer types for consts and fields by looking at initializers on |
97 /// the RHS. For example, in a constant declaration like: | 88 /// the RHS. For example, in a constant declaration like: |
(...skipping 91 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
189 return inferFromOverrides == s.inferFromOverrides && | 180 return inferFromOverrides == s.inferFromOverrides && |
190 inferTransitively == s.inferTransitively && | 181 inferTransitively == s.inferTransitively && |
191 onlyInferConstsAndFinalFields == s.onlyInferConstsAndFinalFields && | 182 onlyInferConstsAndFinalFields == s.onlyInferConstsAndFinalFields && |
192 inferDownwards == s.inferDownwards && | 183 inferDownwards == s.inferDownwards && |
193 relaxedCasts == s.relaxedCasts && | 184 relaxedCasts == s.relaxedCasts && |
194 nonnullableTypes.length == s.nonnullableTypes.length && | 185 nonnullableTypes.length == s.nonnullableTypes.length && |
195 new Set.from(nonnullableTypes).containsAll(s.nonnullableTypes); | 186 new Set.from(nonnullableTypes).containsAll(s.nonnullableTypes); |
196 } | 187 } |
197 } | 188 } |
198 | 189 |
199 typedef void _CheckerReporterLog(Message message); | |
200 typedef ErrorCode _ErrorCodeFactory(String name, String message); | |
201 | |
202 List<String> _optionsToList(String option, | 190 List<String> _optionsToList(String option, |
203 {List<String> defaultValue: const <String>[]}) { | 191 {List<String> defaultValue: const <String>[]}) { |
204 if (option == null) { | 192 if (option == null) { |
205 return defaultValue; | 193 return defaultValue; |
206 } else if (option.isEmpty) { | 194 } else if (option.isEmpty) { |
207 return <String>[]; | 195 return <String>[]; |
208 } else { | 196 } else { |
209 return option.split(','); | 197 return option.split(','); |
210 } | 198 } |
211 } | 199 } |
OLD | NEW |