| // Copyright (c) 2019, 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:analysis_server/src/services/correction/fix.dart'; |
| import 'package:analysis_server/src/services/linter/lint_names.dart'; |
| import 'package:analyzer_plugin/utilities/fixes/fixes.dart'; |
| import 'package:test_reflective_loader/test_reflective_loader.dart'; |
| |
| import 'fix_processor.dart'; |
| |
| void main() { |
| defineReflectiveSuite(() { |
| defineReflectiveTests(AddConst_InvalidCasePatternsTest); |
| defineReflectiveTests(AddConst_InvalidCasePatternsBulkTest); |
| defineReflectiveTests(AddConst_NonConstGenerativeEnumConstructorTest); |
| defineReflectiveTests(AddConst_PatternExpressionMustBeValidConst); |
| defineReflectiveTests(AddConst_PreferConstConstructorsInImmutablesBulkTest); |
| defineReflectiveTests(AddConst_PreferConstConstructorsInImmutablesTest); |
| defineReflectiveTests(AddConst_PreferConstConstructorsBulkTest); |
| defineReflectiveTests(AddConst_PreferConstConstructorsTest); |
| defineReflectiveTests( |
| AddConst_PreferConstLiteralsToCreateImmutablesBulkTest); |
| defineReflectiveTests(AddConst_PreferConstLiteralsToCreateImmutablesTest); |
| }); |
| } |
| |
| @reflectiveTest |
| class AddConst_InvalidCasePatternsBulkTest extends BulkFixProcessorTest { |
| @override |
| String get lintCode => LintNames.invalid_case_patterns; |
| |
| Future<void> test_singleFile() async { |
| await resolveTestCode(r''' |
| //@dart=2.19 |
| class C { |
| const C(); |
| } |
| |
| void f(Object c) { |
| switch (c) { |
| case C(): |
| case [1, 2]: |
| } |
| } |
| '''); |
| await assertHasFix(r''' |
| //@dart=2.19 |
| class C { |
| const C(); |
| } |
| |
| void f(Object c) { |
| switch (c) { |
| case const C(): |
| case const [1, 2]: |
| } |
| } |
| '''); |
| } |
| } |
| |
| @reflectiveTest |
| class AddConst_InvalidCasePatternsTest extends FixProcessorLintTest { |
| @override |
| FixKind get kind => DartFixKind.ADD_CONST; |
| |
| @override |
| String get lintCode => LintNames.invalid_case_patterns; |
| |
| Future<void> test_constructorCall() async { |
| await resolveTestCode(r''' |
| //@dart=2.19 |
| class C { |
| const C(); |
| } |
| |
| void f(Object c) { |
| switch (c) { |
| case C(): |
| } |
| } |
| '''); |
| await assertHasFix(r''' |
| //@dart=2.19 |
| class C { |
| const C(); |
| } |
| |
| void f(Object c) { |
| switch (c) { |
| case const C(): |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_listLiteral() async { |
| await resolveTestCode(r''' |
| //@dart=2.19 |
| void f(Object o) { |
| switch (o) { |
| case [1, 2]: |
| } |
| } |
| '''); |
| await assertHasFix(r''' |
| //@dart=2.19 |
| void f(Object o) { |
| switch (o) { |
| case const [1, 2]: |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_mapLiteral() async { |
| await resolveTestCode(r''' |
| //@dart=2.19 |
| void f(Object o) { |
| switch (o) { |
| case {'k': 'v'}: |
| } |
| } |
| '''); |
| await assertHasFix(r''' |
| //@dart=2.19 |
| void f(Object o) { |
| switch (o) { |
| case const {'k': 'v'}: |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_setLiteral() async { |
| await resolveTestCode(r''' |
| //@dart=2.19 |
| void f(Object o) { |
| switch (o) { |
| case {1}: |
| } |
| } |
| '''); |
| await assertHasFix(r''' |
| //@dart=2.19 |
| void f(Object o) { |
| switch (o) { |
| case const {1}: |
| } |
| } |
| '''); |
| } |
| } |
| |
| @reflectiveTest |
| class AddConst_NonConstGenerativeEnumConstructorTest extends FixProcessorTest { |
| @override |
| FixKind get kind => DartFixKind.ADD_CONST; |
| |
| Future<void> test_named() async { |
| await resolveTestCode(''' |
| enum E { |
| v.named(); |
| E.named(); |
| } |
| '''); |
| await assertHasFix(''' |
| enum E { |
| v.named(); |
| const E.named(); |
| } |
| '''); |
| } |
| |
| Future<void> test_unnamed() async { |
| await resolveTestCode(''' |
| enum E { |
| v; |
| E(); |
| } |
| '''); |
| await assertHasFix(''' |
| enum E { |
| v; |
| const E(); |
| } |
| '''); |
| } |
| } |
| |
| @reflectiveTest |
| class AddConst_PatternExpressionMustBeValidConst extends FixProcessorTest { |
| @override |
| FixKind get kind => DartFixKind.ADD_CONST; |
| |
| Future<void> test_caseBinaryExpression() async { |
| await resolveTestCode(''' |
| void f() { |
| var m = 5; |
| switch(m) { |
| case (5 * 5): break; |
| } |
| } |
| '''); |
| |
| await assertHasFix(''' |
| void f() { |
| var m = 5; |
| switch(m) { |
| case const (5 * 5): break; |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_caseBinaryExpressionNoParen() async { |
| await resolveTestCode(''' |
| void f() { |
| var m = 5; |
| switch(m) { |
| case 5 * 5: break; |
| } |
| } |
| '''); |
| |
| await assertHasFix(''' |
| void f() { |
| var m = 5; |
| switch(m) { |
| case const (5 * 5): break; |
| } |
| } |
| '''); |
| } |
| |
| @FailingTest(issue: 'https://github.com/dart-lang/sdk/issues/51139') |
| Future<void> test_caseConstConstructorCall() async { |
| await resolveTestCode(''' |
| class A { |
| final int a; |
| const A(this.a); |
| } |
| void f() { |
| var m = 5; |
| switch(m) { |
| case A(1): break; |
| } |
| } |
| '''); |
| |
| await assertHasFix(''' |
| class A { |
| final int a; |
| const A(this.a); |
| } |
| void f() { |
| var m = 5; |
| switch(m) { |
| case const A(1): break; |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_caseListExpression() async { |
| await resolveTestCode(''' |
| class A {} |
| void f(Object m) { |
| switch(m) { |
| case List<A>: break; |
| } |
| } |
| '''); |
| |
| await assertHasFix(''' |
| class A {} |
| void f(Object m) { |
| switch(m) { |
| case const (List<A>): break; |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_caseListExpression_with_parens() async { |
| await resolveTestCode(''' |
| class A {} |
| void f(Object m) { |
| switch(m) { |
| case (List<A>): break; |
| } |
| } |
| '''); |
| |
| await assertHasFix(''' |
| class A {} |
| void f(Object m) { |
| switch(m) { |
| case const (List<A>): break; |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_casePrefixExpression() async { |
| await resolveTestCode(''' |
| void f(Object? x) { |
| const m = 5; |
| switch(x) { |
| case (-m): break; |
| } |
| } |
| '''); |
| |
| await assertHasFix(''' |
| void f(Object? x) { |
| const m = 5; |
| switch(x) { |
| case const (-m): break; |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_casePrefixExpressionNoParen() async { |
| await resolveTestCode(''' |
| void f(Object? x) { |
| const m = 5; |
| switch(x) { |
| case -m: break; |
| } |
| } |
| '''); |
| |
| await assertHasFix(''' |
| void f(Object? x) { |
| const m = 5; |
| switch(x) { |
| case const (-m): break; |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_caseWithField() async { |
| await resolveTestCode(''' |
| int x = 1; |
| void f() { |
| var m = 5; |
| switch(m) { |
| case x: break; |
| } |
| } |
| '''); |
| |
| await assertNoFix(); |
| } |
| |
| @FailingTest(reason: 'TODO(keertip): Add support for local variables') |
| Future<void> test_caseWithLocalVariable() async { |
| await resolveTestCode(''' |
| void f() { |
| var m = 5; |
| var x = 1; |
| switch(m) { |
| case x: break; |
| } |
| } |
| '''); |
| |
| await assertHasFix(''' |
| void f() { |
| var m = 5; |
| const x = 1; |
| switch(m) { |
| case x: break; |
| } |
| } |
| '''); |
| } |
| |
| Future<void> test_mapKeyConst() async { |
| await resolveTestCode(''' |
| void f(Object x) { |
| if (x case {A(): 0}) {} |
| } |
| |
| class A { |
| const A(); |
| } |
| '''); |
| |
| await assertHasFix(''' |
| void f(Object x) { |
| if (x case {const A(): 0}) {} |
| } |
| |
| class A { |
| const A(); |
| } |
| '''); |
| } |
| |
| @FailingTest(reason: 'TODO(keertip): Add support for local variables') |
| Future<void> test_relationalExpressionConst() async { |
| await resolveTestCode(''' |
| void f(int x) { |
| final a = 0; |
| if (x case > a) {} |
| } |
| '''); |
| |
| await assertHasFix(''' |
| void f(int x) { |
| const a = 0; |
| if (x case > a) {} |
| } |
| '''); |
| } |
| } |
| |
| @reflectiveTest |
| class AddConst_PreferConstConstructorsBulkTest extends BulkFixProcessorTest { |
| @override |
| String get lintCode => LintNames.prefer_const_constructors; |
| |
| Future<void> test_noKeyword() async { |
| writeTestPackageConfig(meta: true); |
| await resolveTestCode(r''' |
| class C { |
| const C([C? c]); |
| } |
| var c = C(C()); |
| '''); |
| await assertHasFix(r''' |
| class C { |
| const C([C? c]); |
| } |
| var c = const C(C()); |
| '''); |
| } |
| } |
| |
| @reflectiveTest |
| class AddConst_PreferConstConstructorsInImmutablesBulkTest |
| extends BulkFixProcessorTest { |
| @override |
| String get lintCode => LintNames.prefer_const_constructors_in_immutables; |
| |
| Future<void> test_multipleConstructors() async { |
| writeTestPackageConfig(meta: true); |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class A { |
| A(); |
| /// Comment. |
| A.a(); |
| } |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class A { |
| const A(); |
| /// Comment. |
| const A.a(); |
| } |
| '''); |
| } |
| } |
| |
| @reflectiveTest |
| class AddConst_PreferConstConstructorsInImmutablesTest |
| extends FixProcessorLintTest { |
| @override |
| FixKind get kind => DartFixKind.ADD_CONST; |
| |
| @override |
| String get lintCode => LintNames.prefer_const_constructors_in_immutables; |
| |
| @override |
| void setUp() { |
| super.setUp(); |
| writeTestPackageConfig( |
| meta: true, |
| ); |
| } |
| |
| Future<void> test_default() async { |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class A { |
| A(); |
| } |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class A { |
| const A(); |
| } |
| '''); |
| } |
| |
| Future<void> test_default_withComment() async { |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class A { |
| /// Comment. |
| A(); |
| } |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class A { |
| /// Comment. |
| const A(); |
| } |
| '''); |
| } |
| } |
| |
| @reflectiveTest |
| class AddConst_PreferConstConstructorsTest extends FixProcessorLintTest { |
| @override |
| FixKind get kind => DartFixKind.ADD_CONST; |
| |
| @override |
| String get lintCode => LintNames.prefer_const_constructors; |
| |
| Future<void> test_new() async { |
| await resolveTestCode(''' |
| class C { |
| const C(); |
| } |
| void f() { |
| var c = new C(); |
| print(c); |
| } |
| '''); |
| // handled by REPLACE_NEW_WITH_CONST |
| await assertNoFix(); |
| } |
| |
| Future<void> test_noKeyword() async { |
| await resolveTestCode(''' |
| class C { |
| const C(); |
| } |
| void f() { |
| var c = C(); |
| print(c); |
| } |
| '''); |
| await assertHasFix(''' |
| class C { |
| const C(); |
| } |
| void f() { |
| var c = const C(); |
| print(c); |
| } |
| '''); |
| } |
| |
| Future<void> test_withConstList() async { |
| await resolveTestCode(''' |
| class C { |
| const C(List<int> l); |
| } |
| void f() { |
| var c = C(const <int>[]); |
| print(c); |
| } |
| '''); |
| await assertHasFix(''' |
| class C { |
| const C(List<int> l); |
| } |
| void f() { |
| var c = const C(<int>[]); |
| print(c); |
| } |
| '''); |
| } |
| } |
| |
| @reflectiveTest |
| class AddConst_PreferConstLiteralsToCreateImmutablesBulkTest |
| extends BulkFixProcessorTest { |
| @override |
| String get lintCode => LintNames.prefer_const_literals_to_create_immutables; |
| |
| @override |
| void setUp() { |
| super.setUp(); |
| writeTestPackageConfig( |
| meta: true, |
| ); |
| } |
| |
| Future<void> test_map() async { |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| final Map children; |
| const C({required this.children}); |
| } |
| void f() { |
| var c = C(children: { |
| 1 : {} |
| }); |
| print(c); |
| } |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| final Map children; |
| const C({required this.children}); |
| } |
| void f() { |
| var c = C(children: const { |
| 1 : const {} |
| }); |
| print(c); |
| } |
| '''); |
| } |
| } |
| |
| @reflectiveTest |
| class AddConst_PreferConstLiteralsToCreateImmutablesTest |
| extends FixProcessorLintTest { |
| @override |
| FixKind get kind => DartFixKind.ADD_CONST; |
| |
| @override |
| String get lintCode => LintNames.prefer_const_literals_to_create_immutables; |
| @override |
| void setUp() { |
| super.setUp(); |
| writeTestPackageConfig( |
| meta: true, |
| ); |
| } |
| |
| Future<void> test_list() async { |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| final List<C> children; |
| const C({required this.children}); |
| } |
| void f() { |
| var c = C(children: []); |
| print(c); |
| } |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| final List<C> children; |
| const C({required this.children}); |
| } |
| void f() { |
| var c = C(children: const []); |
| print(c); |
| } |
| '''); |
| } |
| |
| Future<void> test_list_list() async { |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| const C(List<C> children); |
| } |
| C c = C([C(const [])]); |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| const C(List<C> children); |
| } |
| C c = C(const [C([])]); |
| '''); |
| } |
| |
| Future<void> test_map() async { |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| final Map children; |
| const C({required this.children}); |
| } |
| void f() { |
| var c = C(children: {}); |
| print(c); |
| } |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| final Map children; |
| const C({required this.children}); |
| } |
| void f() { |
| var c = C(children: const {}); |
| print(c); |
| } |
| '''); |
| } |
| |
| Future<void> test_map_map() async { |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| const C(Map<String, C> children); |
| } |
| C c = C({'c': C(const {})}); |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| const C(Map<String, C> children); |
| } |
| C c = C(const {'c': C({})}); |
| '''); |
| } |
| |
| Future<void> test_set() async { |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| final Set<C> children; |
| const C({required this.children}); |
| } |
| void f() { |
| var c = C(children: {}); |
| print(c); |
| } |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| final Set<C> children; |
| const C({required this.children}); |
| } |
| void f() { |
| var c = C(children: const {}); |
| print(c); |
| } |
| '''); |
| } |
| |
| Future<void> test_set_set() async { |
| await resolveTestCode(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| const C(Set<C> children); |
| } |
| C c = C({C(const {})}); |
| '''); |
| await assertHasFix(''' |
| import 'package:meta/meta.dart'; |
| |
| @immutable |
| class C { |
| const C(Set<C> children); |
| } |
| C c = C(const {C({})}); |
| '''); |
| } |
| } |