| // Copyright (c) 2020, 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:analyzer/dart/analysis/features.dart'; |
| import 'package:analyzer/dart/element/element.dart'; |
| import 'package:analyzer/src/dart/analysis/experiments.dart'; |
| import 'package:analyzer/src/dart/element/element.dart'; |
| import 'package:analyzer/src/error/codes.dart'; |
| import 'package:collection/collection.dart'; |
| import 'package:pub_semver/pub_semver.dart'; |
| import 'package:test/test.dart'; |
| import 'package:test_reflective_loader/test_reflective_loader.dart'; |
| |
| import '../../../util/element_printer.dart'; |
| import '../../../util/tree_string_sink.dart'; |
| import 'context_collection_resolution.dart'; |
| |
| main() { |
| defineReflectiveSuite(() { |
| defineReflectiveTests(LibraryElementTest_featureSet); |
| defineReflectiveTests(LibraryElementTest_scope); |
| defineReflectiveTests(LibraryElementTest_toString); |
| }); |
| } |
| |
| @reflectiveTest |
| class LibraryElementTest_featureSet extends PubPackageResolutionTest { |
| test_language205() async { |
| writeTestPackageConfig( |
| PackageConfigFileBuilder(), |
| languageVersion: '2.5', |
| ); |
| |
| await resolveTestCode(''); |
| |
| _assertLanguageVersion( |
| package: Version.parse('2.5.0'), |
| override: null, |
| ); |
| |
| _assertFeatureSet([ |
| Feature.constant_update_2018, |
| Feature.control_flow_collections, |
| Feature.set_literals, |
| Feature.spread_collections, |
| ]); |
| } |
| |
| test_language208() async { |
| writeTestPackageConfig( |
| PackageConfigFileBuilder(), |
| languageVersion: '2.8', |
| ); |
| |
| await resolveTestCode(''); |
| |
| _assertLanguageVersion( |
| package: Version.parse('2.8.0'), |
| override: null, |
| ); |
| |
| _assertFeatureSet([ |
| Feature.constant_update_2018, |
| Feature.control_flow_collections, |
| Feature.extension_methods, |
| Feature.set_literals, |
| Feature.spread_collections, |
| ]); |
| } |
| |
| test_language208_override205() async { |
| writeTestPackageConfig( |
| PackageConfigFileBuilder(), |
| languageVersion: '2.8', |
| ); |
| |
| await resolveTestCode('// @dart = 2.5'); |
| |
| // Valid override, less than the latest supported language version. |
| _assertLanguageVersion( |
| package: Version.parse('2.8.0'), |
| override: Version.parse('2.5.0'), |
| ); |
| |
| _assertFeatureSet([ |
| Feature.constant_update_2018, |
| Feature.control_flow_collections, |
| Feature.set_literals, |
| Feature.spread_collections, |
| ]); |
| } |
| |
| test_language209() async { |
| writeTestPackageConfig( |
| PackageConfigFileBuilder(), |
| languageVersion: '2.9', |
| ); |
| |
| await resolveTestCode(''); |
| |
| _assertLanguageVersion( |
| package: Version.parse('2.9.0'), |
| override: null, |
| ); |
| |
| _assertFeatureSet([ |
| Feature.constant_update_2018, |
| Feature.control_flow_collections, |
| Feature.extension_methods, |
| Feature.set_literals, |
| Feature.spread_collections, |
| ]); |
| } |
| |
| test_language212_override399() async { |
| writeTestPackageConfig( |
| PackageConfigFileBuilder(), |
| languageVersion: '2.12', |
| ); |
| |
| await resolveTestCode('// @dart = 3.99'); |
| |
| // Invalid override: minor is greater than the latest minor. |
| _assertLanguageVersion( |
| package: Version.parse('2.12.0'), |
| override: null, |
| ); |
| |
| _assertFeatureSet([ |
| Feature.constant_update_2018, |
| Feature.control_flow_collections, |
| Feature.extension_methods, |
| Feature.non_nullable, |
| Feature.set_literals, |
| Feature.spread_collections, |
| ]); |
| } |
| |
| test_language212_override400() async { |
| writeTestPackageConfig( |
| PackageConfigFileBuilder(), |
| languageVersion: '2.12', |
| ); |
| |
| await resolveTestCode('// @dart = 4.00'); |
| |
| // Invalid override: major is greater than the latest major. |
| _assertLanguageVersion( |
| package: Version.parse('2.12.0'), |
| override: null, |
| ); |
| |
| _assertFeatureSet([ |
| Feature.constant_update_2018, |
| Feature.control_flow_collections, |
| Feature.extension_methods, |
| Feature.non_nullable, |
| Feature.set_literals, |
| Feature.spread_collections, |
| ]); |
| } |
| |
| void _assertFeatureSet(List<Feature> expected) { |
| var featureSet = result.libraryElement.featureSet; |
| |
| var actual = ExperimentStatus.knownFeatures.values |
| .where(featureSet.isEnabled) |
| .toSet(); |
| |
| expect(actual, unorderedEquals(expected)); |
| } |
| |
| void _assertLanguageVersion({ |
| required Version package, |
| required Version? override, |
| }) async { |
| var element = result.libraryElement; |
| expect(element.languageVersion.package, package); |
| expect(element.languageVersion.override, override); |
| } |
| } |
| |
| @reflectiveTest |
| class LibraryElementTest_scope extends PubPackageResolutionTest { |
| test_lookup() async { |
| await assertNoErrorsInCode(r''' |
| int foo = 0; |
| '''); |
| |
| var scope = result.libraryElement.scope; |
| |
| assertElement( |
| scope.lookup('foo').getter, |
| findElement.topGet('foo'), |
| ); |
| assertElement( |
| scope.lookup('foo').setter, |
| findElement.topSet('foo'), |
| ); |
| } |
| |
| test_lookup_extension_unnamed() async { |
| await assertNoErrorsInCode(r''' |
| extension on int {} |
| '''); |
| |
| var scope = result.libraryElement.scope; |
| |
| assertElementNull( |
| scope.lookup('').getter, |
| ); |
| } |
| |
| test_lookup_extensions_imported() async { |
| newFile('$testPackageLibPath/a.dart', r''' |
| extension E on int {} |
| '''); |
| |
| await assertErrorsInCode(r''' |
| import 'a.dart'; |
| ''', [ |
| error(WarningCode.UNUSED_IMPORT, 7, 8), |
| ]); |
| |
| _assertLibraryExtensions(result.libraryElement, r''' |
| extensions |
| package:test/a.dart::@extension::E |
| dart:core::@extension::EnumName |
| '''); |
| } |
| |
| test_lookup_extensions_imported_withPrefix() async { |
| newFile('$testPackageLibPath/a.dart', r''' |
| extension E on int {} |
| '''); |
| |
| await assertErrorsInCode(r''' |
| import 'a.dart' as prefix; |
| ''', [ |
| error(WarningCode.UNUSED_IMPORT, 7, 8), |
| ]); |
| |
| _assertLibraryExtensions(result.libraryElement, r''' |
| extensions |
| package:test/a.dart::@extension::E |
| dart:core::@extension::EnumName |
| '''); |
| } |
| |
| test_lookup_extensions_local() async { |
| await assertNoErrorsInCode(r''' |
| extension E on int {} |
| '''); |
| |
| _assertLibraryExtensions(result.libraryElement, r''' |
| extensions |
| self::@extension::E |
| dart:core::@extension::EnumName |
| '''); |
| } |
| |
| test_lookup_extensions_local_withAugmentation() async { |
| newFile('$testPackageLibPath/a.dart', r''' |
| augment library 'test.dart'; |
| |
| augment extension E { |
| void foo() {} |
| } |
| '''); |
| |
| await assertNoErrorsInCode(r''' |
| import augment 'a.dart'; |
| |
| extension E on int {} |
| '''); |
| |
| _assertLibraryExtensions(result.libraryElement, r''' |
| extensions |
| self::@extension::E |
| dart:core::@extension::EnumName |
| '''); |
| } |
| |
| test_lookup_implicitCoreImport() async { |
| await assertNoErrorsInCode(''); |
| |
| var scope = result.libraryElement.scope; |
| |
| assertElement( |
| scope.lookup('int').getter, |
| intElement, |
| ); |
| } |
| |
| test_lookup_notFound() async { |
| await assertNoErrorsInCode(''); |
| |
| var scope = result.libraryElement.scope; |
| |
| assertElementNull( |
| scope.lookup('noSuchGetter').getter, |
| ); |
| |
| assertElementNull( |
| scope.lookup('noSuchSetter').setter, |
| ); |
| } |
| |
| test_lookup_prefersLocal() async { |
| await assertNoErrorsInCode(r''' |
| // ignore:unused_import |
| import 'dart:math'; |
| |
| int sin() => 3; |
| '''); |
| |
| var scope = result.libraryElement.scope; |
| |
| assertElement( |
| scope.lookup('sin').getter, |
| findElement.topFunction('sin'), |
| ); |
| |
| assertElement( |
| scope.lookup('cos').getter, |
| findElement.importFind('dart:math').topFunction('cos'), |
| ); |
| } |
| |
| test_lookup_prefix() async { |
| await assertNoErrorsInCode(r''' |
| // ignore:unused_import |
| import 'dart:math' as math; |
| '''); |
| |
| var scope = result.libraryElement.scope; |
| |
| assertElement( |
| scope.lookup('math').getter, |
| findElement.prefix('math'), |
| ); |
| } |
| |
| test_lookup_respectsCombinator_hide() async { |
| await assertNoErrorsInCode(r''' |
| // ignore:unused_import |
| import 'dart:math' hide sin; |
| '''); |
| |
| var scope = result.libraryElement.scope; |
| |
| assertElementNull( |
| scope.lookup('sin').getter, |
| ); |
| |
| var mathFind = findElement.importFind('dart:math'); |
| assertElement( |
| scope.lookup('cos').getter, |
| mathFind.topFunction('cos'), |
| ); |
| assertElement( |
| scope.lookup('tan').getter, |
| mathFind.topFunction('tan'), |
| ); |
| } |
| |
| test_lookup_respectsCombinator_show() async { |
| await assertNoErrorsInCode(r''' |
| // ignore:unused_import |
| import 'dart:math' show sin; |
| '''); |
| |
| var scope = result.libraryElement.scope; |
| |
| assertElement( |
| scope.lookup('sin').getter, |
| findElement.importFind('dart:math').topFunction('sin'), |
| ); |
| |
| assertElementNull( |
| scope.lookup('cos').getter, |
| ); |
| } |
| |
| void _assertLibraryExtensions(LibraryElement library, String expected) { |
| library as LibraryElementImpl; |
| |
| var buffer = StringBuffer(); |
| var sink = TreeStringSink(sink: buffer, indent: ''); |
| var elementPrinter = ElementPrinter( |
| sink: sink, |
| configuration: ElementPrinterConfiguration(), |
| selfUriStr: result.uri.toString(), |
| ); |
| |
| var extensions = library.scope.extensions; |
| extensions = extensions.sortedBy((e) => e.name ?? ''); |
| elementPrinter.writeElementList('extensions', extensions); |
| |
| var actual = buffer.toString(); |
| if (actual != expected) { |
| fail(''' |
| \r${'-' * 28} Actual ${'-' * 28} |
| \r${actual.trimRight().split('\n').join('\n\r')} |
| \r${'-' * 64} |
| '''); |
| } |
| } |
| } |
| |
| @reflectiveTest |
| class LibraryElementTest_toString extends PubPackageResolutionTest { |
| test_hasLibraryDirective_hasName() async { |
| await assertNoErrorsInCode(r''' |
| library my.name; |
| '''); |
| |
| expect( |
| result.libraryElement.toString(), |
| 'library package:test/test.dart', |
| ); |
| } |
| |
| test_hasLibraryDirective_noName() async { |
| await assertNoErrorsInCode(r''' |
| library; |
| '''); |
| |
| expect( |
| result.libraryElement.toString(), |
| 'library package:test/test.dart', |
| ); |
| } |
| |
| test_noLibraryDirective() async { |
| await assertNoErrorsInCode(r''' |
| class A {} |
| '''); |
| |
| expect( |
| result.libraryElement.toString(), |
| 'library package:test/test.dart', |
| ); |
| } |
| } |