blob: 178e261bb27ac8429c05bf89ddf59211bf41aaa3 [file] [log] [blame]
// 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({})});
''');
}
}