| // Copyright (c) 2021, 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/src/diagnostic/diagnostic.dart' as diag; |
| import 'package:test/test.dart'; |
| import 'package:test_reflective_loader/test_reflective_loader.dart'; |
| |
| import '../dart/resolution/context_collection_resolution.dart'; |
| |
| main() { |
| defineReflectiveSuite(() { |
| defineReflectiveTests(UnnecessaryImportTest); |
| }); |
| } |
| |
| @reflectiveTest |
| class UnnecessaryImportTest extends PubPackageResolutionTest { |
| test_library_annotationOnDirective() async { |
| newFile('$testPackageLibPath/lib1.dart', r''' |
| class A { |
| const A() {} |
| } |
| '''); |
| await assertNoErrorsInCode(r''' |
| @A() |
| import 'lib1.dart'; |
| '''); |
| } |
| |
| test_library_as() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| class A {} |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', ''' |
| export 'lib1.dart'; |
| class B {} |
| '''); |
| await assertNoErrorsInCode(''' |
| import 'lib1.dart'; |
| import 'lib2.dart' as two; |
| f(A a, two.B b) {} |
| '''); |
| } |
| |
| test_library_as_differentPrefixes() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| class A {} |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', ''' |
| export 'lib1.dart'; |
| class B {} |
| '''); |
| await assertNoErrorsInCode(''' |
| import 'lib1.dart' as one; |
| import 'lib2.dart' as two; |
| f(one.A a, two.B b) {} |
| '''); |
| } |
| |
| test_library_as_equalPrefixes_referenced() async { |
| newFile('$testPackageLibPath/lib1.dart', r''' |
| class A {} |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', r''' |
| class B {} |
| '''); |
| await assertNoErrorsInCode(r''' |
| import 'lib1.dart' as one; |
| import 'lib2.dart' as one; |
| f(one.A a, one.B b) {} |
| '''); |
| } |
| |
| test_library_as_equalPrefixes_referenced_via_export() async { |
| newFile('$testPackageLibPath/lib1.dart', r''' |
| class A {} |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', r''' |
| class B {} |
| '''); |
| newFile('$testPackageLibPath/lib3.dart', r''' |
| export 'lib2.dart'; |
| '''); |
| await assertNoErrorsInCode(r''' |
| import 'lib1.dart' as one; |
| import 'lib3.dart' as one; |
| f(one.A a, one.B b) {} |
| '''); |
| } |
| |
| test_library_as_equalPrefixes_unreferenced() async { |
| newFile('$testPackageLibPath/lib1.dart', r''' |
| class A {} |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', r''' |
| class B {} |
| '''); |
| await assertNoErrorsInCode(r''' |
| import 'lib1.dart' as one; |
| import 'lib2.dart' as one; // ignore: unused_import |
| f(one.A a) {} |
| '''); |
| } |
| |
| test_library_as_show_multipleElements() async { |
| newFile('$testPackageLibPath/lib1.dart', r''' |
| class A {} |
| class B {} |
| '''); |
| await assertNoErrorsInCode(r''' |
| import 'lib1.dart' as one show A, B; |
| f(one.A a, one.B b) {} |
| '''); |
| } |
| |
| test_library_as_showTopLevelFunction() async { |
| newFile('$testPackageLibPath/lib1.dart', r''' |
| class One {} |
| topLevelFunction() {} |
| '''); |
| await assertNoErrorsInCode(r''' |
| import 'lib1.dart' hide topLevelFunction; |
| import 'lib1.dart' as one show topLevelFunction; |
| f(One o) { |
| one.topLevelFunction(); |
| } |
| '''); |
| } |
| |
| test_library_as_showTopLevelFunction_multipleDirectives() async { |
| newFile('$testPackageLibPath/lib1.dart', r''' |
| class One {} |
| topLevelFunction() {} |
| '''); |
| await assertNoErrorsInCode(r''' |
| import 'lib1.dart' hide topLevelFunction; |
| import 'lib1.dart' as one show topLevelFunction; |
| import 'lib1.dart' as two show topLevelFunction; |
| f(One o) { |
| one.topLevelFunction(); |
| two.topLevelFunction(); |
| } |
| '''); |
| } |
| |
| test_library_as_systemShadowing() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| class File {} |
| '''); |
| await assertNoErrorsInCode(''' |
| import 'dart:io' as io; |
| import 'lib1.dart' as io; |
| g(io.Directory d, io.File f) {} |
| '''); |
| } |
| |
| test_library_as_unnecessary() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| class A {} |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', ''' |
| export 'lib1.dart'; |
| class B {} |
| '''); |
| await assertErrorsInCode( |
| ''' |
| import 'lib1.dart' as p; |
| import 'lib2.dart' as p; |
| f(p.A a, p.B b) {} |
| ''', |
| [error(diag.unnecessaryImport, 7, 11)], |
| ); |
| } |
| |
| test_library_duplicateImport_differentPrefix() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| class A {} |
| class B {} |
| '''); |
| await assertNoErrorsInCode(''' |
| import 'lib1.dart'; |
| import 'lib1.dart' as p; |
| f(A a1, p.A a2, B b) {} |
| '''); |
| } |
| |
| @FailingTest(issue: 'https://github.com/dart-lang/sdk/issues/61877') |
| test_library_export_and_export() async { |
| var a = newFile('$testPackageLibPath/a.dart', r''' |
| class C {} |
| '''); |
| |
| var b = newFile('$testPackageLibPath/b.dart', r''' |
| export 'a.dart'; |
| '''); |
| |
| var c = newFile('$testPackageLibPath/c.dart', r''' |
| export 'a.dart'; |
| '''); |
| |
| var d = newFile('$testPackageLibPath/d.dart', r''' |
| import 'b.dart'; |
| import 'c.dart'; |
| |
| method() => C(); |
| '''); |
| await assertErrorsInFile2(a, []); |
| await assertErrorsInFile2(b, []); |
| await assertErrorsInFile2(c, []); |
| // Import of 'c.dart' is not marked as unused even though it could be |
| // removed. |
| var result = await resolveFile(d); |
| expect(result.diagnostics, isNotEmpty); |
| } |
| |
| test_library_extension_equalPrefixes_unnecessary() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| extension E1 on int { |
| void foo() {} |
| } |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', ''' |
| export 'lib1.dart'; |
| extension E2 on int { |
| void bar() {} |
| } |
| '''); |
| await assertErrorsInCode( |
| ''' |
| import 'lib1.dart' as prefix; |
| import 'lib2.dart' as prefix; |
| void f() { |
| 0.foo(); |
| 0.bar(); |
| } |
| ''', |
| [error(diag.unnecessaryImport, 7, 11)], |
| ); |
| } |
| |
| test_library_extension_noPrefixes_necessary() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| extension E1 on int { |
| void foo() {} |
| } |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', ''' |
| extension E2 on int { |
| void bar() {} |
| } |
| '''); |
| await assertNoErrorsInCode(''' |
| import 'lib1.dart'; |
| import 'lib2.dart'; |
| void f() { |
| 0.foo(); |
| 0.bar(); |
| } |
| '''); |
| } |
| |
| test_library_extension_noPrefixes_unnecessary() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| extension E1 on int { |
| void foo() {} |
| } |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', ''' |
| export 'lib1.dart'; |
| extension E2 on int { |
| void bar() {} |
| } |
| '''); |
| await assertErrorsInCode( |
| ''' |
| import 'lib1.dart'; |
| import 'lib2.dart'; |
| void f() { |
| 0.foo(); |
| 0.bar(); |
| } |
| ''', |
| [error(diag.unnecessaryImport, 7, 11)], |
| ); |
| } |
| |
| test_library_hasDeprecatedExport_hasNotDeprecatedImport_hasOtherClass() async { |
| newFile('$testPackageLibPath/a.dart', r''' |
| class A {} |
| '''); |
| |
| newFile('$testPackageLibPath/b.dart', r''' |
| library; |
| |
| @deprecated |
| export 'a.dart'; |
| |
| class B {} |
| '''); |
| |
| // `import b` is not reported because provides used `B`. |
| // `A` is from both `a.dart` and `b.dart`, so not reported. |
| await assertNoErrorsInCode(''' |
| import 'a.dart'; |
| import 'b.dart'; |
| |
| void f(A _, B _) {} |
| '''); |
| } |
| |
| test_library_hasDeprecatedExport_hasNotDeprecatedImport_noOtherClass() async { |
| newFile('$testPackageLibPath/a.dart', r''' |
| class A {} |
| '''); |
| |
| newFile('$testPackageLibPath/b.dart', r''' |
| class B {} |
| '''); |
| |
| newFile('$testPackageLibPath/c.dart', r''' |
| library; |
| |
| @deprecated |
| export 'b.dart'; |
| |
| class C {} |
| '''); |
| |
| // `import c` is unnecessary because we use only `B` from it. |
| // But the export of `B` from `c.dart` is deprecated. |
| // We can get `B` from `import b`, in a not deprecated way. |
| // It also declares `C`, but we don't use it. |
| await assertErrorsInCode( |
| ''' |
| import 'a.dart'; |
| import 'b.dart'; |
| import 'c.dart'; |
| |
| void f(A _, B _) {} |
| ''', |
| [error(diag.unnecessaryImport, 41, 8)], |
| ); |
| } |
| |
| test_library_hasDeprecatedExport_noNotDeprecatedImport() async { |
| newFile('$testPackageLibPath/a.dart', r''' |
| class A {} |
| '''); |
| |
| newFile('$testPackageLibPath/b.dart', r''' |
| class B {} |
| '''); |
| |
| newFile('$testPackageLibPath/c.dart', r''' |
| library; |
| |
| @deprecated |
| export 'b.dart'; |
| '''); |
| |
| // `import c` is not marked as unnecessary because of there is |
| // `DEPRECATED_EXPORT_USE` already reported. |
| await assertErrorsInCode( |
| ''' |
| import 'a.dart'; |
| import 'c.dart'; |
| |
| void f(A _, B _) {} |
| ''', |
| [error(diag.deprecatedExportUse, 47, 1)], |
| ); |
| } |
| |
| test_library_hide() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| class A {} |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', ''' |
| export 'lib1.dart' hide A; |
| class B {} |
| '''); |
| await assertNoErrorsInCode(''' |
| import 'lib1.dart'; |
| import 'lib2.dart'; |
| f(A a, B b) {} |
| '''); |
| } |
| |
| @FailingTest(issue: 'https://github.com/dart-lang/sdk/issues/61877') |
| test_library_import_and_export() async { |
| var a = newFile('$testPackageLibPath/a.dart', r''' |
| class C {} |
| '''); |
| |
| var b = newFile('$testPackageLibPath/b.dart', r''' |
| export 'a.dart'; |
| '''); |
| |
| var c = newFile('$testPackageLibPath/c.dart', r''' |
| import 'a.dart'; |
| import 'b.dart'; |
| |
| method() => C(); |
| '''); |
| await assertErrorsInFile2(a, []); |
| await assertErrorsInFile2(b, []); |
| // Import of 'b.dart' is not marked as unused even though it could be |
| // removed. |
| var result = await resolveFile(c); |
| expect(result.diagnostics, isNotEmpty); |
| } |
| |
| test_library_systemShadowing() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| class File {} |
| '''); |
| await assertNoErrorsInCode(''' |
| import 'dart:io'; |
| import 'lib1.dart'; |
| g(Directory d, File f) {} |
| '''); |
| } |
| |
| test_library_unnecessary_hasError() async { |
| newFile('$testPackageLibPath/a.dart', ''' |
| class A {} |
| '''); |
| |
| newFile('$testPackageLibPath/b.dart', ''' |
| export 'a.dart'; |
| class B {} |
| '''); |
| |
| await assertErrorsInCode( |
| ''' |
| import 'a.dart'; |
| import 'b.dart'; |
| void f(A _, B _, C _) {} |
| ''', |
| [error(diag.undefinedClass, 51, 1)], |
| ); |
| } |
| |
| test_library_unnecessaryImport() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| class A {} |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', ''' |
| export 'lib1.dart'; |
| class B {} |
| '''); |
| await assertErrorsInCode( |
| ''' |
| import 'lib1.dart'; |
| import 'lib2.dart'; |
| f(A a, B b) {} |
| ''', |
| [error(diag.unnecessaryImport, 7, 11)], |
| ); |
| } |
| |
| test_library_unnecessaryImport_sameUri() async { |
| newFile('$testPackageLibPath/lib1.dart', ''' |
| class A {} |
| '''); |
| newFile('$testPackageLibPath/lib2.dart', ''' |
| export 'lib1.dart'; |
| class B {} |
| '''); |
| await assertErrorsInCode( |
| ''' |
| import 'dart:async'; |
| import 'dart:async' show Completer; |
| f(FutureOr<int> a, Completer<int> b) {} |
| ''', |
| [error(diag.unnecessaryImport, 28, 12)], |
| ); |
| } |
| |
| test_library_uriDoesNotExist() async { |
| newFile('$testPackageLibPath/a.dart', ''' |
| class A {} |
| '''); |
| |
| await assertErrorsInCode( |
| ''' |
| import 'a.dart'; |
| import 'b.dart'; |
| void f(A _) {} |
| ''', |
| [error(diag.uriDoesNotExist, 24, 8)], |
| ); |
| } |
| |
| test_part_inside_unnecessary() async { |
| newFile('$testPackageLibPath/x.dart', ''' |
| class A {} |
| class B {} |
| '''); |
| |
| var a = newFile('$testPackageLibPath/a.dart', r''' |
| part 'b.dart'; |
| '''); |
| |
| var b = newFile('$testPackageLibPath/b.dart', r''' |
| part of 'a.dart'; |
| import 'x.dart' hide B; |
| import 'x.dart'; |
| void f(A _, B _) {} |
| '''); |
| |
| await assertErrorsInFile2(a, []); |
| |
| await assertErrorsInFile2(b, [error(diag.unnecessaryImport, 25, 8)]); |
| } |
| |
| test_part_inside_unnecessary_prefixed() async { |
| newFile('$testPackageLibPath/x.dart', ''' |
| class A {} |
| class B {} |
| '''); |
| |
| var a = newFile('$testPackageLibPath/a.dart', r''' |
| part 'b.dart'; |
| '''); |
| |
| var b = newFile('$testPackageLibPath/b.dart', r''' |
| part of 'a.dart'; |
| import 'x.dart' as prefix hide B; |
| import 'x.dart' as prefix; |
| void f(prefix.A _, prefix.B _) {} |
| '''); |
| |
| await assertErrorsInFile2(a, []); |
| |
| await assertErrorsInFile2(b, [error(diag.unnecessaryImport, 25, 8)]); |
| } |
| } |