| // Copyright (c) 2014, 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 'package:_fe_analyzer_shared/src/scanner/error_token.dart'; |
| import 'package:analyzer/dart/analysis/features.dart'; |
| import 'package:analyzer/dart/ast/token.dart'; |
| import 'package:analyzer/error/listener.dart'; |
| import 'package:analyzer/source/line_info.dart'; |
| import 'package:analyzer/src/dart/scanner/reader.dart'; |
| import 'package:analyzer/src/dart/scanner/scanner.dart'; |
| import 'package:analyzer/src/string_source.dart'; |
| import 'package:analyzer_testing/resource_provider_mixin.dart'; |
| import 'package:test/test.dart'; |
| import 'package:test_reflective_loader/test_reflective_loader.dart'; |
| |
| import 'test_support.dart'; |
| |
| main() { |
| defineReflectiveSuite(() { |
| defineReflectiveTests(LineInfoTest); |
| defineReflectiveTests(ScannerTest); |
| }); |
| } |
| |
| class CharacterRangeReaderTest { |
| void test_advance() { |
| CharSequenceReader baseReader = CharSequenceReader("xyzzy"); |
| CharacterRangeReader reader = CharacterRangeReader(baseReader, 1, 4); |
| expect(reader.advance(), 0x79); |
| expect(reader.advance(), 0x80); |
| expect(reader.advance(), 0x80); |
| expect(reader.advance(), -1); |
| expect(reader.advance(), -1); |
| } |
| |
| void test_creation() { |
| CharSequenceReader baseReader = CharSequenceReader("xyzzy"); |
| CharacterRangeReader reader = CharacterRangeReader(baseReader, 1, 4); |
| expect(reader, isNotNull); |
| } |
| |
| void test_getOffset() { |
| CharSequenceReader baseReader = CharSequenceReader("xyzzy"); |
| CharacterRangeReader reader = CharacterRangeReader(baseReader, 1, 2); |
| expect(reader.offset, 1); |
| reader.advance(); |
| expect(reader.offset, 2); |
| reader.advance(); |
| expect(reader.offset, 2); |
| } |
| |
| void test_getString() { |
| CharSequenceReader baseReader = CharSequenceReader("__xyzzy__"); |
| CharacterRangeReader reader = CharacterRangeReader(baseReader, 2, 7); |
| reader.offset = 5; |
| expect(reader.getString(3, 0), "yzz"); |
| expect(reader.getString(4, 1), "zzy"); |
| } |
| |
| void test_peek() { |
| CharSequenceReader baseReader = CharSequenceReader("xyzzy"); |
| CharacterRangeReader reader = CharacterRangeReader(baseReader, 1, 3); |
| expect(reader.peek(), 0x79); |
| expect(reader.peek(), 0x79); |
| reader.advance(); |
| expect(reader.peek(), 0x80); |
| expect(reader.peek(), 0x80); |
| reader.advance(); |
| expect(reader.peek(), -1); |
| expect(reader.peek(), -1); |
| } |
| |
| void test_setOffset() { |
| CharSequenceReader baseReader = CharSequenceReader("xyzzy"); |
| CharacterRangeReader reader = CharacterRangeReader(baseReader, 1, 4); |
| reader.offset = 2; |
| expect(reader.offset, 2); |
| } |
| } |
| |
| @reflectiveTest |
| class LineInfoTest { |
| final featureSet = FeatureSet.latestLanguageVersion(); |
| |
| void test_lineInfo_multilineComment() { |
| String source = "/*\r\n *\r\n */"; |
| _assertLineInfo(source, [ |
| ScannerTest_ExpectedLocation(0, 1, 1), |
| ScannerTest_ExpectedLocation(5, 2, 2), |
| ScannerTest_ExpectedLocation(source.length - 1, 3, 3), |
| ]); |
| } |
| |
| void test_lineInfo_multilineString() { |
| String source = "'''a\r\nbc\r\nd'''"; |
| _assertLineInfo(source, [ |
| ScannerTest_ExpectedLocation(0, 1, 1), |
| ScannerTest_ExpectedLocation(7, 2, 2), |
| ScannerTest_ExpectedLocation(source.length - 1, 3, 4), |
| ]); |
| } |
| |
| void test_lineInfo_multilineString_raw() { |
| String source = "var a = r'''\nblah\n''';\n\nfoo"; |
| _assertLineInfo(source, [ |
| ScannerTest_ExpectedLocation(0, 1, 1), |
| ScannerTest_ExpectedLocation(14, 2, 2), |
| ScannerTest_ExpectedLocation(source.length - 2, 5, 2), |
| ]); |
| } |
| |
| void test_lineInfo_simpleClass() { |
| String source = |
| "class Test {\r\n String s = '...';\r\n int get x => s.MISSING_GETTER;\r\n}"; |
| _assertLineInfo(source, [ |
| ScannerTest_ExpectedLocation(0, 1, 1), |
| ScannerTest_ExpectedLocation(source.indexOf("MISSING_GETTER"), 3, 20), |
| ScannerTest_ExpectedLocation(source.length - 1, 4, 1), |
| ]); |
| } |
| |
| void test_lineInfo_slashN() { |
| String source = "class Test {\n}"; |
| _assertLineInfo(source, [ |
| ScannerTest_ExpectedLocation(0, 1, 1), |
| ScannerTest_ExpectedLocation(source.indexOf("}"), 2, 1), |
| ]); |
| } |
| |
| void test_linestarts() { |
| String source = "var\r\ni\n=\n1;\n"; |
| GatheringDiagnosticListener listener = GatheringDiagnosticListener(); |
| Scanner scanner = |
| Scanner(TestSource(), CharSequenceReader(source), listener) |
| ..configureFeatures( |
| featureSetForOverriding: featureSet, |
| featureSet: featureSet, |
| ); |
| var token = scanner.tokenize(); |
| expect(token.lexeme, 'var'); |
| var lineStarts = scanner.lineStarts; |
| expect(lineStarts, orderedEquals([0, 5, 7, 9, 12])); |
| } |
| |
| void test_translate_missing_closing_gt_error() { |
| // Ensure that the UnmatchedToken error for missing '>' is translated |
| // to the correct analyzer error code. |
| // See https://github.com/dart-lang/sdk/issues/30320 |
| String source = '<!-- @Component('; |
| GatheringDiagnosticListener listener = GatheringDiagnosticListener(); |
| Scanner scanner = |
| Scanner(TestSource(), CharSequenceReader(source), listener) |
| ..configureFeatures( |
| featureSetForOverriding: featureSet, |
| featureSet: featureSet, |
| ); |
| Token token = scanner.tokenize(reportScannerErrors: false); |
| expect(token, TypeMatcher<UnmatchedToken>()); |
| token = token.next!; |
| expect(token, TypeMatcher<UnmatchedToken>()); |
| token = token.next!; |
| expect(token, isNot(TypeMatcher<ErrorToken>())); |
| } |
| |
| void _assertLineInfo( |
| String source, |
| List<ScannerTest_ExpectedLocation> expectedLocations, |
| ) { |
| GatheringDiagnosticListener listener = GatheringDiagnosticListener(); |
| _scanWithListener(source, listener); |
| listener.assertNoErrors(); |
| LineInfo info = listener.getLineInfo(TestSource())!; |
| expect(info, isNotNull); |
| int count = expectedLocations.length; |
| for (int i = 0; i < count; i++) { |
| ScannerTest_ExpectedLocation expectedLocation = expectedLocations[i]; |
| var location = info.getLocation(expectedLocation._offset); |
| expect( |
| location.lineNumber, |
| expectedLocation._lineNumber, |
| reason: 'Line number in location $i', |
| ); |
| expect( |
| location.columnNumber, |
| expectedLocation._columnNumber, |
| reason: 'Column number in location $i', |
| ); |
| } |
| } |
| |
| Token _scanWithListener(String source, GatheringDiagnosticListener listener) { |
| Scanner scanner = |
| Scanner(TestSource(), CharSequenceReader(source), listener) |
| ..configureFeatures( |
| featureSetForOverriding: featureSet, |
| featureSet: featureSet, |
| ); |
| Token result = scanner.tokenize(); |
| LineInfo lineInfo = LineInfo(scanner.lineStarts); |
| listener.setLineInfo(TestSource(), lineInfo); |
| return result; |
| } |
| } |
| |
| @reflectiveTest |
| class ScannerTest with ResourceProviderMixin { |
| test_featureSet() { |
| var scanner = _createScanner(r''' |
| // @dart = 2.0 |
| '''); |
| var defaultFeatureSet = FeatureSet.latestLanguageVersion(); |
| expect(defaultFeatureSet.isEnabled(Feature.extension_methods), isTrue); |
| |
| scanner.configureFeatures( |
| featureSetForOverriding: FeatureSet.latestLanguageVersion(), |
| featureSet: FeatureSet.latestLanguageVersion(), |
| ); |
| scanner.tokenize(); |
| |
| var featureSet = scanner.featureSet; |
| expect(featureSet.isEnabled(Feature.extension_methods), isFalse); |
| } |
| |
| test_featureSet_majorOverflow() { |
| var scanner = _createScanner(r''' |
| // @dart = 99999999999999999999999999999999.0 |
| '''); |
| var featureSet = FeatureSet.latestLanguageVersion(); |
| scanner.configureFeatures( |
| featureSetForOverriding: featureSet, |
| featureSet: featureSet, |
| ); |
| scanner.tokenize(); |
| // Don't check features, but should not crash. |
| } |
| |
| test_featureSet_minorOverflow() { |
| var scanner = _createScanner(r''' |
| // @dart = 3.99999999999999999999999999999999 |
| '''); |
| var featureSet = FeatureSet.latestLanguageVersion(); |
| scanner.configureFeatures( |
| featureSetForOverriding: featureSet, |
| featureSet: featureSet, |
| ); |
| scanner.tokenize(); |
| // Don't check features, but should not crash. |
| } |
| |
| Scanner _createScanner(String content) { |
| var path = convertPath('/test/lib/a.dart'); |
| var source = StringSource(content, path); |
| var reader = CharSequenceReader(content); |
| var diagnosticCollector = RecordingDiagnosticListener(); |
| return Scanner(source, reader, diagnosticCollector); |
| } |
| } |
| |
| /// An `ExpectedLocation` encodes information about the expected location of a |
| /// given offset in source code. |
| class ScannerTest_ExpectedLocation { |
| final int _offset; |
| |
| final int _lineNumber; |
| |
| final int _columnNumber; |
| |
| ScannerTest_ExpectedLocation( |
| this._offset, |
| this._lineNumber, |
| this._columnNumber, |
| ); |
| } |