blob: 49d54a84247c4cb60115780d60c857137293c4d1 [file] [log] [blame]
// Copyright (c) 2018, 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 'dart:async';
import 'package:analyzer/dart/analysis/results.dart';
import 'package:analyzer/dart/ast/ast.dart';
import 'package:analyzer/dart/ast/token.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/dart/element/type.dart';
import 'package:analyzer/src/dart/analysis/driver.dart';
import 'package:analyzer/src/dart/analysis/experiments.dart';
import 'package:analyzer/src/dart/element/element.dart';
import 'package:analyzer/src/dart/element/member.dart';
import 'package:analyzer/src/dart/element/type.dart';
import 'package:analyzer/src/generated/engine.dart';
import 'package:analyzer/src/generated/resolver.dart';
import 'package:analyzer/src/generated/utilities_dart.dart';
import 'package:analyzer/src/test_utilities/find_element.dart';
import 'package:analyzer/src/test_utilities/find_node.dart';
import 'package:test/test.dart';
import 'package:test_reflective_loader/test_reflective_loader.dart';
import '../../../utils.dart';
import 'base.dart';
main() {
defineReflectiveSuite(() {
defineReflectiveTests(AnalysisDriverResolutionTest);
defineReflectiveTests(DriverResolutionWithExperimentsTest);
});
}
final isBottomType = new TypeMatcher<BottomTypeImpl>();
final isDynamicType = new TypeMatcher<DynamicTypeImpl>();
final isVoidType = new TypeMatcher<VoidTypeImpl>();
/**
* Integration tests for resolution.
*/
@reflectiveTest
class AnalysisDriverResolutionTest extends BaseAnalysisDriverTest {
ResolvedUnitResult result;
FindNode findNode;
FindElement findElement;
ClassElement get boolElement => typeProvider.boolType.element;
ClassElement get doubleElement => typeProvider.doubleType.element;
InterfaceType get doubleType => typeProvider.doubleType;
DynamicTypeImpl get dynamicType => DynamicTypeImpl.instance;
ClassElement get futureElement => typeProvider.futureType.element;
ClassElement get intElement => typeProvider.intType.element;
InterfaceType get intType => typeProvider.intType;
ClassElement get listElement => typeProvider.listType.element;
ClassElement get mapElement => typeProvider.mapType.element;
InterfaceType get mapType => typeProvider.mapType;
ClassElement get numElement => typeProvider.numType.element;
InterfaceType get numType => typeProvider.numType;
ClassElement get stringElement => typeProvider.stringType.element;
TypeProvider get typeProvider =>
result.unit.declaredElement.context.typeProvider;
void assertElement(AstNode node, Element expected) {
Element actual = getNodeElement(node);
expect(actual, same(expected));
}
void assertElementNull(Expression node) {
Element actual = getNodeElement(node);
expect(actual, isNull);
}
void assertIdentifierTopGetRef(SimpleIdentifier ref, String name) {
var getter = findElement.topGet(name);
assertElement(ref, getter);
var type = getter.returnType.toString();
assertType(ref, type);
}
void assertInvokeType(InvocationExpression expression, String expected) {
DartType actual = expression.staticInvokeType;
expect(actual?.toString(), expected);
}
void assertMember(
Expression node, String expectedDefiningType, Element expectedBase) {
Member actual = getNodeElement(node);
expect(actual.definingType.toString(), expectedDefiningType);
expect(actual.baseElement, same(expectedBase));
}
void assertTopGetRef(String search, String name) {
var ref = findNode.simple(search);
assertIdentifierTopGetRef(ref, name);
}
void assertType(AstNode node, String expected) {
DartType actual;
if (node is Expression) {
actual = node.staticType;
} else if (node is GenericFunctionType) {
actual = node.type;
} else if (node is TypeName) {
actual = node.type;
} else {
fail('Unsupported node: (${node.runtimeType}) $node');
}
expect(actual?.toString(), expected);
}
/// Test that [argumentList] has exactly two type items `int` and `double`.
void assertTypeArguments(
TypeArgumentList argumentList, List<DartType> expectedTypes) {
expect(argumentList.arguments, hasLength(expectedTypes.length));
for (int i = 0; i < expectedTypes.length; i++) {
_assertTypeNameSimple(argumentList.arguments[i], expectedTypes[i]);
}
}
void assertTypeDynamic(Expression expression) {
DartType actual = expression.staticType;
expect(actual, isDynamicType);
}
void assertTypeName(
TypeName node, Element expectedElement, String expectedType,
{PrefixElement expectedPrefix}) {
assertType(node, expectedType);
if (expectedPrefix == null) {
var name = node.name as SimpleIdentifier;
assertElement(name, expectedElement);
assertType(name, expectedType);
} else {
var name = node.name as PrefixedIdentifier;
assertElement(name.prefix, expectedPrefix);
expect(name.prefix.staticType, isNull);
assertElement(name.identifier, expectedElement);
expect(name.identifier.staticType, isNull);
}
}
void assertTypeNull(Identifier node) {
expect(node.staticType, isNull);
}
void assertUnresolvedInvokeType(DartType invokeType) {
expect(invokeType, isDynamicType);
}
/// Creates a function that checks that an expression is a reference to a top
/// level variable with the given [name].
void Function(Expression) checkTopVarRef(String name) {
return (Expression e) {
TopLevelVariableElement variable = _getTopLevelVariable(result, name);
SimpleIdentifier node = e as SimpleIdentifier;
expect(node.staticElement, same(variable.getter));
expect(node.staticType, variable.type);
};
}
/// Creates a function that checks that an expression is a named argument
/// that references a top level variable with the given [name], where the
/// name of the named argument is undefined.
void Function(Expression) checkTopVarUndefinedNamedRef(String name) {
return (Expression e) {
TopLevelVariableElement variable = _getTopLevelVariable(result, name);
NamedExpression named = e as NamedExpression;
expect(named.staticType, variable.type);
SimpleIdentifier nameIdentifier = named.name.label;
expect(nameIdentifier.staticElement, isNull);
expect(nameIdentifier.staticType, isNull);
SimpleIdentifier expression = named.expression;
expect(expression.staticElement, same(variable.getter));
expect(expression.staticType, variable.type);
};
}
Element getNodeElement(Expression node) {
if (node is AssignmentExpression) {
return node.staticElement;
} else if (node is Identifier) {
return node.staticElement;
} else if (node is IndexExpression) {
return node.staticElement;
} else if (node is InstanceCreationExpression) {
return node.staticElement;
} else if (node is PostfixExpression) {
return node.staticElement;
} else if (node is PrefixExpression) {
return node.staticElement;
} else {
fail('Unsupported node: (${node.runtimeType}) $node');
}
}
Future resolveTestFile() async {
result = await driver.getResult(testFile);
findNode = new FindNode(result.content, result.unit);
findElement = new FindElement(result.unit);
}
test_adjacentStrings() async {
String content = r'''
void main() {
'aaa' 'bbb' 'ccc';
}
''';
addTestFile(content);
await resolveTestFile();
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
AdjacentStrings expression = statement.expression;
expect(expression.staticType, typeProvider.stringType);
expect(expression.strings, hasLength(3));
StringLiteral literal_1 = expression.strings[0];
expect(literal_1.staticType, typeProvider.stringType);
StringLiteral literal_2 = expression.strings[1];
expect(literal_2.staticType, typeProvider.stringType);
StringLiteral literal_3 = expression.strings[2];
expect(literal_3.staticType, typeProvider.stringType);
}
test_annotation() async {
String content = r'''
const myAnnotation = 1;
@myAnnotation
class C {
@myAnnotation
int field1 = 2, field2 = 3;
@myAnnotation
C() {}
@myAnnotation
void method() {}
}
@myAnnotation
int topLevelVariable1 = 4, topLevelVariable2 = 5;
@myAnnotation
void topLevelFunction() {}
''';
addTestFile(content);
await resolveTestFile();
TopLevelVariableDeclaration myDeclaration = result.unit.declarations[0];
VariableDeclaration myVariable = myDeclaration.variables.variables[0];
TopLevelVariableElement myElement = myVariable.declaredElement;
void assertMyAnnotation(AnnotatedNode node) {
Annotation annotation = node.metadata[0];
expect(annotation.element, same(myElement.getter));
SimpleIdentifier identifier_1 = annotation.name;
expect(identifier_1.staticElement, same(myElement.getter));
expect(identifier_1.staticType, typeProvider.intType);
}
{
ClassDeclaration classNode = result.unit.declarations[1];
assertMyAnnotation(classNode);
{
FieldDeclaration node = classNode.members[0];
assertMyAnnotation(node);
}
{
ConstructorDeclaration node = classNode.members[1];
assertMyAnnotation(node);
}
{
MethodDeclaration node = classNode.members[2];
assertMyAnnotation(node);
}
}
{
TopLevelVariableDeclaration node = result.unit.declarations[2];
assertMyAnnotation(node);
}
{
FunctionDeclaration node = result.unit.declarations[3];
assertMyAnnotation(node);
}
}
test_annotation_onDirective_export() async {
addTestFile(r'''
@a
export 'dart:math';
const a = 1;
''');
await resolveTestFile();
var directive = findNode.export('dart:math');
expect(directive.metadata, hasLength(1));
Annotation annotation = directive.metadata[0];
expect(annotation.element, findElement.topGet('a'));
SimpleIdentifier aRef = annotation.name;
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_annotation_onDirective_import() async {
addTestFile(r'''
@a
import 'dart:math';
const a = 1;
''');
await resolveTestFile();
var directive = findNode.import('dart:math');
expect(directive.metadata, hasLength(1));
Annotation annotation = directive.metadata[0];
expect(annotation.element, findElement.topGet('a'));
SimpleIdentifier aRef = annotation.name;
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_annotation_onDirective_library() async {
addTestFile(r'''
@a
library test;
const a = 1;
''');
await resolveTestFile();
var directive = findNode.libraryDirective;
expect(directive.metadata, hasLength(1));
Annotation annotation = directive.metadata[0];
expect(annotation.element, findElement.topGet('a'));
SimpleIdentifier aRef = annotation.name;
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_annotation_onDirective_part() async {
newFile('/test/lib/a.dart', content: r'''
part of 'test.dart';
''');
addTestFile(r'''
@a
part 'a.dart';
const a = 1;
''');
await resolveTestFile();
var directive = findNode.part('a.dart');
expect(directive.metadata, hasLength(1));
Annotation annotation = directive.metadata[0];
expect(annotation.element, findElement.topGet('a'));
SimpleIdentifier aRef = annotation.name;
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_annotation_onDirective_partOf() async {
newFile('/test/lib/a.dart', content: r'''
part 'test.dart';
''');
addTestFile(r'''
@a
part of 'a.dart';
const a = 1;
''');
await resolveTestFile();
var directive = findNode.partOf('a.dart');
expect(directive.metadata, hasLength(1));
Annotation annotation = directive.metadata[0];
expect(annotation.element, findElement.topGet('a'));
SimpleIdentifier aRef = annotation.name;
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_annotation_onFormalParameter_redirectingFactory() async {
addTestFile(r'''
class C {
factory C(@a @b x, y, {@c z}) = C.named;
C.named(int p);
}
const a = 1;
const b = 2;
const c = 2;
''');
await resolveTestFile();
void assertTopGetAnnotation(Annotation annotation, String name) {
var getter = findElement.topGet(name);
expect(annotation.element, getter);
SimpleIdentifier ref = annotation.name;
assertElement(ref, getter);
assertType(ref, 'int');
}
{
var parameter = findNode.simpleParameter('x, ');
expect(parameter.metadata, hasLength(2));
assertTopGetAnnotation(parameter.metadata[0], 'a');
assertTopGetAnnotation(parameter.metadata[1], 'b');
}
{
var parameter = findNode.simpleParameter('z}');
expect(parameter.metadata, hasLength(1));
assertTopGetAnnotation(parameter.metadata[0], 'c');
}
}
test_annotation_onVariableList_constructor() async {
String content = r'''
class C {
final Object x;
const C(this.x);
}
main() {
@C(C(42))
var foo = null;
}
''';
addTestFile(content);
await resolveTestFile();
ClassDeclaration c = result.unit.declarations[0];
ConstructorDeclaration constructor = c.members[1];
ConstructorElement element = constructor.declaredElement;
FunctionDeclaration main = result.unit.declarations[1];
VariableDeclarationStatement statement =
(main.functionExpression.body as BlockFunctionBody).block.statements[0];
Annotation annotation = statement.variables.metadata[0];
expect(annotation.element, same(element));
SimpleIdentifier identifier_1 = annotation.name;
expect(identifier_1.staticElement, same(c.declaredElement));
}
test_annotation_onVariableList_topLevelVariable() async {
String content = r'''
const myAnnotation = 1;
class C {
void method() {
@myAnnotation
int var1 = 4, var2 = 5;
}
}
''';
addTestFile(content);
await resolveTestFile();
TopLevelVariableDeclaration myDeclaration = result.unit.declarations[0];
VariableDeclaration myVariable = myDeclaration.variables.variables[0];
TopLevelVariableElement myElement = myVariable.declaredElement;
ClassDeclaration classNode = result.unit.declarations[1];
MethodDeclaration node = classNode.members[0];
VariableDeclarationStatement statement =
(node.body as BlockFunctionBody).block.statements[0];
Annotation annotation = statement.variables.metadata[0];
expect(annotation.element, same(myElement.getter));
SimpleIdentifier identifier_1 = annotation.name;
expect(identifier_1.staticElement, same(myElement.getter));
expect(identifier_1.staticType, typeProvider.intType);
}
test_annotation_prefixed_classField() async {
newFile('/test/lib/a.dart', content: r'''
class A {
static const a = 1;
}
''');
addTestFile(r'''
import 'a.dart' as p;
@p.A.a
main() {}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
ImportElement aImport = unit.declaredElement.library.imports[0];
PrefixElement aPrefix = aImport.prefix;
LibraryElement aLibrary = aImport.importedLibrary;
CompilationUnitElement aUnitElement = aLibrary.definingCompilationUnit;
ClassElement aClass = aUnitElement.getType('A');
var aGetter = aClass.getField('a').getter;
Annotation annotation = unit.declarations[0].metadata.single;
expect(annotation.element, same(aGetter));
PrefixedIdentifier prefixed = annotation.name;
expect(prefixed.prefix.staticElement, same(aPrefix));
expect(prefixed.prefix.staticType, isNull);
expect(prefixed.identifier.staticElement, same(aClass));
expect(prefixed.prefix.staticType, isNull);
expect(annotation.constructorName.staticElement, aGetter);
expect(annotation.constructorName.staticType, typeProvider.intType);
expect(annotation.arguments, isNull);
}
test_annotation_prefixed_constructor() async {
newFile('/test/lib/a.dart', content: r'''
class A {
const A(int a, {int b});
}
''');
addTestFile(r'''
import 'a.dart' as p;
@p.A(1, b: 2)
main() {}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
ImportElement aImport = unit.declaredElement.library.imports[0];
PrefixElement aPrefix = aImport.prefix;
LibraryElement aLibrary = aImport.importedLibrary;
CompilationUnitElement aUnitElement = aLibrary.definingCompilationUnit;
ClassElement aClass = aUnitElement.getType('A');
ConstructorElement constructor = aClass.unnamedConstructor;
Annotation annotation = unit.declarations[0].metadata.single;
expect(annotation.element, same(constructor));
PrefixedIdentifier prefixed = annotation.name;
expect(prefixed.prefix.staticElement, same(aPrefix));
expect(prefixed.prefix.staticType, isNull);
expect(prefixed.identifier.staticElement, same(aClass));
expect(prefixed.prefix.staticType, isNull);
expect(annotation.constructorName, isNull);
var arguments = annotation.arguments.arguments;
var parameters = constructor.parameters;
_assertArgumentToParameter(arguments[0], parameters[0]);
_assertArgumentToParameter(arguments[1], parameters[1]);
}
test_annotation_prefixed_constructor_named() async {
newFile('/test/lib/a.dart', content: r'''
class A {
const A.named(int a, {int b});
}
''');
addTestFile(r'''
import 'a.dart' as p;
@p.A.named(1, b: 2)
main() {}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
ImportElement aImport = unit.declaredElement.library.imports[0];
PrefixElement aPrefix = aImport.prefix;
LibraryElement aLibrary = aImport.importedLibrary;
CompilationUnitElement aUnitElement = aLibrary.definingCompilationUnit;
ClassElement aClass = aUnitElement.getType('A');
ConstructorElement constructor = aClass.getNamedConstructor('named');
Annotation annotation = unit.declarations[0].metadata.single;
expect(annotation.element, same(constructor));
PrefixedIdentifier prefixed = annotation.name;
expect(prefixed.prefix.staticElement, same(aPrefix));
expect(prefixed.prefix.staticType, isNull);
expect(prefixed.identifier.staticElement, same(aClass));
expect(prefixed.prefix.staticType, isNull);
SimpleIdentifier constructorName = annotation.constructorName;
expect(constructorName.staticElement, same(constructor));
expect(constructorName.staticType.toString(), '(int, {b: int}) → A');
var arguments = annotation.arguments.arguments;
var parameters = constructor.parameters;
_assertArgumentToParameter(arguments[0], parameters[0]);
_assertArgumentToParameter(arguments[1], parameters[1]);
}
test_annotation_prefixed_topLevelVariable() async {
newFile('/test/lib/a.dart', content: r'''
const topAnnotation = 1;
''');
addTestFile(r'''
import 'a.dart' as p;
@p.topAnnotation
main() {}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
ImportElement aImport = unit.declaredElement.library.imports[0];
PrefixElement aPrefix = aImport.prefix;
LibraryElement aLibrary = aImport.importedLibrary;
CompilationUnitElement aUnitElement = aLibrary.definingCompilationUnit;
var topAnnotation = aUnitElement.topLevelVariables[0].getter;
Annotation annotation = unit.declarations[0].metadata.single;
expect(annotation.element, same(topAnnotation));
PrefixedIdentifier prefixed = annotation.name;
expect(prefixed.prefix.staticElement, same(aPrefix));
expect(prefixed.prefix.staticType, isNull);
expect(prefixed.identifier.staticElement, same(topAnnotation));
expect(prefixed.prefix.staticType, isNull);
expect(annotation.constructorName, isNull);
expect(annotation.arguments, isNull);
}
test_annotation_unprefixed_classField() async {
addTestFile(r'''
@A.a
main() {}
class A {
static const a = 1;
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
var typeProvider = unitElement.context.typeProvider;
ClassElement aClass = unitElement.getType('A');
var aGetter = aClass.getField('a').getter;
Annotation annotation = unit.declarations[0].metadata.single;
expect(annotation.element, same(aGetter));
PrefixedIdentifier prefixed = annotation.name;
expect(prefixed.prefix.staticElement, same(aClass));
expect(prefixed.prefix.staticType, aClass.type);
expect(prefixed.identifier.staticElement, same(aGetter));
expect(prefixed.identifier.staticType, typeProvider.intType);
expect(annotation.constructorName, isNull);
expect(annotation.arguments, isNull);
}
test_annotation_unprefixed_constructor() async {
addTestFile(r'''
@A(1, b: 2)
main() {}
class A {
const A(int a, {int b});
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
ClassElement aClass = unitElement.getType('A');
ConstructorElement constructor = aClass.unnamedConstructor;
Annotation annotation = unit.declarations[0].metadata.single;
expect(annotation.element, same(constructor));
SimpleIdentifier name = annotation.name;
expect(name.staticElement, same(aClass));
expect(annotation.constructorName, isNull);
var arguments = annotation.arguments.arguments;
var parameters = constructor.parameters;
_assertArgumentToParameter(arguments[0], parameters[0]);
_assertArgumentToParameter(arguments[1], parameters[1]);
}
test_annotation_unprefixed_constructor_named() async {
addTestFile(r'''
@A.named(1, b: 2)
main() {}
class A {
const A.named(int a, {int b});
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
ClassElement aClass = unitElement.getType('A');
ConstructorElement constructor = aClass.constructors.single;
Annotation annotation = unit.declarations[0].metadata.single;
expect(annotation.element, same(constructor));
PrefixedIdentifier prefixed = annotation.name;
expect(prefixed.prefix.staticElement, same(aClass));
expect(prefixed.prefix.staticType, aClass.type);
expect(prefixed.identifier.staticElement, same(constructor));
expect(prefixed.identifier.staticType.toString(), '(int, {b: int}) → A');
expect(annotation.constructorName, isNull);
var arguments = annotation.arguments.arguments;
var parameters = constructor.parameters;
_assertArgumentToParameter(arguments[0], parameters[0]);
_assertArgumentToParameter(arguments[1], parameters[1]);
}
test_annotation_unprefixed_constructor_withNestedConstructorInvocation() async {
addTestFile('''
class C {
const C();
}
class D {
final C c;
const D(this.c);
}
@D(const C())
f() {}
''');
await resolveTestFile();
var elementC = AstFinder.getClass(result.unit, 'C').declaredElement;
var constructorC = elementC.constructors[0];
var elementD = AstFinder.getClass(result.unit, 'D').declaredElement;
var constructorD = elementD.constructors[0];
var atD = AstFinder.getTopLevelFunction(result.unit, 'f').metadata[0];
InstanceCreationExpression constC = atD.arguments.arguments[0];
expect(atD.name.staticElement, elementD);
expect(atD.element, constructorD);
expect(constC.staticElement, constructorC);
expect(constC.staticType, elementC.type);
expect(constC.constructorName.staticElement, constructorC);
expect(constC.constructorName.type.type, elementC.type);
}
test_annotation_unprefixed_topLevelVariable() async {
String content = r'''
const annotation_1 = 1;
const annotation_2 = 1;
@annotation_1
@annotation_2
void main() {
print(42);
}
''';
addTestFile(content);
await resolveTestFile();
TopLevelVariableDeclaration declaration_1 = result.unit.declarations[0];
VariableDeclaration variable_1 = declaration_1.variables.variables[0];
TopLevelVariableElement element_1 = variable_1.declaredElement;
TopLevelVariableDeclaration declaration_2 = result.unit.declarations[1];
VariableDeclaration variable_2 = declaration_2.variables.variables[0];
TopLevelVariableElement element_2 = variable_2.declaredElement;
FunctionDeclaration main = result.unit.declarations[2];
Annotation annotation_1 = main.metadata[0];
expect(annotation_1.element, same(element_1.getter));
SimpleIdentifier identifier_1 = annotation_1.name;
expect(identifier_1.staticElement, same(element_1.getter));
expect(identifier_1.staticType, typeProvider.intType);
Annotation annotation_2 = main.metadata[1];
expect(annotation_2.element, same(element_2.getter));
SimpleIdentifier identifier_2 = annotation_2.name;
expect(identifier_2.staticElement, same(element_2.getter));
expect(identifier_2.staticType, typeProvider.intType);
}
test_asExpression() async {
String content = r'''
void main() {
num v = 42;
v as int;
}
''';
addTestFile(content);
await resolveTestFile();
expect(result.path, testFile);
expect(result.errors, isEmpty);
NodeList<Statement> statements = _getMainStatements(result);
// num v = 42;
VariableElement vElement;
{
VariableDeclarationStatement statement = statements[0];
vElement = statement.variables.variables[0].name.staticElement;
expect(vElement.type, typeProvider.numType);
}
// v as int;
{
ExpressionStatement statement = statements[1];
AsExpression asExpression = statement.expression;
expect(asExpression.staticType, typeProvider.intType);
SimpleIdentifier target = asExpression.expression;
expect(target.staticElement, vElement);
expect(target.staticType, typeProvider.numType);
TypeName intName = asExpression.type;
expect(intName.name.staticElement, typeProvider.intType.element);
expect(intName.name.staticType, typeProvider.intType);
}
}
test_binary_operator_with_synthetic_operands() async {
addTestFile('''
void main() {
var list = *;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
// Note: no further expectations. We don't care how the error is recovered
// from, provided it is recovered from in a way that doesn't crash the
// analyzer/FE integration.
}
test_binaryExpression() async {
String content = r'''
main() {
var v = 1 + 2;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> mainStatements = _getMainStatements(result);
VariableDeclarationStatement statement = mainStatements[0];
VariableDeclaration vNode = statement.variables.variables[0];
VariableElement vElement = vNode.declaredElement;
expect(vElement.type, typeProvider.intType);
BinaryExpression value = vNode.initializer;
expect(value.leftOperand.staticType, typeProvider.intType);
expect(value.rightOperand.staticType, typeProvider.intType);
expect(value.staticElement.name, '+');
expect(value.staticType, typeProvider.intType);
}
test_binaryExpression_ifNull() async {
String content = r'''
int x = 3;
main() {
1.2 ?? x;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> mainStatements = _getMainStatements(result);
ExpressionStatement statement = mainStatements[0];
BinaryExpression binary = statement.expression;
expect(binary.operator.type, TokenType.QUESTION_QUESTION);
expect(binary.staticElement, isNull);
expect(binary.staticType, typeProvider.numType);
expect(binary.leftOperand.staticType, typeProvider.doubleType);
expect(binary.rightOperand.staticType, typeProvider.intType);
}
test_binaryExpression_logical() async {
addTestFile(r'''
main() {
true && true;
true || true;
}
''');
await resolveTestFile();
List<Statement> statements = _getMainStatements(result);
{
ExpressionStatement statement = statements[0];
BinaryExpression binaryExpression = statement.expression;
expect(binaryExpression.staticElement, isNull);
expect(binaryExpression.staticType, typeProvider.boolType);
}
{
ExpressionStatement statement = statements[1];
BinaryExpression binaryExpression = statement.expression;
expect(binaryExpression.staticElement, isNull);
expect(binaryExpression.staticType, typeProvider.boolType);
}
}
test_binaryExpression_notEqual() async {
String content = r'''
main() {
1 != 2;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
BinaryExpression expression = statement.expression;
expect(expression.operator.type, TokenType.BANG_EQ);
expect(expression.leftOperand.staticType, typeProvider.intType);
expect(expression.rightOperand.staticType, typeProvider.intType);
expect(expression.staticElement.name, '==');
expect(expression.staticType, typeProvider.boolType);
}
test_cascade_get_with_numeric_getter_name() async {
addTestFile('''
void f(x) {
x..42;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
// Note: no further expectations. We don't care how the error is recovered
// from, provided it is recovered from in a way that doesn't crash the
// analyzer/FE integration.
}
test_cascade_method_call_with_synthetic_method_name() async {
addTestFile('''
void f(x) {
x..(42);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
// Note: no further expectations. We don't care how the error is recovered
// from, provided it is recovered from in a way that doesn't crash the
// analyzer/FE integration.
}
test_cascadeExpression() async {
String content = r'''
void main() {
new A()..a()..b();
}
class A {
void a() {}
void b() {}
}
''';
addTestFile(content);
await resolveTestFile();
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
CascadeExpression expression = statement.expression;
expect(expression.target.staticType, isNotNull);
NodeList<Expression> sections = expression.cascadeSections;
MethodInvocation a = sections[0];
expect(a.methodName.staticElement, isNotNull);
expect(a.staticType, isNotNull);
MethodInvocation b = sections[1];
expect(b.methodName.staticElement, isNotNull);
expect(b.staticType, isNotNull);
}
test_closure() async {
addTestFile(r'''
main() {
var items = <int>[1, 2, 3];
items.forEach((item) {
item;
});
items.forEach((item) {
item;
});
}
''');
await resolveTestFile();
FunctionDeclaration mainDeclaration = result.unit.declarations[0];
FunctionElement mainElement = mainDeclaration.declaredElement;
BlockFunctionBody mainBody = mainDeclaration.functionExpression.body;
List<Statement> mainStatements = mainBody.block.statements;
VariableDeclarationStatement itemsStatement = mainStatements[0];
var itemsElement = itemsStatement.variables.variables[0].declaredElement;
// First closure.
ParameterElement itemElement1;
{
ExpressionStatement forStatement = mainStatements[1];
MethodInvocation forInvocation = forStatement.expression;
SimpleIdentifier forTarget = forInvocation.target;
expect(forTarget.staticElement, itemsElement);
var closureTypeStr = '(int) → Null';
FunctionExpression closure = forInvocation.argumentList.arguments[0];
FunctionElementImpl closureElement = closure.declaredElement;
expect(closureElement.enclosingElement, same(mainElement));
ParameterElement itemElement = closureElement.parameters[0];
itemElement1 = itemElement;
expect(closureElement.returnType, typeProvider.nullType);
expect(closureElement.type.element, same(closureElement));
expect(closureElement.type.toString(), closureTypeStr);
expect(closure.staticType, same(closureElement.type));
List<FormalParameter> closureParameters = closure.parameters.parameters;
expect(closureParameters, hasLength(1));
SimpleFormalParameter itemNode = closureParameters[0];
_assertSimpleParameter(itemNode, itemElement,
name: 'item',
offset: 56,
kind: ParameterKind.REQUIRED,
type: typeProvider.intType);
BlockFunctionBody closureBody = closure.body;
List<Statement> closureStatements = closureBody.block.statements;
ExpressionStatement itemStatement = closureStatements[0];
SimpleIdentifier itemIdentifier = itemStatement.expression;
expect(itemIdentifier.staticElement, itemElement);
expect(itemIdentifier.staticType, typeProvider.intType);
}
// Second closure, same names, different elements.
{
ExpressionStatement forStatement = mainStatements[2];
MethodInvocation forInvocation = forStatement.expression;
SimpleIdentifier forTarget = forInvocation.target;
expect(forTarget.staticElement, itemsElement);
var closureTypeStr = '(int) → Null';
FunctionExpression closure = forInvocation.argumentList.arguments[0];
FunctionElementImpl closureElement = closure.declaredElement;
expect(closureElement.enclosingElement, same(mainElement));
ParameterElement itemElement = closureElement.parameters[0];
expect(itemElement, isNot(same(itemElement1)));
expect(closureElement.returnType, typeProvider.nullType);
expect(closureElement.type.element, same(closureElement));
expect(closureElement.type.toString(), closureTypeStr);
expect(closure.staticType, same(closureElement.type));
List<FormalParameter> closureParameters = closure.parameters.parameters;
expect(closureParameters, hasLength(1));
SimpleFormalParameter itemNode = closureParameters[0];
_assertSimpleParameter(itemNode, itemElement,
name: 'item',
offset: 97,
kind: ParameterKind.REQUIRED,
type: typeProvider.intType);
BlockFunctionBody closureBody = closure.body;
List<Statement> closureStatements = closureBody.block.statements;
ExpressionStatement itemStatement = closureStatements[0];
SimpleIdentifier itemIdentifier = itemStatement.expression;
expect(itemIdentifier.staticElement, itemElement);
expect(itemIdentifier.staticType, typeProvider.intType);
}
}
test_closure_generic() async {
addTestFile(r'''
main() {
foo(<T>() => new List<T>(4));
}
void foo(List<T> Function<T>() createList) {}
''');
await resolveTestFile();
var closure = findNode.functionExpression('<T>() =>');
assertType(closure, '<T>() → List<T>');
FunctionElementImpl closureElement = closure.declaredElement;
expect(closureElement.enclosingElement, findElement.function('main'));
expect(closureElement.returnType.toString(), 'List<T>');
expect(closureElement.parameters, isEmpty);
var typeParameters = closureElement.typeParameters;
expect(typeParameters, hasLength(1));
TypeParameterElement tElement = typeParameters[0];
expect(tElement.name, 'T');
expect(tElement.nameOffset, 16);
var creation = findNode.instanceCreation('new List');
assertType(creation, 'List<T>');
var tRef = findNode.simple('T>(4)');
assertElement(tRef, tElement);
}
test_closure_inField() async {
addTestFile(r'''
class C {
var v = (() => 42)();
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
ClassDeclaration c = unit.declarations[0];
FieldDeclaration declaration = c.members[0];
VariableDeclaration field = declaration.fields.variables[0];
FunctionElement fieldInitializer = field.declaredElement.initializer;
FunctionExpressionInvocation invocation = field.initializer;
FunctionExpression closure = invocation.function.unParenthesized;
FunctionElementImpl closureElement = closure.declaredElement;
expect(closureElement.enclosingElement, same(fieldInitializer));
}
test_closure_inTopLevelVariable() async {
addTestFile(r'''
var v = (() => 42)();
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
TopLevelVariableDeclaration declaration = unit.declarations[0];
VariableDeclaration variable = declaration.variables.variables[0];
TopLevelVariableElement variableElement = variable.declaredElement;
FunctionElement variableInitializer = variableElement.initializer;
FunctionExpressionInvocation invocation = variable.initializer;
FunctionExpression closure = invocation.function.unParenthesized;
FunctionElementImpl closureElement = closure.declaredElement;
expect(closureElement.enclosingElement, same(variableInitializer));
}
test_conditionalExpression() async {
String content = r'''
void main() {
true ? 1 : 2.3;
}
''';
addTestFile(content);
await resolveTestFile();
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
ConditionalExpression expression = statement.expression;
expect(expression.staticType, typeProvider.numType);
expect(expression.condition.staticType, typeProvider.boolType);
expect(expression.thenExpression.staticType, typeProvider.intType);
expect(expression.elseExpression.staticType, typeProvider.doubleType);
}
test_const_constructor_calls_non_const_super() async {
addTestFile('''
class A {
final a;
A(this.a);
}
class B extends A {
const B() : super(5);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
// Note: no further expectations. We don't care how the error is recovered
// from, provided it is recovered from in a way that doesn't crash the
// analyzer/FE integration.
}
test_constructor_context() async {
addTestFile(r'''
class C {
C(int p) {
p;
}
}
''');
await resolveTestFile();
ClassDeclaration cNode = result.unit.declarations[0];
ConstructorDeclaration constructorNode = cNode.members[0];
ParameterElement pElement = constructorNode.declaredElement.parameters[0];
BlockFunctionBody constructorBody = constructorNode.body;
ExpressionStatement pStatement = constructorBody.block.statements[0];
SimpleIdentifier pIdentifier = pStatement.expression;
expect(pIdentifier.staticElement, same(pElement));
expect(pIdentifier.staticType, typeProvider.intType);
}
test_constructor_initializer_field() async {
addTestFile(r'''
class C {
int f;
C(int p) : f = p {
f;
}
}
''');
await resolveTestFile();
ClassDeclaration cNode = result.unit.declarations[0];
ClassElement cElement = cNode.declaredElement;
FieldElement fElement = cElement.getField('f');
ConstructorDeclaration constructorNode = cNode.members[1];
ParameterElement pParameterElement =
constructorNode.declaredElement.parameters[0];
{
ConstructorFieldInitializer initializer = constructorNode.initializers[0];
expect(initializer.fieldName.staticElement, same(fElement));
SimpleIdentifier expression = initializer.expression;
expect(expression.staticElement, same(pParameterElement));
}
}
test_constructor_initializer_super() async {
addTestFile(r'''
class A {
A(int a);
A.named(int a, {int b});
}
class B extends A {
B.one(int p) : super(p + 1);
B.two(int p) : super.named(p + 1, b: p + 2);
}
''');
await resolveTestFile();
ClassDeclaration aNode = result.unit.declarations[0];
ClassElement aElement = aNode.declaredElement;
ClassDeclaration bNode = result.unit.declarations[1];
{
ConstructorDeclaration constructor = bNode.members[0];
SuperConstructorInvocation initializer = constructor.initializers[0];
expect(initializer.staticElement, same(aElement.unnamedConstructor));
expect(initializer.constructorName, isNull);
}
{
var namedConstructor = aElement.getNamedConstructor('named');
ConstructorDeclaration constructor = bNode.members[1];
SuperConstructorInvocation initializer = constructor.initializers[0];
expect(initializer.staticElement, same(namedConstructor));
var constructorName = initializer.constructorName;
expect(constructorName.staticElement, same(namedConstructor));
expect(constructorName.staticType, isNull);
List<Expression> arguments = initializer.argumentList.arguments;
_assertArgumentToParameter(arguments[0], namedConstructor.parameters[0]);
_assertArgumentToParameter(arguments[1], namedConstructor.parameters[1]);
}
}
test_constructor_initializer_this() async {
addTestFile(r'''
class C {
C(int a, [int b]);
C.named(int a, {int b});
C.one(int p) : this(1, 2);
C.two(int p) : this.named(3, b: 4);
}
''');
await resolveTestFile();
ClassDeclaration cNode = result.unit.declarations[0];
ClassElement cElement = cNode.declaredElement;
{
var unnamedConstructor = cElement.constructors[0];
ConstructorDeclaration constructor = cNode.members[2];
RedirectingConstructorInvocation initializer =
constructor.initializers[0];
expect(initializer.staticElement, same(unnamedConstructor));
expect(initializer.constructorName, isNull);
List<Expression> arguments = initializer.argumentList.arguments;
_assertArgumentToParameter(
arguments[0], unnamedConstructor.parameters[0]);
_assertArgumentToParameter(
arguments[1], unnamedConstructor.parameters[1]);
}
{
var namedConstructor = cElement.constructors[1];
ConstructorDeclaration constructor = cNode.members[3];
RedirectingConstructorInvocation initializer =
constructor.initializers[0];
expect(initializer.staticElement, same(namedConstructor));
var constructorName = initializer.constructorName;
expect(constructorName.staticElement, same(namedConstructor));
expect(constructorName.staticType, isNull);
List<Expression> arguments = initializer.argumentList.arguments;
_assertArgumentToParameter(arguments[0], namedConstructor.parameters[0]);
_assertArgumentToParameter(arguments[1], namedConstructor.parameters[1]);
}
}
test_constructor_redirected() async {
addTestFile(r'''
class A implements B {
A(int a);
A.named(double a);
}
class B {
factory B.one(int b) = A;
factory B.two(double b) = A.named;
}
''');
await resolveTestFile();
expect(result.errors, isEmpty);
ClassDeclaration aNode = result.unit.declarations[0];
ClassElement aElement = aNode.declaredElement;
ClassDeclaration bNode = result.unit.declarations[1];
{
ConstructorElement aUnnamed = aElement.constructors[0];
ConstructorDeclaration constructor = bNode.members[0];
ConstructorElement element = constructor.declaredElement;
expect(element.redirectedConstructor, same(aUnnamed));
var constructorName = constructor.redirectedConstructor;
expect(constructorName.staticElement, same(aUnnamed));
TypeName typeName = constructorName.type;
expect(typeName.type, aElement.type);
SimpleIdentifier identifier = typeName.name;
expect(identifier.staticElement, same(aElement));
expect(identifier.staticType, aElement.type);
expect(constructorName.name, isNull);
}
{
ConstructorElement aNamed = aElement.constructors[1];
ConstructorDeclaration constructor = bNode.members[1];
ConstructorElement element = constructor.declaredElement;
expect(element.redirectedConstructor, same(aNamed));
var constructorName = constructor.redirectedConstructor;
expect(constructorName.staticElement, same(aNamed));
TypeName typeName = constructorName.type;
expect(typeName.type, aElement.type);
SimpleIdentifier identifier = typeName.name;
expect(identifier.staticElement, same(aElement));
expect(identifier.staticType, aElement.type);
expect(constructorName.name.staticElement, aNamed);
expect(constructorName.name.staticType, isNull);
}
}
test_constructor_redirected_generic() async {
addTestFile(r'''
class A<T> implements B<T> {
A(int a);
A.named(double a);
}
class B<U> {
factory B.one(int b) = A<U>;
factory B.two(double b) = A<U>.named;
}
''');
await resolveTestFile();
expect(result.errors, isEmpty);
ClassDeclaration aNode = result.unit.declarations[0];
ClassElement aElement = aNode.declaredElement;
ClassDeclaration bNode = result.unit.declarations[1];
TypeParameterType uType = bNode.declaredElement.typeParameters[0].type;
InterfaceType auType = aElement.type.instantiate([uType]);
{
ConstructorElement expectedElement = aElement.constructors[0];
ConstructorDeclaration constructor = bNode.members[0];
ConstructorElement element = constructor.declaredElement;
ConstructorMember actualMember = element.redirectedConstructor;
expect(actualMember.baseElement, same(expectedElement));
expect(actualMember.definingType, auType);
var constructorName = constructor.redirectedConstructor;
expect(constructorName.staticElement, same(actualMember));
TypeName typeName = constructorName.type;
expect(typeName.type, auType);
SimpleIdentifier identifier = typeName.name;
expect(identifier.staticElement, same(aElement));
expect(identifier.staticType, auType);
expect(constructorName.name, isNull);
}
{
ConstructorElement expectedElement = aElement.constructors[1];
ConstructorDeclaration constructor = bNode.members[1];
ConstructorElement element = constructor.declaredElement;
ConstructorMember actualMember = element.redirectedConstructor;
expect(actualMember.baseElement, same(expectedElement));
expect(actualMember.definingType, auType);
var constructorName = constructor.redirectedConstructor;
expect(constructorName.staticElement, same(actualMember));
TypeName typeName = constructorName.type;
expect(typeName.type, auType);
SimpleIdentifier identifier = typeName.name;
expect(identifier.staticElement, same(aElement));
expect(identifier.staticType, auType);
expect(constructorName.name.staticElement, same(actualMember));
expect(constructorName.name.staticType, isNull);
}
}
@failingTest
test_deferredImport_loadLibrary_invocation() async {
newFile('/test/lib/a.dart');
addTestFile(r'''
import 'a.dart' deferred as a;
main() {
a.loadLibrary();
}
''');
await resolveTestFile();
var import = findElement.import('package:test/a.dart');
var invocation = findNode.methodInvocation('loadLibrary');
assertType(invocation, 'Future<dynamic>');
assertInvokeType(invocation, '() → Future<dynamic>');
SimpleIdentifier target = invocation.target;
assertElement(target, import.prefix);
assertType(target, null);
var name = invocation.methodName;
assertElement(name, import.importedLibrary.loadLibraryFunction);
assertType(name, '() → Future<dynamic>');
}
@failingTest
test_deferredImport_loadLibrary_invocation_argument() async {
newFile('/test/lib/a.dart');
addTestFile(r'''
import 'a.dart' deferred as a;
var b = 1;
var c = 2;
main() {
a.loadLibrary(b, c);
}
''');
await resolveTestFile();
var import = findElement.import('package:test/a.dart');
var invocation = findNode.methodInvocation('loadLibrary');
assertType(invocation, 'Future<dynamic>');
assertInvokeType(invocation, '() → Future<dynamic>');
SimpleIdentifier target = invocation.target;
assertElement(target, import.prefix);
assertType(target, null);
var name = invocation.methodName;
assertElement(name, import.importedLibrary.loadLibraryFunction);
assertType(name, '() → Future<dynamic>');
var bRef = invocation.argumentList.arguments[0];
assertElement(bRef, findElement.topGet('b'));
assertType(bRef, 'int');
var cRef = invocation.argumentList.arguments[1];
assertElement(cRef, findElement.topGet('c'));
assertType(cRef, 'int');
}
test_deferredImport_loadLibrary_tearOff() async {
newFile('/test/lib/a.dart');
addTestFile(r'''
import 'a.dart' deferred as a;
main() {
a.loadLibrary;
}
''');
await resolveTestFile();
var import = findElement.import('package:test/a.dart');
var prefixed = findNode.prefixed('a.loadLibrary');
assertType(prefixed, '() → Future<dynamic>');
var prefix = prefixed.prefix;
assertElement(prefix, import.prefix);
assertType(prefix, null);
var identifier = prefixed.identifier;
assertElement(identifier, import.importedLibrary.loadLibraryFunction);
assertType(identifier, '() → Future<dynamic>');
}
test_deferredImport_variable() async {
newFile('/test/lib/a.dart', content: 'var v = 0;');
addTestFile(r'''
import 'a.dart' deferred as a;
main() async {
a.v;
a.v = 1;
}
''');
await resolveTestFile();
var import = findElement.import('package:test/a.dart');
TopLevelVariableElement v = (import.importedLibrary.publicNamespace.get('v')
as PropertyAccessorElement)
.variable;
{
var prefixed = findNode.prefixed('a.v;');
assertElement(prefixed, v.getter);
assertType(prefixed, 'int');
assertElement(prefixed.prefix, import.prefix);
assertType(prefixed.prefix, null);
assertElement(prefixed.identifier, v.getter);
assertType(prefixed.identifier, 'int');
}
{
var prefixed = findNode.prefixed('a.v = 1;');
assertElement(prefixed, v.setter);
assertType(prefixed, 'int');
assertElement(prefixed.prefix, import.prefix);
assertType(prefixed.prefix, null);
assertElement(prefixed.identifier, v.setter);
assertType(prefixed.identifier, 'int');
}
}
test_directive_export() async {
newFile('/test/lib/a.dart', content: r'''
class MyClass {}
int myVar;
int get myGetter => 0;
int set mySetter(_) {}
''');
addTestFile(r'''
export 'a.dart' show MyClass, myVar, myGetter, mySetter, Unresolved;
''');
await resolveTestFile();
var export = findElement.export('package:test/a.dart');
var namespace = export.exportedLibrary.exportNamespace;
{
var ref = findNode.simple('MyClass');
assertElement(ref, namespace.get('MyClass'));
assertType(ref, null);
}
{
var ref = findNode.simple('myVar');
PropertyAccessorElement getter = namespace.get('myVar');
assertElement(ref, getter.variable);
assertType(ref, null);
}
{
var ref = findNode.simple('myGetter');
PropertyAccessorElement getter = namespace.get('myGetter');
assertElement(ref, getter.variable);
assertType(ref, null);
}
{
var ref = findNode.simple('mySetter');
PropertyAccessorElement getter = namespace.get('mySetter=');
assertElement(ref, getter.variable);
assertType(ref, null);
}
{
var ref = findNode.simple('Unresolved');
assertElementNull(ref);
assertType(ref, null);
}
}
test_directive_import_hide() async {
newFile('/test/lib/a.dart', content: r'''
class MyClass {}
int myVar;
int get myGetter => 0;
int set mySetter(_) {}
''');
addTestFile(r'''
import 'a.dart' hide MyClass, myVar, myGetter, mySetter, Unresolved;
''');
await resolveTestFile();
var import = findElement.import('package:test/a.dart');
var namespace = import.importedLibrary.exportNamespace;
{
var ref = findNode.simple('MyClass');
assertElement(ref, namespace.get('MyClass'));
assertType(ref, null);
}
{
var ref = findNode.simple('myVar');
PropertyAccessorElement getter = namespace.get('myVar');
assertElement(ref, getter.variable);
assertType(ref, null);
}
{
var ref = findNode.simple('myGetter');
PropertyAccessorElement getter = namespace.get('myGetter');
assertElement(ref, getter.variable);
assertType(ref, null);
}
{
var ref = findNode.simple('mySetter');
PropertyAccessorElement getter = namespace.get('mySetter=');
assertElement(ref, getter.variable);
assertType(ref, null);
}
{
var ref = findNode.simple('Unresolved');
assertElementNull(ref);
assertType(ref, null);
}
}
test_directive_import_show() async {
newFile('/test/lib/a.dart', content: r'''
class MyClass {}
int myVar;
int get myGetter => 0;
int set mySetter(_) {}
''');
addTestFile(r'''
import 'a.dart' show MyClass, myVar, myGetter, mySetter, Unresolved;
''');
await resolveTestFile();
var import = findElement.import('package:test/a.dart');
var namespace = import.importedLibrary.exportNamespace;
{
var ref = findNode.simple('MyClass');
assertElement(ref, namespace.get('MyClass'));
assertType(ref, null);
}
{
var ref = findNode.simple('myVar');
PropertyAccessorElement getter = namespace.get('myVar');
assertElement(ref, getter.variable);
assertType(ref, null);
}
{
var ref = findNode.simple('myGetter');
PropertyAccessorElement getter = namespace.get('myGetter');
assertElement(ref, getter.variable);
assertType(ref, null);
}
{
var ref = findNode.simple('mySetter');
PropertyAccessorElement getter = namespace.get('mySetter=');
assertElement(ref, getter.variable);
assertType(ref, null);
}
{
var ref = findNode.simple('Unresolved');
assertElementNull(ref);
assertType(ref, null);
}
}
test_enum_toString() async {
addTestFile(r'''
enum MyEnum { A, B, C }
main(MyEnum e) {
e.toString();
}
''');
await resolveTestFile();
EnumDeclaration enumNode = result.unit.declarations[0];
ClassElement enumElement = enumNode.declaredElement;
List<Statement> mainStatements = _getMainStatements(result);
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.staticInvokeType.toString(), '() → String');
MethodElement methodElement = invocation.methodName.staticElement;
expect(methodElement.name, 'toString');
expect(methodElement.enclosingElement, same(enumElement));
}
test_error_unresolvedTypeAnnotation() async {
String content = r'''
main() {
Foo<int> v = null;
}
''';
addTestFile(content);
await resolveTestFile();
var statements = _getMainStatements(result);
VariableDeclarationStatement statement = statements[0];
TypeName typeName = statement.variables.type;
expect(typeName.type, isDynamicType);
expect(typeName.typeArguments.arguments[0].type, typeProvider.intType);
VariableDeclaration vNode = statement.variables.variables[0];
expect(vNode.name.staticType, isDynamicType);
expect(vNode.declaredElement.type, isDynamicType);
}
test_field_context() async {
addTestFile(r'''
class C<T> {
var f = <T>[];
}
''');
await resolveTestFile();
ClassDeclaration cNode = result.unit.declarations[0];
var tElement = cNode.declaredElement.typeParameters[0];
FieldDeclaration fDeclaration = cNode.members[0];
VariableDeclaration fNode = fDeclaration.fields.variables[0];
FieldElement fElement = fNode.declaredElement;
expect(fElement.type, typeProvider.listType.instantiate([tElement.type]));
}
test_field_generic() async {
addTestFile(r'''
class C<T> {
T f;
}
main(C<int> c) {
c.f; // ref
c.f = 1;
}
''');
await resolveTestFile();
{
var fRef = findNode.simple('f; // ref');
assertMember(fRef, 'C<int>', findElement.getter('f'));
assertType(fRef, 'int');
}
{
var fRef = findNode.simple('f = 1;');
assertMember(fRef, 'C<int>', findElement.setter('f'));
assertType(fRef, 'int');
}
}
test_formalParameter_functionTyped() async {
addTestFile(r'''
class A {
A(String p(int a));
}
''');
await resolveTestFile();
ClassDeclaration clazz = result.unit.declarations[0];
ConstructorDeclaration constructor = clazz.members[0];
List<FormalParameter> parameters = constructor.parameters.parameters;
FunctionTypedFormalParameter p = parameters[0];
expect(p.declaredElement, same(constructor.declaredElement.parameters[0]));
{
FunctionType type = p.identifier.staticType;
expect(type.returnType, typeProvider.stringType);
expect(type.parameters, hasLength(1));
expect(type.parameters[0].type, typeProvider.intType);
}
_assertTypeNameSimple(p.returnType, typeProvider.stringType);
{
SimpleFormalParameter a = p.parameters.parameters[0];
_assertTypeNameSimple(a.type, typeProvider.intType);
expect(a.identifier.staticType, typeProvider.intType);
}
}
test_formalParameter_functionTyped_fieldFormal_typed() async {
// TODO(scheglov) Add "untyped" version with precise type in field.
addTestFile(r'''
class A {
Function f;
A(String this.f(int a));
}
''');
await resolveTestFile();
ClassDeclaration clazz = result.unit.declarations[0];
FieldDeclaration fDeclaration = clazz.members[0];
VariableDeclaration fNode = fDeclaration.fields.variables[0];
FieldElement fElement = fNode.declaredElement;
ConstructorDeclaration constructor = clazz.members[1];
FieldFormalParameterElement pElement =
constructor.declaredElement.parameters[0];
expect(pElement.field, same(fElement));
List<FormalParameter> parameters = constructor.parameters.parameters;
FieldFormalParameter p = parameters[0];
expect(p.declaredElement, same(pElement));
expect(p.identifier.staticElement, same(pElement));
expect(p.identifier.staticType.toString(), '(int) → String');
{
FunctionType type = p.identifier.staticType;
expect(type.returnType, typeProvider.stringType);
expect(type.parameters, hasLength(1));
expect(type.parameters[0].type, typeProvider.intType);
}
_assertTypeNameSimple(p.type, typeProvider.stringType);
{
SimpleFormalParameter a = p.parameters.parameters[0];
_assertTypeNameSimple(a.type, typeProvider.intType);
expect(a.identifier.staticType, typeProvider.intType);
}
}
test_formalParameter_simple_fieldFormal() async {
addTestFile(r'''
class A {
int f;
A(this.f);
}
''');
await resolveTestFile();
ClassDeclaration clazz = result.unit.declarations[0];
FieldDeclaration fDeclaration = clazz.members[0];
VariableDeclaration fNode = fDeclaration.fields.variables[0];
FieldElement fElement = fNode.declaredElement;
ConstructorDeclaration constructor = clazz.members[1];
List<FormalParameter> parameters = constructor.parameters.parameters;
FieldFormalParameterElement parameterElement =
constructor.declaredElement.parameters[0];
expect(parameterElement.field, same(fElement));
FieldFormalParameter parameterNode = parameters[0];
expect(parameterNode.type, isNull);
expect(parameterNode.declaredElement, same(parameterElement));
expect(parameterNode.identifier.staticElement, same(parameterElement));
expect(parameterNode.identifier.staticType, typeProvider.intType);
}
test_formalParameter_simple_fieldFormal_typed() async {
addTestFile(r'''
class A {
int f;
A(int this.f);
}
''');
await resolveTestFile();
ClassDeclaration clazz = result.unit.declarations[0];
FieldDeclaration fDeclaration = clazz.members[0];
VariableDeclaration fNode = fDeclaration.fields.variables[0];
FieldElement fElement = fNode.declaredElement;
ConstructorDeclaration constructor = clazz.members[1];
List<FormalParameter> parameters = constructor.parameters.parameters;
FieldFormalParameterElement parameterElement =
constructor.declaredElement.parameters[0];
expect(parameterElement.field, same(fElement));
FieldFormalParameter parameterNode = parameters[0];
_assertTypeNameSimple(parameterNode.type, typeProvider.intType);
expect(parameterNode.declaredElement, same(parameterElement));
expect(parameterNode.identifier.staticElement, same(parameterElement));
expect(parameterNode.identifier.staticType, typeProvider.intType);
}
test_forwardingStub_class() async {
addTestFile(r'''
class A<T> {
void m(T t) {}
}
class B extends A<int> {}
main(B b) {
b.m(1);
}
''');
await resolveTestFile();
// ClassDeclaration aNode = result.unit.declarations[0];
// ClassElement eElement = aNode.declaredElement;
// MethodElement mElement = eElement.getMethod('m');
List<Statement> mainStatements = _getMainStatements(result);
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.staticInvokeType.toString(), '(int) → void');
// TODO(scheglov) Check for MethodElement
// expect(invocation.methodName.staticElement, same(mElement));
}
test_function_call_with_synthetic_arguments() async {
addTestFile('''
void f(x) {}
class C {
m() {
f(,);
}
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
// Note: no further expectations. We don't care how the error is recovered
// from, provided it is recovered from in a way that doesn't crash the
// analyzer/FE integration.
}
test_functionExpressionInvocation() async {
addTestFile(r'''
typedef Foo<S> = S Function<T>(T x);
void main(f) {
(f as Foo<int>)<String>('hello');
}
''');
await resolveTestFile();
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
FunctionExpressionInvocation invocation = statement.expression;
expect(invocation.staticElement, isNull);
expect(invocation.staticInvokeType.toString(), '(String) → int');
expect(invocation.staticType, typeProvider.intType);
List<TypeAnnotation> typeArguments = invocation.typeArguments.arguments;
expect(typeArguments, hasLength(1));
_assertTypeNameSimple(typeArguments[0], typeProvider.stringType);
}
test_functionExpressionInvocation_namedArgument() async {
addTestFile(r'''
int a;
main(f) {
(f)(p: a);
}
''');
await resolveTestFile();
assertTopGetRef('a);', 'a');
}
test_generic_function_type() async {
addTestFile('''
main() {
void Function<T>(T) f;
}
''');
await resolveTestFile();
var f = findNode.simple('f;');
assertType(f, '<T>(T) → void');
var fType = f.staticType as FunctionType;
var fTypeTypeParameter = fType.typeFormals[0];
var fTypeParameter = fType.normalParameterTypes[0] as TypeParameterType;
expect(fTypeParameter.element, same(fTypeTypeParameter));
var tRef = findNode.simple('T>');
var functionTypeNode = tRef.parent.parent.parent as GenericFunctionType;
var functionType = functionTypeNode.type as FunctionType;
assertElement(tRef, functionType.typeFormals[0]);
}
test_indexExpression() async {
String content = r'''
main() {
var items = <int>[1, 2, 3];
items[0];
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
InterfaceType intType = typeProvider.intType;
InterfaceType listType = typeProvider.listType;
InterfaceType listIntType = listType.instantiate([intType]);
List<Statement> mainStatements = _getMainStatements(result);
VariableElement itemsElement;
{
VariableDeclarationStatement statement = mainStatements[0];
VariableDeclaration itemsNode = statement.variables.variables[0];
itemsElement = itemsNode.declaredElement;
expect(itemsElement.type, listIntType);
}
ExpressionStatement statement = mainStatements[1];
IndexExpression indexExpression = statement.expression;
expect(indexExpression.staticType, intType);
MethodMember actualElement = indexExpression.staticElement;
MethodMember expectedElement = listIntType.getMethod('[]');
expect(actualElement.name, '[]');
expect(actualElement.baseElement, same(expectedElement.baseElement));
expect(actualElement.returnType, intType);
expect(actualElement.parameters[0].type, intType);
}
test_instanceCreation_factory() async {
String content = r'''
class C {
factory C() => null;
factory C.named() => null;
}
var a = new C();
var b = new C.named();
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
ClassDeclaration cNode = unit.declarations[0];
ClassElement cElement = cNode.declaredElement;
ConstructorElement defaultConstructor = cElement.constructors[0];
ConstructorElement namedConstructor = cElement.constructors[1];
{
TopLevelVariableDeclaration aDeclaration = unit.declarations[1];
VariableDeclaration aNode = aDeclaration.variables.variables[0];
InstanceCreationExpression value = aNode.initializer;
expect(value.staticElement, defaultConstructor);
expect(value.staticType, cElement.type);
TypeName typeName = value.constructorName.type;
expect(typeName.typeArguments, isNull);
Identifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, cElement);
expect(typeIdentifier.staticType, cElement.type);
expect(value.constructorName.name, isNull);
}
{
TopLevelVariableDeclaration bDeclaration = unit.declarations[2];
VariableDeclaration bNode = bDeclaration.variables.variables[0];
InstanceCreationExpression value = bNode.initializer;
expect(value.staticElement, namedConstructor);
expect(value.staticType, cElement.type);
TypeName typeName = value.constructorName.type;
expect(typeName.typeArguments, isNull);
SimpleIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, cElement);
expect(typeIdentifier.staticType, cElement.type);
SimpleIdentifier constructorName = value.constructorName.name;
expect(constructorName.staticElement, namedConstructor);
expect(constructorName.staticType, isNull);
}
}
test_instanceCreation_namedArgument() async {
addTestFile(r'''
class X {
X(int a, {bool b, double c});
}
var v = new X(1, b: true, c: 3.0);
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
ClassDeclaration xNode = unit.declarations[0];
ClassElement xElement = xNode.declaredElement;
ConstructorElement constructorElement = xElement.constructors[0];
TopLevelVariableDeclaration vDeclaration = unit.declarations[1];
VariableDeclaration vNode = vDeclaration.variables.variables[0];
InstanceCreationExpression creation = vNode.initializer;
List<Expression> arguments = creation.argumentList.arguments;
expect(creation.staticElement, constructorElement);
expect(creation.staticType, xElement.type);
TypeName typeName = creation.constructorName.type;
expect(typeName.typeArguments, isNull);
Identifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, xElement);
expect(typeIdentifier.staticType, xElement.type);
expect(creation.constructorName.name, isNull);
_assertArgumentToParameter(arguments[0], constructorElement.parameters[0]);
_assertArgumentToParameter(arguments[1], constructorElement.parameters[1]);
_assertArgumentToParameter(arguments[2], constructorElement.parameters[2]);
}
test_instanceCreation_noTypeArguments() async {
String content = r'''
class C {
C(int p);
C.named(int p);
}
var a = new C(1);
var b = new C.named(2);
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
ClassDeclaration cNode = unit.declarations[0];
ClassElement cElement = cNode.declaredElement;
ConstructorElement defaultConstructor = cElement.constructors[0];
ConstructorElement namedConstructor = cElement.constructors[1];
{
TopLevelVariableDeclaration aDeclaration = unit.declarations[1];
VariableDeclaration aNode = aDeclaration.variables.variables[0];
InstanceCreationExpression value = aNode.initializer;
expect(value.staticElement, defaultConstructor);
expect(value.staticType, cElement.type);
TypeName typeName = value.constructorName.type;
expect(typeName.typeArguments, isNull);
Identifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, cElement);
expect(typeIdentifier.staticType, cElement.type);
expect(value.constructorName.name, isNull);
Expression argument = value.argumentList.arguments[0];
_assertArgumentToParameter(argument, defaultConstructor.parameters[0]);
}
{
TopLevelVariableDeclaration bDeclaration = unit.declarations[2];
VariableDeclaration bNode = bDeclaration.variables.variables[0];
InstanceCreationExpression value = bNode.initializer;
expect(value.staticElement, namedConstructor);
expect(value.staticType, cElement.type);
TypeName typeName = value.constructorName.type;
expect(typeName.typeArguments, isNull);
SimpleIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, cElement);
expect(typeIdentifier.staticType, cElement.type);
SimpleIdentifier constructorName = value.constructorName.name;
expect(constructorName.staticElement, namedConstructor);
expect(constructorName.staticType, isNull);
Expression argument = value.argumentList.arguments[0];
_assertArgumentToParameter(argument, namedConstructor.parameters[0]);
}
}
test_instanceCreation_prefixed() async {
newFile('/test/lib/a.dart', content: r'''
class C<T> {
C(T p);
C.named(T p);
}
''');
addTestFile(r'''
import 'a.dart' as p;
main() {
new p.C(0);
new p.C.named(1.2);
new p.C<bool>.named(false);
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
ImportElement aImport = unit.declaredElement.library.imports[0];
LibraryElement aLibrary = aImport.importedLibrary;
ClassElement cElement = aLibrary.getType('C');
ConstructorElement defaultConstructor = cElement.constructors[0];
ConstructorElement namedConstructor = cElement.constructors[1];
InterfaceType cType = cElement.type;
var cTypeDynamic = cType.instantiate([DynamicTypeImpl.instance]);
var statements = _getMainStatements(result);
{
var cTypeInt = cType.instantiate([typeProvider.intType]);
ExpressionStatement statement = statements[0];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticElement, defaultConstructor);
expect(creation.staticType, cTypeInt);
TypeName typeName = creation.constructorName.type;
expect(typeName.typeArguments, isNull);
PrefixedIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, same(cElement));
expect(typeIdentifier.staticType, cTypeDynamic);
SimpleIdentifier typePrefix = typeIdentifier.prefix;
expect(typePrefix.name, 'p');
expect(typePrefix.staticElement, same(aImport.prefix));
expect(typePrefix.staticType, isNull);
expect(typeIdentifier.identifier.staticElement, same(cElement));
expect(creation.constructorName.name, isNull);
}
{
var cTypeDouble = cType.instantiate([typeProvider.doubleType]);
ExpressionStatement statement = statements[1];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticElement, namedConstructor);
expect(creation.staticType, cTypeDouble);
TypeName typeName = creation.constructorName.type;
expect(typeName.typeArguments, isNull);
PrefixedIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, cElement);
expect(typeIdentifier.staticType, cTypeDynamic);
SimpleIdentifier typePrefix = typeIdentifier.prefix;
expect(typePrefix.name, 'p');
expect(typePrefix.staticElement, same(aImport.prefix));
expect(typePrefix.staticType, isNull);
expect(typeIdentifier.identifier.staticElement, same(cElement));
SimpleIdentifier constructorName = creation.constructorName.name;
expect(constructorName.staticElement, namedConstructor);
expect(constructorName.staticType, isNull);
}
{
var cTypeBool = cType.instantiate([typeProvider.boolType]);
ExpressionStatement statement = statements[2];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticElement, namedConstructor);
expect(creation.staticType, cTypeBool);
TypeName typeName = creation.constructorName.type;
expect(typeName.typeArguments.arguments, hasLength(1));
_assertTypeNameSimple(
typeName.typeArguments.arguments[0], typeProvider.boolType);
PrefixedIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, cElement);
expect(typeIdentifier.staticType, cTypeBool);
SimpleIdentifier typePrefix = typeIdentifier.prefix;
expect(typePrefix.name, 'p');
expect(typePrefix.staticElement, same(aImport.prefix));
expect(typePrefix.staticType, isNull);
expect(typeIdentifier.identifier.staticElement, same(cElement));
SimpleIdentifier constructorName = creation.constructorName.name;
expect(constructorName.staticElement, namedConstructor);
expect(constructorName.staticType, isNull);
}
}
@failingTest
test_instanceCreation_unprefixed() async {
addTestFile(r'''
main() {
new C(0);
new C<bool>(false);
new C.named(1.2);
new C<bool>.named(false);
}
class C<T> {
C(T p);
C.named(T p);
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
var typeProvider = unitElement.context.typeProvider;
ClassElement cElement = unitElement.getType('C');
ConstructorElement defaultConstructor = cElement.constructors[0];
ConstructorElement namedConstructor = cElement.constructors[1];
InterfaceType cType = cElement.type;
var statements = _getMainStatements(result);
{
var cTypeInt = cType.instantiate([typeProvider.intType]);
ExpressionStatement statement = statements[0];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticElement, defaultConstructor);
expect(creation.staticType, cTypeInt);
TypeName typeName = creation.constructorName.type;
expect(typeName.typeArguments, isNull);
SimpleIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, same(cElement));
expect(typeIdentifier.staticType, isNull);
expect(creation.constructorName.name, isNull);
}
{
var cTypeBool = cType.instantiate([typeProvider.boolType]);
ExpressionStatement statement = statements[1];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticElement, defaultConstructor);
expect(creation.staticType, cTypeBool);
TypeName typeName = creation.constructorName.type;
expect(typeName.typeArguments.arguments, hasLength(1));
_assertTypeNameSimple(
typeName.typeArguments.arguments[0], typeProvider.boolType);
SimpleIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, same(cElement));
expect(typeIdentifier.staticType, isNull);
expect(creation.constructorName.name, isNull);
}
{
var cTypeDouble = cType.instantiate([typeProvider.doubleType]);
ExpressionStatement statement = statements[2];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticElement, namedConstructor);
expect(creation.staticType, cTypeDouble);
TypeName typeName = creation.constructorName.type;
expect(typeName.typeArguments, isNull);
SimpleIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, cElement);
expect(typeIdentifier.staticType, isNull);
expect(typeIdentifier.staticElement, same(cElement));
SimpleIdentifier constructorName = creation.constructorName.name;
expect(constructorName.staticElement, namedConstructor);
expect(constructorName.staticType, isNull);
}
{
var cTypeBool = cType.instantiate([typeProvider.boolType]);
ExpressionStatement statement = statements[3];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticElement, namedConstructor);
expect(creation.staticType, cTypeBool);
TypeName typeName = creation.constructorName.type;
expect(typeName.typeArguments.arguments, hasLength(1));
_assertTypeNameSimple(
typeName.typeArguments.arguments[0], typeProvider.boolType);
SimpleIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, cElement);
expect(typeIdentifier.staticType, isNull);
SimpleIdentifier constructorName = creation.constructorName.name;
expect(constructorName.staticElement, namedConstructor);
expect(constructorName.staticType, isNull);
}
}
test_instanceCreation_withTypeArguments() async {
addTestFile(r'''
class C<K, V> {
C(K k, V v);
C.named(K k, V v);
}
var a = new C<int, double>(1, 2.3);
var b = new C<num, String>.named(4, 'five');
''');
await resolveTestFile();
var cElement = findElement.class_('C');
var defaultConstructor = cElement.unnamedConstructor;
var namedConstructor = cElement.getNamedConstructor('named');
{
var creation = findNode.instanceCreation('new C<int, double>(1, 2.3);');
assertMember(creation, 'C<int, double>', defaultConstructor);
assertType(creation, 'C<int, double>');
var typeName = creation.constructorName.type;
assertTypeName(typeName, cElement, 'C<int, double>');
var typeArguments = typeName.typeArguments.arguments;
assertTypeName(typeArguments[0], intElement, 'int');
assertTypeName(typeArguments[1], doubleElement, 'double');
expect(creation.constructorName.name, isNull);
Expression argument = creation.argumentList.arguments[0];
_assertArgumentToParameter(argument, defaultConstructor.parameters[0]);
}
{
var creation = findNode.instanceCreation('new C<num, String>.named');
assertMember(creation, 'C<num, String>', namedConstructor);
assertType(creation, 'C<num, String>');
var typeName = creation.constructorName.type;
assertTypeName(typeName, cElement, 'C<num, String>');
var typeArguments = typeName.typeArguments.arguments;
assertTypeName(typeArguments[0], numElement, 'num');
assertTypeName(typeArguments[1], stringElement, 'String');
var constructorName = creation.constructorName.name;
assertMember(constructorName, 'C<num, String>', namedConstructor);
assertType(constructorName, null);
var argument = creation.argumentList.arguments[0];
_assertArgumentToParameter(argument, namedConstructor.parameters[0]);
}
}
test_invalid_annotation_on_variable_declaration() async {
addTestFile(r'''
const a = null;
main() {
int x, @a y;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
// Note: no further expectations. We don't care how the error is recovered
// from, provided it is recovered from in a way that doesn't crash the
// analyzer/FE integration.
}
test_invalid_annotation_on_variable_declaration_for() async {
addTestFile(r'''
const a = null;
main() {
for (var @a x = 0;;) {}
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
// Note: no further expectations. We don't care how the error is recovered
// from, provided it is recovered from in a way that doesn't crash the
// analyzer/FE integration.
}
test_invalid_catch_parameters_3() async {
addTestFile(r'''
main() {
try { } catch (x, y, z) { }
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
assertTypeDynamic(findNode.simple('x,'));
assertType(findNode.simple('y,'), 'StackTrace');
}
test_invalid_catch_parameters_empty() async {
addTestFile(r'''
main() {
try { } catch () { }
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
}
test_invalid_catch_parameters_named_stack() async {
addTestFile(r'''
main() {
try { } catch (e, {s}) { }
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
assertTypeDynamic(findNode.simple('e,'));
assertType(findNode.simple('s})'), 'StackTrace');
}
test_invalid_catch_parameters_optional_stack() async {
addTestFile(r'''
main() {
try { } catch (e, [s]) { }
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
assertTypeDynamic(findNode.simple('e,'));
assertType(findNode.simple('s])'), 'StackTrace');
}
test_invalid_const_as() async {
addTestFile(r'''
const num a = 1.2;
const int b = a as int;
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a as int');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'num');
assertTypeName(findNode.typeName('int;'), intElement, 'int');
}
test_invalid_const_constructor_initializer_field_multiple() async {
addTestFile(r'''
var a = 0;
class A {
final x = 0;
const A() : x = a;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.simple('x = a');
assertElement(xRef, findElement.field('x'));
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_const_methodInvocation() async {
addTestFile(r'''
const a = 'foo';
const b = 0;
const c = a.codeUnitAt(b);
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var invocation = findNode.methodInvocation('codeUnitAt');
assertType(invocation, 'int');
assertInvokeType(invocation, '(int) → int');
assertElement(invocation.methodName, stringElement.getMethod('codeUnitAt'));
var aRef = invocation.target;
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'String');
var bRef = invocation.argumentList.arguments[0];
assertElement(bRef, findElement.topGet('b'));
assertType(bRef, 'int');
}
test_invalid_const_methodInvocation_static() async {
addTestFile(r'''
const c = A.m();
class A {
static int m() => 0;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var invocation = findNode.methodInvocation('m();');
assertType(invocation, 'int');
assertInvokeType(invocation, '() → int');
assertElement(invocation.methodName, findElement.method('m'));
}
test_invalid_const_methodInvocation_topLevelFunction() async {
addTestFile(r'''
const id = identical;
const a = 0;
const b = 0;
const c = id(a, b);
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var invocation = findNode.methodInvocation('id(');
assertType(invocation, 'bool');
assertInvokeType(invocation, '(Object, Object) → bool');
assertElement(invocation.methodName, findElement.topGet('id'));
var aRef = invocation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
var bRef = invocation.argumentList.arguments[1];
assertElement(bRef, findElement.topGet('b'));
assertType(bRef, 'int');
}
test_invalid_const_throw_local() async {
addTestFile(r'''
main() {
const c = throw 42;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var throwExpression = findNode.throw_('throw 42;');
expect(throwExpression.staticType, isBottomType);
assertType(throwExpression.expression, 'int');
}
test_invalid_const_throw_topLevel() async {
addTestFile(r'''
const c = throw 42;
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var throwExpression = findNode.throw_('throw 42;');
expect(throwExpression.staticType, isBottomType);
assertType(throwExpression.expression, 'int');
}
test_invalid_constructor_initializer_field_class() async {
addTestFile(r'''
var a = 0;
class A {
A() : X = a;
}
class X {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.simple('X = ');
assertElementNull(xRef);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_constructor_initializer_field_getter() async {
addTestFile(r'''
var a = 0;
class A {
A() : x = a;
int get x => 0;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.simple('x = ');
assertElement(xRef, findElement.field('x'));
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_constructor_initializer_field_importPrefix() async {
addTestFile(r'''
import 'dart:async' as x;
var a = 0;
class A {
A() : x = a;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.simple('x = ');
assertElementNull(xRef);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_constructor_initializer_field_method() async {
addTestFile(r'''
var a = 0;
class A {
A() : x = a;
void x() {}
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.simple('x = ');
assertElementNull(xRef);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_constructor_initializer_field_setter() async {
addTestFile(r'''
var a = 0;
class A {
A() : x = a;
set x(_) {}
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.simple('x = ');
assertElement(xRef, findElement.field('x'));
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_constructor_initializer_field_topLevelFunction() async {
addTestFile(r'''
var a = 0;
class A {
A() : x = a;
}
void x() {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.simple('x = ');
assertElementNull(xRef);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_constructor_initializer_field_topLevelVar() async {
addTestFile(r'''
var a = 0;
class A {
A() : x = a;
}
int x;
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.simple('x = ');
assertElementNull(xRef);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_constructor_initializer_field_typeParameter() async {
addTestFile(r'''
var a = 0;
class A<T> {
A() : T = a;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var tRef = findNode.simple('T = ');
assertElementNull(tRef);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_constructor_initializer_field_unresolved() async {
addTestFile(r'''
var a = 0;
class A {
A() : x = a;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_constructor_return_blockBody() async {
addTestFile(r'''
int a = 0;
class C {
C() {
return a;
}
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
assertTopGetRef('a;', 'a');
}
test_invalid_constructor_return_expressionBody() async {
addTestFile(r'''
int a = 0;
class C {
C() => a;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
assertTopGetRef('a;', 'a');
}
test_invalid_deferred_type_localVariable() async {
addTestFile(r'''
import 'dart:async' deferred as a;
main() {
a.Future<int> v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
assertTypeName(
findNode.typeName('a.Future'),
futureElement,
'Future<int>',
expectedPrefix: findElement.import('dart:async').prefix,
);
assertTypeName(findNode.typeName('int>'), intElement, 'int');
}
test_invalid_fieldInitializer_field() async {
addTestFile(r'''
class C {
final int a = 0;
final int b = a + 1;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a + 1');
assertElement(aRef, findElement.getter('a'));
assertType(aRef, 'int');
}
test_invalid_fieldInitializer_getter() async {
addTestFile(r'''
class C {
int get a => 0;
final int b = a + 1;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a + 1');
assertElement(aRef, findElement.getter('a'));
assertType(aRef, 'int');
}
test_invalid_fieldInitializer_method() async {
addTestFile(r'''
class C {
int a() => 0;
final int b = a + 1;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a + 1');
assertElement(aRef, findElement.method('a'));
assertType(aRef, '() → int');
}
test_invalid_fieldInitializer_this() async {
addTestFile(r'''
class C {
final b = this;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var thisRef = findNode.this_('this');
assertType(thisRef, 'C');
}
test_invalid_generator_async_return_blockBody() async {
addTestFile(r'''
int a = 0;
f() async* {
return a;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_generator_async_return_expressionBody() async {
addTestFile(r'''
int a = 0;
f() async* => a;
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_generator_sync_return_blockBody() async {
addTestFile(r'''
int a = 0;
f() sync* {
return a;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_generator_sync_return_expressionBody() async {
addTestFile(r'''
int a = 0;
f() sync* => a;
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_getter_parameters() async {
addTestFile(r'''
get m(int a, double b) {
a;
b;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('a;');
assertElement(aRef, findElement.parameter('a'));
assertType(aRef, 'int');
var bRef = findNode.simple('b;');
assertElement(bRef, findElement.parameter('b'));
assertType(bRef, 'double');
}
@failingTest
test_invalid_instanceCreation_abstract() async {
addTestFile(r'''
abstract class C<T> {
C(T a);
C.named(T a);
C.named2();
}
var a = 0;
var b = true;
main() {
new C(a);
new C.named(b);
new C<double>.named2();
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var c = findElement.class_('C');
{
var creation = findNode.instanceCreation('new C(a)');
assertType(creation, 'C<int>');
ConstructorName constructorName = creation.constructorName;
expect(constructorName.name, isNull);
TypeName type = constructorName.type;
expect(type.typeArguments, isNull);
assertElement(type.name, c);
assertTypeNull(type.name);
SimpleIdentifier aRef = creation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
{
var creation = findNode.instanceCreation('new C.named(b)');
assertType(creation, 'C<bool>');
ConstructorName constructorName = creation.constructorName;
expect(constructorName.name.name, 'named');
TypeName type = constructorName.type;
expect(type.typeArguments, isNull);
assertElement(type.name, c);
assertType(type.name, 'C<bool>');
SimpleIdentifier bRef = creation.argumentList.arguments[0];
assertElement(bRef, findElement.topGet('b'));
assertType(bRef, 'bool');
}
{
var creation = findNode.instanceCreation('new C<double>.named2()');
assertType(creation, 'C<double>');
ConstructorName constructorName = creation.constructorName;
expect(constructorName.name.name, 'named2');
TypeName type = constructorName.type;
assertTypeArguments(type.typeArguments, [doubleType]);
assertElement(type.name, c);
assertType(type.name, 'C<double>');
}
}
test_invalid_instanceCreation_arguments_named() async {
addTestFile(r'''
class C {
C();
}
var a = 0;
main() {
new C(x: a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var classElement = findElement.class_('C');
var creation = findNode.instanceCreation('new C(x: a)');
_assertConstructorInvocation(creation, classElement);
NamedExpression argument = creation.argumentList.arguments[0];
assertElementNull(argument.name.label);
var aRef = argument.expression;
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_instanceCreation_arguments_required_01() async {
addTestFile(r'''
class C {
C();
}
var a = 0;
main() {
new C(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var classElement = findElement.class_('C');
var creation = findNode.instanceCreation('new C(a)');
_assertConstructorInvocation(creation, classElement);
var aRef = creation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_instanceCreation_arguments_required_21() async {
addTestFile(r'''
class C {
C(a, b);
}
var a = 0;
main() {
new C(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var classElement = findElement.class_('C');
var creation = findNode.instanceCreation('new C(a)');
_assertConstructorInvocation(creation, classElement);
var aRef = creation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_instanceCreation_constOfNotConst_factory() async {
addTestFile(r'''
class C {
factory C(x) => null;
}
var a = 0;
main() {
const C(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var classElement = findElement.class_('C');
var creation = findNode.instanceCreation('const C(a)');
_assertConstructorInvocation(creation, classElement);
var aRef = creation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_instanceCreation_constOfNotConst_generative() async {
addTestFile(r'''
class C {
C(x);
}
var a = 0;
main() {
const C(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var classElement = findElement.class_('C');
var creation = findNode.instanceCreation('const C(a)');
_assertConstructorInvocation(creation, classElement);
var aRef = creation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
@failingTest
test_invalid_instanceCreation_prefixAsType() async {
addTestFile(r'''
import 'dart:math' as p;
int a;
main() {
new p(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
ImportElement import = findNode.import('dart:math').element;
var pRef = findNode.simple('p(a)');
assertElement(pRef, import.prefix);
assertTypeDynamic(pRef);
var aRef = findNode.simple('a);');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_invocation_arguments_instance_method() async {
addTestFile(r'''
class C {
void m() {}
}
var a = 0;
main(C c) {
c.m(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var m = findElement.method('m');
var invocation = findNode.methodInvocation('m(a)');
assertElement(invocation.methodName, m);
assertType(invocation.methodName, '() → void');
assertType(invocation, 'void');
var aRef = invocation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_invocation_arguments_named_duplicate2() async {
addTestFile(r'''
void f({p}) {}
int a, b;
main() {
f(p: a, p: b);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var f = findElement.function('f');
var invocation = findNode.methodInvocation('f(p: a');
assertElement(invocation.methodName, f);
assertType(invocation.methodName, '({p: dynamic}) → void');
assertType(invocation, 'void');
NamedExpression arg0 = invocation.argumentList.arguments[0];
assertElement(arg0.name.label, f.parameters[0]);
assertIdentifierTopGetRef(arg0.expression, 'a');
NamedExpression arg1 = invocation.argumentList.arguments[1];
assertElement(arg1.name.label, f.parameters[0]);
assertIdentifierTopGetRef(arg1.expression, 'b');
}
test_invalid_invocation_arguments_named_duplicate3() async {
addTestFile(r'''
void f({p}) {}
int a, b, c;
main() {
f(p: a, p: b, p: c);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var f = findElement.function('f');
var invocation = findNode.methodInvocation('f(p: a');
assertElement(invocation.methodName, f);
assertType(invocation.methodName, '({p: dynamic}) → void');
assertType(invocation, 'void');
NamedExpression arg0 = invocation.argumentList.arguments[0];
assertElement(arg0.name.label, f.parameters[0]);
assertIdentifierTopGetRef(arg0.expression, 'a');
NamedExpression arg1 = invocation.argumentList.arguments[1];
assertElement(arg1.name.label, f.parameters[0]);
assertIdentifierTopGetRef(arg1.expression, 'b');
NamedExpression arg2 = invocation.argumentList.arguments[2];
assertElement(arg2.name.label, f.parameters[0]);
assertIdentifierTopGetRef(arg2.expression, 'c');
}
test_invalid_invocation_arguments_requiredAfterNamed() async {
addTestFile(r'''
var a = 0;
var b = 0;
main() {
f(p: a, b);
}
void f({p}) {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
assertTopGetRef('a, ', 'a');
assertTopGetRef('b);', 'b');
}
test_invalid_invocation_arguments_static_method() async {
addTestFile(r'''
class C {
static void m() {}
}
var a = 0;
main() {
C.m(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var m = findElement.method('m');
var invocation = findNode.methodInvocation('m(a)');
assertElement(invocation.methodName, m);
assertType(invocation.methodName, '() → void');
assertType(invocation, 'void');
var aRef = invocation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_invocation_arguments_static_redirectingConstructor() async {
addTestFile(r'''
class C {
factory C() = C.named;
C.named();
}
int a;
main() {
new C(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var classElement = findElement.class_('C');
var creation = findNode.instanceCreation('new C(a)');
_assertConstructorInvocation(creation, classElement);
var aRef = creation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_invocation_arguments_static_topLevelFunction() async {
addTestFile(r'''
void f() {}
var a = 0;
main() {
f(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var f = findElement.function('f');
var invocation = findNode.methodInvocation('f(a)');
assertElement(invocation.methodName, f);
assertType(invocation.methodName, '() → void');
assertType(invocation, 'void');
var aRef = invocation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_invocation_arguments_static_topLevelVariable() async {
addTestFile(r'''
void Function() f;
var a = 0;
main() {
f(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var f = findElement.topGet('f');
var invocation = findNode.methodInvocation('f(a)');
assertElement(invocation.methodName, f);
assertType(invocation.methodName, '() → void');
assertType(invocation, 'void');
var aRef = invocation.argumentList.arguments[0];
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_invocation_prefixAsMethodName() async {
addTestFile(r'''
import 'dart:math' as p;
int a;
main() {
p(a);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
ImportElement import = findNode.import('dart:math').element;
var invocation = findNode.methodInvocation('p(a)');
expect(invocation.staticType, isDynamicType);
assertUnresolvedInvokeType(invocation.staticInvokeType);
var pRef = invocation.methodName;
assertElement(pRef, import.prefix);
assertTypeDynamic(pRef);
var aRef = findNode.simple('a);');
assertElement(aRef, findElement.topGet('a'));
assertType(aRef, 'int');
}
test_invalid_methodInvocation_simpleIdentifier() async {
addTestFile(r'''
int foo = 0;
main() {
foo(1);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
TopLevelVariableElement foo = _getTopLevelVariable(result, 'foo');
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.staticType, isDynamicType);
assertUnresolvedInvokeType(invocation.staticInvokeType);
SimpleIdentifier name = invocation.methodName;
expect(name.staticElement, same(foo.getter));
expect(name.staticType, typeProvider.intType);
}
@failingTest
test_invalid_nonTypeAsType_class_constructor() async {
addTestFile(r'''
class A {
A.T();
}
main() {
A.T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('A.T v;');
assertElement(aRef, findElement.class_('A'));
assertTypeDynamic(aRef);
var tRef = findNode.simple('T v;');
assertElement(tRef, null);
assertTypeDynamic(tRef);
}
@failingTest
test_invalid_nonTypeAsType_class_instanceField() async {
addTestFile(r'''
class A {
int T;
}
main() {
A.T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('A.T v;');
assertElement(aRef, findElement.class_('A'));
assertTypeDynamic(aRef);
var tRef = findNode.simple('T v;');
assertElement(tRef, null);
assertTypeDynamic(tRef);
}
@failingTest
test_invalid_nonTypeAsType_class_instanceMethod() async {
addTestFile(r'''
class A {
int T() => 0;
}
main() {
A.T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('A.T v;');
assertElement(aRef, findElement.class_('A'));
assertTypeDynamic(aRef);
var tRef = findNode.simple('T v;');
assertElement(tRef, null);
assertTypeDynamic(tRef);
}
@failingTest
test_invalid_nonTypeAsType_class_staticField() async {
addTestFile(r'''
class A {
static int T;
}
main() {
A.T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('A.T v;');
assertElement(aRef, findElement.class_('A'));
assertTypeDynamic(aRef);
var tRef = findNode.simple('T v;');
assertElement(tRef, null);
assertTypeDynamic(tRef);
}
@failingTest
test_invalid_nonTypeAsType_class_staticMethod() async {
addTestFile(r'''
class A {
static int T() => 0;
}
main() {
A.T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('A.T v;');
assertElement(aRef, findElement.class_('A'));
assertTypeDynamic(aRef);
var tRef = findNode.simple('T v;');
assertElement(tRef, null);
assertTypeDynamic(tRef);
}
@failingTest
test_invalid_nonTypeAsType_topLevelFunction() async {
addTestFile(r'''
int T() => 0;
main() {
T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var tRef = findNode.simple('T v;');
assertElement(tRef, findElement.topFunction('T'));
assertTypeDynamic(tRef);
}
@failingTest
test_invalid_nonTypeAsType_topLevelFunction_prefixed() async {
newFile('/test/lib/a.dart', content: r'''
int T() => 0;
''');
addTestFile(r'''
import 'a.dart' as p;
main() {
p.T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
ImportElement import = findNode.import('a.dart').element;
var tElement = import.importedLibrary.publicNamespace.get('T');
var prefixedName = findNode.prefixed('p.T');
assertTypeDynamic(prefixedName);
var pRef = prefixedName.prefix;
assertElement(pRef, import.prefix);
expect(pRef.staticType, null);
var tRef = prefixedName.identifier;
assertElement(tRef, tElement);
assertTypeDynamic(tRef);
TypeName typeName = prefixedName.parent;
expect(typeName.type, isDynamicType);
}
@failingTest
test_invalid_nonTypeAsType_topLevelVariable() async {
addTestFile(r'''
int T;
main() {
T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var tRef = findNode.simple('T v;');
assertElement(tRef, findElement.topGet('T'));
assertTypeDynamic(tRef);
}
@failingTest
test_invalid_nonTypeAsType_topLevelVariable_name() async {
addTestFile(r'''
int A;
main() {
A.T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var aRef = findNode.simple('A.T v;');
assertElement(aRef, findElement.topGet('A'));
assertTypeDynamic(aRef);
var tRef = findNode.simple('T v;');
assertElement(tRef, null);
assertTypeDynamic(tRef);
}
@failingTest
test_invalid_nonTypeAsType_topLevelVariable_prefixed() async {
newFile('/test/lib/a.dart', content: r'''
int T;
''');
addTestFile(r'''
import 'a.dart' as p;
main() {
p.T v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
ImportElement import = findNode.import('a.dart').element;
var tElement = import.importedLibrary.publicNamespace.get('T');
var prefixedName = findNode.prefixed('p.T');
assertTypeDynamic(prefixedName);
var pRef = prefixedName.prefix;
assertElement(pRef, import.prefix);
expect(pRef.staticType, null);
var tRef = prefixedName.identifier;
assertElement(tRef, tElement);
assertTypeDynamic(tRef);
TypeName typeName = prefixedName.parent;
expect(typeName.type, isDynamicType);
}
@failingTest
test_invalid_nonTypeAsType_typeParameter_name() async {
addTestFile(r'''
main<T>() {
T.U v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var tRef = findNode.simple('T.U v;');
var tElement = findNode.typeParameter('T>()').declaredElement;
assertElement(tRef, tElement);
assertTypeDynamic(tRef);
}
@failingTest
test_invalid_nonTypeAsType_unresolved_name() async {
addTestFile(r'''
main() {
T.U v;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var tRef = findNode.simple('T.U v;');
assertElementNull(tRef);
assertTypeDynamic(tRef);
}
test_invalid_rethrow() async {
addTestFile('''
main() {
rethrow;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var rethrowExpression = findNode.rethrow_('rethrow;');
expect(rethrowExpression.staticType, isBottomType);
}
test_invalid_tryCatch_1() async {
addTestFile(r'''
main() {
try {}
catch String catch (e) {}
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
}
test_invalid_tryCatch_2() async {
addTestFile(r'''
main() {
try {}
catch catch (e) {}
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
}
test_isExpression() async {
String content = r'''
void main() {
var v = 42;
v is num;
}
''';
addTestFile(content);
await resolveTestFile();
expect(result.path, testFile);
expect(result.errors, isEmpty);
NodeList<Statement> statements = _getMainStatements(result);
// var v = 42;
VariableElement vElement;
{
VariableDeclarationStatement statement = statements[0];
vElement = statement.variables.variables[0].name.staticElement;
}
// v is num;
{
ExpressionStatement statement = statements[1];
IsExpression isExpression = statement.expression;
expect(isExpression.notOperator, isNull);
expect(isExpression.staticType, typeProvider.boolType);
SimpleIdentifier target = isExpression.expression;
expect(target.staticElement, vElement);
expect(target.staticType, typeProvider.intType);
TypeName numName = isExpression.type;
expect(numName.name.staticElement, typeProvider.numType.element);
expect(numName.name.staticType, typeProvider.numType);
}
}
test_isExpression_not() async {
String content = r'''
void main() {
var v = 42;
v is! num;
}
''';
addTestFile(content);
await resolveTestFile();
expect(result.path, testFile);
expect(result.errors, isEmpty);
NodeList<Statement> statements = _getMainStatements(result);
// var v = 42;
VariableElement vElement;
{
VariableDeclarationStatement statement = statements[0];
vElement = statement.variables.variables[0].name.staticElement;
}
// v is! num;
{
ExpressionStatement statement = statements[1];
IsExpression isExpression = statement.expression;
expect(isExpression.notOperator, isNotNull);
expect(isExpression.staticType, typeProvider.boolType);
SimpleIdentifier target = isExpression.expression;
expect(target.staticElement, vElement);
expect(target.staticType, typeProvider.intType);
TypeName numName = isExpression.type;
expect(numName.name.staticElement, typeProvider.numType.element);
expect(numName.name.staticType, typeProvider.numType);
}
}
test_label_while() async {
addTestFile(r'''
main() {
myLabel:
while (true) {
continue myLabel;
break myLabel;
}
}
''');
await resolveTestFile();
List<Statement> statements = _getMainStatements(result);
LabeledStatement statement = statements[0];
Label label = statement.labels.single;
LabelElement labelElement = label.label.staticElement;
WhileStatement whileStatement = statement.statement;
Block whileBlock = whileStatement.body;
ContinueStatement continueStatement = whileBlock.statements[0];
expect(continueStatement.label.staticElement, same(labelElement));
expect(continueStatement.label.staticType, isNull);
BreakStatement breakStatement = whileBlock.statements[1];
expect(breakStatement.label.staticElement, same(labelElement));
expect(breakStatement.label.staticType, isNull);
}
test_listLiteral_01() async {
addTestFile(r'''
main() {
var v = [];
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var literal = findNode.listLiteral('[];');
expect(literal.typeArguments, isNull);
assertType(literal, 'List<dynamic>');
}
test_listLiteral_02() async {
addTestFile(r'''
main() {
var v = <>[];
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var literal = findNode.listLiteral('[];');
expect(literal.typeArguments, isNotNull);
assertType(literal, 'List<dynamic>');
}
test_listLiteral_2() async {
addTestFile(r'''
main() {
var v = <int, double>[];
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var literal = findNode.listLiteral('<int, double>[]');
assertType(literal, 'List<dynamic>');
var intRef = findNode.simple('int, double');
assertElement(intRef, intElement);
assertType(intRef, 'int');
var doubleRef = findNode.simple('double>[]');
assertElement(doubleRef, doubleElement);
assertType(doubleRef, 'double');
}
test_local_function() async {
addTestFile(r'''
void main() {
double f(int a, String b) {}
var v = f(1, '2');
}
''');
String fTypeString = '(int, String) → double';
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
InterfaceType doubleType = typeProvider.doubleType;
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
FunctionExpression fExpression = fNode.functionExpression;
FunctionElement fElement = fNode.declaredElement;
expect(fElement, isNotNull);
expect(fElement.type.toString(), fTypeString);
expect(fNode.name.staticElement, same(fElement));
expect(fNode.name.staticType, fElement.type);
TypeName fReturnTypeNode = fNode.returnType;
expect(fReturnTypeNode.name.staticElement, same(doubleType.element));
expect(fReturnTypeNode.type, doubleType);
expect(fExpression.declaredElement, same(fElement));
{
List<ParameterElement> elements = fElement.parameters;
expect(elements, hasLength(2));
List<FormalParameter> nodes = fExpression.parameters.parameters;
expect(nodes, hasLength(2));
_assertSimpleParameter(nodes[0], elements[0],
name: 'a',
offset: 29,
kind: ParameterKind.REQUIRED,
type: typeProvider.intType);
_assertSimpleParameter(nodes[1], elements[1],
name: 'b',
offset: 39,
kind: ParameterKind.REQUIRED,
type: typeProvider.stringType);
}
VariableDeclarationStatement vStatement = mainStatements[1];
VariableDeclaration vDeclaration = vStatement.variables.variables[0];
expect(vDeclaration.declaredElement.type, doubleType);
MethodInvocation fInvocation = vDeclaration.initializer;
expect(fInvocation.methodName.staticElement, same(fElement));
expect(fInvocation.methodName.staticType.toString(), fTypeString);
expect(fInvocation.staticType, doubleType);
expect(fInvocation.staticInvokeType.toString(), fTypeString);
}
test_local_function_call_with_incomplete_closure_argument() async {
addTestFile('''
void main() {
f(x) => null;
f(=> 42);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
// Note: no further expectations. We don't care how the error is recovered
// from, provided it is recovered from in a way that doesn't crash the
// analyzer/FE integration.
}
@failingTest
test_local_function_generic() async {
addTestFile(r'''
void main() {
T f<T, U>(T a, U b) {
a;
b;
}
var v = f(1, '2');
}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
FunctionExpression fExpression = fNode.functionExpression;
FunctionElement fElement = fNode.declaredElement;
TypeParameterElement tElement = fElement.typeParameters[0];
TypeParameterElement uElement = fElement.typeParameters[1];
{
var fTypeParameters = fExpression.typeParameters.typeParameters;
expect(fTypeParameters, hasLength(2));
TypeParameter tNode = fTypeParameters[0];
expect(tNode.declaredElement, same(tElement));
expect(tNode.name.staticElement, same(tElement));
expect(tNode.name.staticType, typeProvider.typeType);
TypeParameter uNode = fTypeParameters[1];
expect(uNode.declaredElement, same(uElement));
expect(uNode.name.staticElement, same(uElement));
expect(uNode.name.staticType, typeProvider.typeType);
}
expect(fElement, isNotNull);
expect(fElement.type.toString(), '<T,U>(T, U) → T');
expect(fNode.name.staticElement, same(fElement));
expect(fNode.name.staticType, fElement.type);
TypeName fReturnTypeNode = fNode.returnType;
expect(fReturnTypeNode.name.staticElement, same(tElement));
expect(fReturnTypeNode.type, tElement.type);
expect(fExpression.declaredElement, same(fElement));
{
List<ParameterElement> parameters = fElement.parameters;
expect(parameters, hasLength(2));
List<FormalParameter> nodes = fExpression.parameters.parameters;
expect(nodes, hasLength(2));
_assertSimpleParameter(nodes[0], parameters[0],
name: 'a',
offset: 28,
kind: ParameterKind.REQUIRED,
type: tElement.type);
_assertSimpleParameter(nodes[1], parameters[1],
name: 'b',
offset: 33,
kind: ParameterKind.REQUIRED,
type: uElement.type);
var aRef = findNode.simple('a;');
assertElement(aRef, parameters[0]);
assertType(aRef, 'T');
var bRef = findNode.simple('b;');
assertElement(bRef, parameters[1]);
assertType(bRef, 'U');
}
VariableDeclarationStatement vStatement = mainStatements[1];
VariableDeclaration vDeclaration = vStatement.variables.variables[0];
expect(vDeclaration.declaredElement.type, typeProvider.intType);
MethodInvocation fInvocation = vDeclaration.initializer;
expect(fInvocation.methodName.staticElement, same(fElement));
expect(fInvocation.staticType, typeProvider.intType);
assertTypeNull(fInvocation.methodName);
expect(fInvocation.staticInvokeType.toString(), '(int, String) → int');
}
test_local_function_generic_f_bounded() async {
addTestFile('''
void main() {
void F<T extends U, U, V extends U>(T x, U y, V z) {}
}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
FunctionElement fElement = fNode.declaredElement;
expect(fElement.type.toString(),
'<T extends U,U,V extends U>(T, U, V) → void');
var tElement = fElement.typeParameters[0];
var uElement = fElement.typeParameters[1];
var vElement = fElement.typeParameters[2];
expect((tElement.bound as TypeParameterType).element, same(uElement));
expect((vElement.bound as TypeParameterType).element, same(uElement));
}
test_local_function_generic_with_named_parameter() async {
addTestFile('''
void main() {
void F<T>({T x}) {}
}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
FunctionElement fElement = fNode.declaredElement;
expect(fElement.type.toString(), '<T>({x: T}) → void');
var tElement = fElement.typeParameters[0];
expect(fElement.type.typeFormals[0], same(tElement));
expect((fElement.type.parameters[0].type as TypeParameterType).element,
same(tElement));
}
test_local_function_generic_with_optional_parameter() async {
addTestFile('''
void main() {
void F<T>([T x]) {}
}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
FunctionElement fElement = fNode.declaredElement;
expect(fElement.type.toString(), '<T>([T]) → void');
var tElement = fElement.typeParameters[0];
expect(fElement.type.typeFormals[0], same(tElement));
expect((fElement.type.parameters[0].type as TypeParameterType).element,
same(tElement));
}
test_local_function_namedParameters() async {
addTestFile(r'''
void main() {
double f(int a, {String b, bool c: false}) {}
f(1, b: '2', c: true);
}
''');
String fTypeString = '(int, {b: String, c: bool}) → double';
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
InterfaceType doubleType = typeProvider.doubleType;
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
FunctionExpression fExpression = fNode.functionExpression;
FunctionElement fElement = fNode.declaredElement;
expect(fElement, isNotNull);
expect(fElement.type.toString(), fTypeString);
expect(fNode.name.staticElement, same(fElement));
expect(fNode.name.staticType, fElement.type);
TypeName fReturnTypeNode = fNode.returnType;
expect(fReturnTypeNode.name.staticElement, same(doubleType.element));
expect(fReturnTypeNode.type, doubleType);
expect(fExpression.declaredElement, same(fElement));
{
List<ParameterElement> elements = fElement.parameters;
expect(elements, hasLength(3));
List<FormalParameter> nodes = fExpression.parameters.parameters;
expect(nodes, hasLength(3));
_assertSimpleParameter(nodes[0], elements[0],
name: 'a',
offset: 29,
kind: ParameterKind.REQUIRED,
type: typeProvider.intType);
_assertDefaultParameter(nodes[1], elements[1],
name: 'b',
offset: 40,
kind: ParameterKind.NAMED,
type: typeProvider.stringType);
_assertDefaultParameter(nodes[2], elements[2],
name: 'c',
offset: 48,
kind: ParameterKind.NAMED,
type: typeProvider.boolType);
}
{
ExpressionStatement statement = mainStatements[1];
MethodInvocation invocation = statement.expression;
List<Expression> arguments = invocation.argumentList.arguments;
_assertArgumentToParameter(arguments[0], fElement.parameters[0]);
_assertArgumentToParameter(arguments[1], fElement.parameters[1]);
_assertArgumentToParameter(arguments[2], fElement.parameters[2]);
}
}
test_local_function_noReturnType() async {
addTestFile(r'''
void main() {
f() {}
}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
FunctionExpression fExpression = fNode.functionExpression;
FunctionElement fElement = fNode.declaredElement;
expect(fNode.returnType, isNull);
expect(fElement, isNotNull);
expect(fElement.type.toString(), '() → Null');
expect(fNode.name.staticElement, same(fElement));
expect(fNode.name.staticType, fElement.type);
expect(fExpression.declaredElement, same(fElement));
}
test_local_function_optionalParameters() async {
addTestFile(r'''
void main() {
double f(int a, [String b, bool c]) {}
var v = f(1, '2', true);
}
''');
String fTypeString = '(int, [String, bool]) → double';
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
InterfaceType doubleType = typeProvider.doubleType;
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
FunctionExpression fExpression = fNode.functionExpression;
FunctionElement fElement = fNode.declaredElement;
expect(fElement, isNotNull);
expect(fElement.type.toString(), fTypeString);
expect(fNode.name.staticElement, same(fElement));
expect(fNode.name.staticType, fElement.type);
TypeName fReturnTypeNode = fNode.returnType;
expect(fReturnTypeNode.name.staticElement, same(doubleType.element));
expect(fReturnTypeNode.type, doubleType);
expect(fExpression.declaredElement, same(fElement));
{
List<ParameterElement> elements = fElement.parameters;
expect(elements, hasLength(3));
List<FormalParameter> nodes = fExpression.parameters.parameters;
expect(nodes, hasLength(3));
_assertSimpleParameter(nodes[0], elements[0],
name: 'a',
offset: 29,
kind: ParameterKind.REQUIRED,
type: typeProvider.intType);
_assertDefaultParameter(nodes[1], elements[1],
name: 'b',
offset: 40,
kind: ParameterKind.POSITIONAL,
type: typeProvider.stringType);
_assertDefaultParameter(nodes[2], elements[2],
name: 'c',
offset: 48,
kind: ParameterKind.POSITIONAL,
type: typeProvider.boolType);
}
{
VariableDeclarationStatement statement = mainStatements[1];
VariableDeclaration declaration = statement.variables.variables[0];
expect(declaration.declaredElement.type, doubleType);
MethodInvocation invocation = declaration.initializer;
expect(invocation.methodName.staticElement, same(fElement));
expect(invocation.methodName.staticType.toString(), fTypeString);
expect(invocation.staticType, doubleType);
expect(invocation.staticInvokeType.toString(), fTypeString);
List<Expression> arguments = invocation.argumentList.arguments;
_assertArgumentToParameter(arguments[0], fElement.parameters[0]);
_assertArgumentToParameter(arguments[1], fElement.parameters[1]);
_assertArgumentToParameter(arguments[2], fElement.parameters[2]);
}
}
test_local_function_with_function_typed_parameter() async {
addTestFile('''
class C {}
class D {}
class E {}
void f() {
void g(C callback<T extends E>(D d)) {}
}
''');
await resolveTestFile();
var callback = findNode.simple('callback');
assertType(callback, '<T extends E>(D) → C');
var cReference = findNode.simple('C callback');
var cElement = findElement.class_('C');
assertType(cReference, 'C');
assertElement(cReference, cElement);
var dReference = findNode.simple('D d');
var dElement = findElement.class_('D');
assertType(dReference, 'D');
assertElement(dReference, dElement);
var eReference = findNode.simple('E>');
var eElement = findElement.class_('E');
assertType(eReference, 'E');
assertElement(eReference, eElement);
}
test_local_parameter() async {
String content = r'''
void main(int p) {
p;
}
''';
addTestFile(content);
await resolveTestFile();
expect(result.path, testFile);
expect(result.errors, isEmpty);
InterfaceType intType = typeProvider.intType;
FunctionDeclaration main = result.unit.declarations[0];
List<Statement> statements = _getMainStatements(result);
// (int p)
VariableElement pElement = main.declaredElement.parameters[0];
expect(pElement.type, intType);
// p;
{
ExpressionStatement statement = statements[0];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, pElement);
expect(identifier.staticType, intType);
}
}
test_local_parameter_ofLocalFunction() async {
addTestFile(r'''
void main() {
void f(int a) {
a;
void g(double b) {
b;
}
}
}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
// f(int a) {}
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
FunctionExpression fExpression = fNode.functionExpression;
FunctionElement fElement = fNode.declaredElement;
ParameterElement aElement = fElement.parameters[0];
_assertSimpleParameter(fExpression.parameters.parameters[0], aElement,
name: 'a',
offset: 27,
kind: ParameterKind.REQUIRED,
type: typeProvider.intType);
BlockFunctionBody fBody = fExpression.body;
List<Statement> fStatements = fBody.block.statements;
// a;
ExpressionStatement aStatement = fStatements[0];
SimpleIdentifier aNode = aStatement.expression;
expect(aNode.staticElement, same(aElement));
expect(aNode.staticType, typeProvider.intType);
// g(double b) {}
FunctionDeclarationStatement gStatement = fStatements[1];
FunctionDeclaration gNode = gStatement.functionDeclaration;
FunctionExpression gExpression = gNode.functionExpression;
FunctionElement gElement = gNode.declaredElement;
ParameterElement bElement = gElement.parameters[0];
_assertSimpleParameter(gExpression.parameters.parameters[0], bElement,
name: 'b',
offset: 57,
kind: ParameterKind.REQUIRED,
type: typeProvider.doubleType);
BlockFunctionBody gBody = gExpression.body;
List<Statement> gStatements = gBody.block.statements;
// b;
ExpressionStatement bStatement = gStatements[0];
SimpleIdentifier bNode = bStatement.expression;
expect(bNode.staticElement, same(bElement));
expect(bNode.staticType, typeProvider.doubleType);
}
test_local_type_parameter_reference_as_expression() async {
addTestFile('''
void main() {
void f<T>(T x) {
T;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var bodyStatement = body.block.statements[0] as ExpressionStatement;
var tReference = bodyStatement.expression as SimpleIdentifier;
assertElement(tReference, tElement);
assertType(tReference, 'Type');
}
test_local_type_parameter_reference_function_named_parameter_type() async {
addTestFile('''
void main() {
void f<T>(T x) {
void Function({T t}) g = null;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var gDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var gType = gDeclaration.variables.type as GenericFunctionType;
var gTypeType = gType.type as FunctionType;
var gTypeParameterType =
gTypeType.namedParameterTypes['t'] as TypeParameterType;
expect(gTypeParameterType.element, same(tElement));
var gParameterType =
((gType.parameters.parameters[0] as DefaultFormalParameter).parameter
as SimpleFormalParameter)
.type as TypeName;
var tReference = gParameterType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_type_parameter_reference_function_normal_parameter_type() async {
addTestFile('''
void main() {
void f<T>(T x) {
void Function(T) g = null;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var gDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var gType = gDeclaration.variables.type as GenericFunctionType;
var gTypeType = gType.type as FunctionType;
var gTypeParameterType =
gTypeType.normalParameterTypes[0] as TypeParameterType;
expect(gTypeParameterType.element, same(tElement));
var gParameterType =
(gType.parameters.parameters[0] as SimpleFormalParameter).type
as TypeName;
var tReference = gParameterType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_type_parameter_reference_function_optional_parameter_type() async {
addTestFile('''
void main() {
void f<T>(T x) {
void Function([T]) g = null;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var gDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var gType = gDeclaration.variables.type as GenericFunctionType;
var gTypeType = gType.type as FunctionType;
var gTypeParameterType =
gTypeType.optionalParameterTypes[0] as TypeParameterType;
expect(gTypeParameterType.element, same(tElement));
var gParameterType =
((gType.parameters.parameters[0] as DefaultFormalParameter).parameter
as SimpleFormalParameter)
.type as TypeName;
var tReference = gParameterType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_type_parameter_reference_function_return_type() async {
addTestFile('''
void main() {
void f<T>(T x) {
T Function() g = () => x;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var gDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var gType = gDeclaration.variables.type as GenericFunctionType;
var gTypeType = gType.type as FunctionType;
var gTypeReturnType = gTypeType.returnType as TypeParameterType;
expect(gTypeReturnType.element, same(tElement));
var gReturnType = gType.returnType as TypeName;
var tReference = gReturnType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_type_parameter_reference_interface_type_parameter() async {
addTestFile('''
void main() {
void f<T>(T x) {
List<T> y = [x];
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var yDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var yType = yDeclaration.variables.type as TypeName;
var yTypeType = yType.type as InterfaceType;
var yTypeTypeArgument = yTypeType.typeArguments[0] as TypeParameterType;
expect(yTypeTypeArgument.element, same(tElement));
var yElementType = yType.typeArguments.arguments[0] as TypeName;
var tReference = yElementType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_type_parameter_reference_simple() async {
addTestFile('''
void main() {
void f<T>(T x) {
T y = x;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var yDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var yType = yDeclaration.variables.type as TypeName;
var tReference = yType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_type_parameter_reference_typedef_named_parameter_type() async {
addTestFile('''
typedef void Consumer<U>({U u});
void main() {
void f<T>(T x) {
Consumer<T> g = null;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var gDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var gType = gDeclaration.variables.type as TypeName;
var gTypeType = gType.type as FunctionType;
var gTypeTypeArgument = gTypeType.typeArguments[0] as TypeParameterType;
expect(gTypeTypeArgument.element, same(tElement));
var gTypeParameterType =
gTypeType.namedParameterTypes['u'] as TypeParameterType;
expect(gTypeParameterType.element, same(tElement));
var gArgumentType = gType.typeArguments.arguments[0] as TypeName;
var tReference = gArgumentType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_type_parameter_reference_typedef_normal_parameter_type() async {
addTestFile('''
typedef void Consumer<U>(U u);
void main() {
void f<T>(T x) {
Consumer<T> g = null;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var gDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var gType = gDeclaration.variables.type as TypeName;
var gTypeType = gType.type as FunctionType;
var gTypeTypeArgument = gTypeType.typeArguments[0] as TypeParameterType;
expect(gTypeTypeArgument.element, same(tElement));
var gTypeParameterType =
gTypeType.normalParameterTypes[0] as TypeParameterType;
expect(gTypeParameterType.element, same(tElement));
var gArgumentType = gType.typeArguments.arguments[0] as TypeName;
var tReference = gArgumentType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_type_parameter_reference_typedef_optional_parameter_type() async {
addTestFile('''
typedef void Consumer<U>([U u]);
void main() {
void f<T>(T x) {
Consumer<T> g = null;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var gDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var gType = gDeclaration.variables.type as TypeName;
var gTypeType = gType.type as FunctionType;
var gTypeTypeArgument = gTypeType.typeArguments[0] as TypeParameterType;
expect(gTypeTypeArgument.element, same(tElement));
var gTypeParameterType =
gTypeType.optionalParameterTypes[0] as TypeParameterType;
expect(gTypeParameterType.element, same(tElement));
var gArgumentType = gType.typeArguments.arguments[0] as TypeName;
var tReference = gArgumentType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_type_parameter_reference_typedef_return_type() async {
addTestFile('''
typedef U Producer<U>();
void main() {
void f<T>(T x) {
Producer<T> g = () => x;
}
f(1);
}
''');
await resolveTestFile();
var mainStatements = _getMainStatements(result);
var fDeclaration = mainStatements[0] as FunctionDeclarationStatement;
var fElement = fDeclaration.functionDeclaration.declaredElement;
var tElement = fElement.typeParameters[0];
var body = fDeclaration.functionDeclaration.functionExpression.body
as BlockFunctionBody;
var gDeclaration = body.block.statements[0] as VariableDeclarationStatement;
var gType = gDeclaration.variables.type as TypeName;
var gTypeType = gType.type as FunctionType;
var gTypeTypeArgument = gTypeType.typeArguments[0] as TypeParameterType;
expect(gTypeTypeArgument.element, same(tElement));
var gTypeReturnType = gTypeType.returnType as TypeParameterType;
expect(gTypeReturnType.element, same(tElement));
var gArgumentType = gType.typeArguments.arguments[0] as TypeName;
var tReference = gArgumentType.name;
assertElement(tReference, tElement);
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
}
test_local_variable() async {
addTestFile(r'''
void main() {
var v = 42;
v;
}
''');
await resolveTestFile();
expect(result.path, testFile);
expect(result.errors, isEmpty);
InterfaceType intType = typeProvider.intType;
FunctionDeclaration main = result.unit.declarations[0];
expect(main.declaredElement, isNotNull);
expect(main.name.staticElement, isNotNull);
expect(main.name.staticType.toString(), '() → void');
BlockFunctionBody body = main.functionExpression.body;
NodeList<Statement> statements = body.block.statements;
// var v = 42;
VariableElement vElement;
{
VariableDeclarationStatement statement = statements[0];
VariableDeclaration vNode = statement.variables.variables[0];
expect(vNode.name.staticType, intType);
expect(vNode.initializer.staticType, intType);
vElement = vNode.name.staticElement;
expect(vElement, isNotNull);
expect(vElement.type, isNotNull);
expect(vElement.type, intType);
}
// v;
{
ExpressionStatement statement = statements[1];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, same(vElement));
expect(identifier.staticType, intType);
}
}
test_local_variable_forIn_identifier_field() async {
addTestFile(r'''
class C {
num v;
void foo() {
for (v in <int>[]) {
v;
}
}
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
ClassDeclaration cDeclaration = unit.declarations[0];
FieldDeclaration vDeclaration = cDeclaration.members[0];
VariableDeclaration vNode = vDeclaration.fields.variables[0];
FieldElement vElement = vNode.declaredElement;
expect(vElement.type, typeProvider.numType);
MethodDeclaration fooDeclaration = cDeclaration.members[1];
BlockFunctionBody fooBody = fooDeclaration.body;
List<Statement> statements = fooBody.block.statements;
ForStatement forEachStatement = statements[0];
Block forBlock = forEachStatement.body;
var forEachParts =
forEachStatement.forLoopParts as ForEachPartsWithIdentifier;
SimpleIdentifier vInFor = forEachParts.identifier;
expect(vInFor.staticElement, same(vElement.setter));
expect(vInFor.staticType, typeProvider.numType);
ExpressionStatement statement = forBlock.statements[0];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, same(vElement.getter));
expect(identifier.staticType, typeProvider.numType);
}
test_local_variable_forIn_identifier_localVariable() async {
addTestFile(r'''
void main() {
num v;
for (v in <int>[]) {
v;
}
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> statements = _getMainStatements(result);
VariableDeclarationStatement vStatement = statements[0];
VariableDeclaration vNode = vStatement.variables.variables[0];
LocalVariableElement vElement = vNode.declaredElement;
expect(vElement.type, typeProvider.numType);
ForStatement forEachStatement = statements[1];
Block forBlock = forEachStatement.body;
var forEachParts =
forEachStatement.forLoopParts as ForEachPartsWithIdentifier;
SimpleIdentifier vInFor = forEachParts.identifier;
expect(vInFor.staticElement, vElement);
expect(vInFor.staticType, typeProvider.numType);
ExpressionStatement statement = forBlock.statements[0];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, same(vElement));
expect(identifier.staticType, typeProvider.numType);
}
test_local_variable_forIn_identifier_topLevelVariable() async {
addTestFile(r'''
void main() {
for (v in <int>[]) {
v;
}
}
num v;
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> statements = _getMainStatements(result);
TopLevelVariableDeclaration vDeclaration = unit.declarations[1];
VariableDeclaration vNode = vDeclaration.variables.variables[0];
TopLevelVariableElement vElement = vNode.declaredElement;
expect(vElement.type, typeProvider.numType);
ForStatement forEachStatement = statements[0];
Block forBlock = forEachStatement.body;
var forEachParts =
forEachStatement.forLoopParts as ForEachPartsWithIdentifier;
SimpleIdentifier vInFor = forEachParts.identifier;
expect(vInFor.staticElement, same(vElement.setter));
expect(vInFor.staticType, typeProvider.numType);
ExpressionStatement statement = forBlock.statements[0];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, same(vElement.getter));
expect(identifier.staticType, typeProvider.numType);
}
test_local_variable_forIn_loopVariable() async {
addTestFile(r'''
void main() {
for (var v in <int>[]) {
v;
}
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> statements = _getMainStatements(result);
ForStatement forEachStatement = statements[0];
Block forBlock = forEachStatement.body;
var forEachParts =
forEachStatement.forLoopParts as ForEachPartsWithDeclaration;
DeclaredIdentifier vNode = forEachParts.loopVariable;
LocalVariableElement vElement = vNode.declaredElement;
expect(vElement.type, typeProvider.intType);
expect(vNode.identifier.staticElement, vElement);
expect(vNode.identifier.staticType, typeProvider.intType);
ExpressionStatement statement = forBlock.statements[0];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, vElement);
expect(identifier.staticType, typeProvider.intType);
}
test_local_variable_forIn_loopVariable_explicitType() async {
addTestFile(r'''
void main() {
for (num v in <int>[]) {
v;
}
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> statements = _getMainStatements(result);
ForStatement forEachStatement = statements[0];
Block forBlock = forEachStatement.body;
var forEachParts =
forEachStatement.forLoopParts as ForEachPartsWithDeclaration;
DeclaredIdentifier vNode = forEachParts.loopVariable;
LocalVariableElement vElement = vNode.declaredElement;
expect(vElement.type, typeProvider.numType);
TypeName vTypeName = vNode.type;
expect(vTypeName.type, typeProvider.numType);
SimpleIdentifier vTypeIdentifier = vTypeName.name;
expect(vTypeIdentifier.staticElement, typeProvider.numType.element);
expect(vTypeIdentifier.staticType, typeProvider.numType);
expect(vNode.identifier.staticElement, vElement);
expect(vNode.identifier.staticType, typeProvider.numType);
ExpressionStatement statement = forBlock.statements[0];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, vElement);
expect(identifier.staticType, typeProvider.numType);
}
test_local_variable_multiple() async {
addTestFile(r'''
void main() {
var a = 1, b = 2.3;
}
''');
await resolveTestFile();
List<Statement> statements = _getMainStatements(result);
VariableDeclarationStatement declarationStatement = statements[0];
VariableDeclaration aNode = declarationStatement.variables.variables[0];
LocalVariableElement aElement = aNode.declaredElement;
expect(aElement.type, typeProvider.intType);
VariableDeclaration bNode = declarationStatement.variables.variables[1];
LocalVariableElement bElement = bNode.declaredElement;
expect(bElement.type, typeProvider.doubleType);
}
test_local_variable_ofLocalFunction() async {
addTestFile(r'''
void main() {
void f() {
int a;
a;
void g() {
double b;
a;
b;
}
}
}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
// f() {}
FunctionDeclarationStatement fStatement = mainStatements[0];
FunctionDeclaration fNode = fStatement.functionDeclaration;
BlockFunctionBody fBody = fNode.functionExpression.body;
List<Statement> fStatements = fBody.block.statements;
// int a;
VariableDeclarationStatement aDeclaration = fStatements[0];
VariableElement aElement =
aDeclaration.variables.variables[0].declaredElement;
// a;
{
ExpressionStatement aStatement = fStatements[1];
SimpleIdentifier aNode = aStatement.expression;
expect(aNode.staticElement, same(aElement));
expect(aNode.staticType, typeProvider.intType);
}
// g(double b) {}
FunctionDeclarationStatement gStatement = fStatements[2];
FunctionDeclaration gNode = gStatement.functionDeclaration;
BlockFunctionBody gBody = gNode.functionExpression.body;
List<Statement> gStatements = gBody.block.statements;
// double b;
VariableDeclarationStatement bDeclaration = gStatements[0];
VariableElement bElement =
bDeclaration.variables.variables[0].declaredElement;
// a;
{
ExpressionStatement aStatement = gStatements[1];
SimpleIdentifier aNode = aStatement.expression;
expect(aNode.staticElement, same(aElement));
expect(aNode.staticType, typeProvider.intType);
}
// b;
{
ExpressionStatement bStatement = gStatements[2];
SimpleIdentifier bNode = bStatement.expression;
expect(bNode.staticElement, same(bElement));
expect(bNode.staticType, typeProvider.doubleType);
}
}
test_mapLiteral() async {
addTestFile(r'''
void main() {
<int, double>{};
const <bool, String>{};
}
''');
await resolveTestFile();
var statements = _getMainStatements(result);
{
ExpressionStatement statement = statements[0];
SetOrMapLiteral mapLiteral = statement.expression;
expect(
mapLiteral.staticType,
typeProvider.mapType
.instantiate([typeProvider.intType, typeProvider.doubleType]));
}
{
ExpressionStatement statement = statements[1];
SetOrMapLiteral mapLiteral = statement.expression;
expect(
mapLiteral.staticType,
typeProvider.mapType
.instantiate([typeProvider.boolType, typeProvider.stringType]));
}
}
test_mapLiteral_3() async {
addTestFile(r'''
main() {
var v = <bool, int, double>{};
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var literal = findNode.setOrMapLiteral('<bool, int, double>{}');
assertType(literal, 'Map<dynamic, dynamic>');
var boolRef = findNode.simple('bool, ');
assertElement(boolRef, boolElement);
assertType(boolRef, 'bool');
var intRef = findNode.simple('int, ');
assertElement(intRef, intElement);
assertType(intRef, 'int');
var doubleRef = findNode.simple('double>');
assertElement(doubleRef, doubleElement);
assertType(doubleRef, 'double');
}
test_method_namedParameters() async {
addTestFile(r'''
class C {
double f(int a, {String b, bool c: false}) {}
}
void g(C c) {
c.f(1, b: '2', c: true);
}
''');
String fTypeString = '(int, {b: String, c: bool}) → double';
await resolveTestFile();
ClassDeclaration classDeclaration = result.unit.declarations[0];
MethodDeclaration methodDeclaration = classDeclaration.members[0];
MethodElement methodElement = methodDeclaration.declaredElement;
InterfaceType doubleType = typeProvider.doubleType;
expect(methodElement, isNotNull);
expect(methodElement.type.toString(), fTypeString);
expect(methodDeclaration.name.staticElement, same(methodElement));
expect(methodDeclaration.name.staticType, methodElement.type);
TypeName fReturnTypeNode = methodDeclaration.returnType;
expect(fReturnTypeNode.name.staticElement, same(doubleType.element));
expect(fReturnTypeNode.type, doubleType);
//
// Validate the parameters at the declaration site.
//
List<ParameterElement> elements = methodElement.parameters;
expect(elements, hasLength(3));
List<FormalParameter> nodes = methodDeclaration.parameters.parameters;
expect(nodes, hasLength(3));
_assertSimpleParameter(nodes[0], elements[0],
name: 'a',
offset: 25,
kind: ParameterKind.REQUIRED,
type: typeProvider.intType);
_assertDefaultParameter(nodes[1], elements[1],
name: 'b',
offset: 36,
kind: ParameterKind.NAMED,
type: typeProvider.stringType);
_assertDefaultParameter(nodes[2], elements[2],
name: 'c',
offset: 44,
kind: ParameterKind.NAMED,
type: typeProvider.boolType);
//
// Validate the arguments at the call site.
//
FunctionDeclaration functionDeclaration = result.unit.declarations[1];
BlockFunctionBody body = functionDeclaration.functionExpression.body;
ExpressionStatement statement = body.block.statements[0];
MethodInvocation invocation = statement.expression;
List<Expression> arguments = invocation.argumentList.arguments;
_assertArgumentToParameter(arguments[0], methodElement.parameters[0]);
_assertArgumentToParameter(arguments[1], methodElement.parameters[1]);
_assertArgumentToParameter(arguments[2], methodElement.parameters[2]);
}
test_methodInvocation_explicitCall_classTarget() async {
addTestFile(r'''
class C {
double call(int p) => 0.0;
}
main() {
new C().call(0);
}
''');
await resolveTestFile();
expect(result.errors, isEmpty);
ClassDeclaration cNode = result.unit.declarations[0];
ClassElement cElement = cNode.declaredElement;
MethodElement callElement = cElement.methods[0];
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.staticType, typeProvider.doubleType);
expect(invocation.staticInvokeType.toString(), '(int) → double');
SimpleIdentifier methodName = invocation.methodName;
expect(methodName.staticElement, same(callElement));
expect(methodName.staticType.toString(), '(int) → double');
}
test_methodInvocation_explicitCall_functionTarget() async {
addTestFile(r'''
main(double computation(int p)) {
computation.call(1);
}
''');
await resolveTestFile();
expect(result.errors, isEmpty);
FunctionDeclaration main = result.unit.declarations[0];
FunctionElement mainElement = main.declaredElement;
ParameterElement parameter = mainElement.parameters[0];
BlockFunctionBody mainBody = main.functionExpression.body;
List<Statement> statements = mainBody.block.statements;
ExpressionStatement statement = statements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.staticType, typeProvider.doubleType);
expect(invocation.staticInvokeType.toString(), '(int) → double');
SimpleIdentifier target = invocation.target;
expect(target.staticElement, same(parameter));
expect(target.staticType.toString(), '(int) → double');
SimpleIdentifier methodName = invocation.methodName;
expect(methodName.staticElement, isNull);
expect(methodName.staticType, dynamicType);
}
test_methodInvocation_instanceMethod_forwardingStub() async {
addTestFile(r'''
class A {
void foo(int x) {}
}
abstract class I<T> {
void foo(T x);
}
class B extends A implements I<int> {}
main(B b) {
b.foo(1);
}
''');
await resolveTestFile();
ClassDeclaration aNode = result.unit.declarations[0];
MethodDeclaration fooNode = aNode.members[0];
MethodElement fooElement = fooNode.declaredElement;
List<Statement> mainStatements = _getMainStatements(result);
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.methodName.staticElement, same(fooElement));
var invokeTypeStr = '(int) → void';
expect(invocation.staticType.toString(), 'void');
expect(invocation.staticInvokeType.toString(), invokeTypeStr);
}
test_methodInvocation_instanceMethod_genericClass() async {
addTestFile(r'''
main() {
new C<int, double>().m(1);
}
class C<T, U> {
void m(T p) {}
}
''');
await resolveTestFile();
MethodElement mElement = findElement.method('m');
{
var invocation = findNode.methodInvocation('m(1)');
List<Expression> arguments = invocation.argumentList.arguments;
var invokeTypeStr = '(int) → void';
assertType(invocation, 'void');
assertInvokeType(invocation, invokeTypeStr);
assertMember(invocation.methodName, 'C<int, double>', mElement);
assertType(invocation.methodName, invokeTypeStr);
_assertArgumentToParameter(arguments[0], mElement.parameters[0]);
}
}
test_methodInvocation_instanceMethod_genericClass_genericMethod() async {
addTestFile(r'''
main() {
new C<int>().m(1, 2.3);
}
class C<T> {
Map<T, U> m<U>(T a, U b) => null;
}
''');
await resolveTestFile();
MethodElement mElement = findElement.method('m');
{
var invocation = findNode.methodInvocation('m(1, 2.3)');
List<Expression> arguments = invocation.argumentList.arguments;
var invokeTypeStr = '(int, double) → Map<int, double>';
assertType(invocation, 'Map<int, double>');
assertInvokeType(invocation, invokeTypeStr);
assertMember(invocation.methodName, 'C<int>', mElement);
assertType(invocation.methodName, '<U>(int, U) → Map<int, U>');
if (AnalysisDriver.useSummary2) {
_assertArgumentToParameter2(arguments[0], 'int');
_assertArgumentToParameter2(arguments[1], 'double');
} else {
_assertArgumentToParameter(arguments[0], mElement.parameters[0]);
_assertArgumentToParameter(arguments[1], mElement.parameters[1]);
}
}
}
test_methodInvocation_namedArgument() async {
addTestFile(r'''
void main() {
foo(1, b: true, c: 3.0);
}
void foo(int a, {bool b, double c}) {}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
FunctionDeclaration foo = result.unit.declarations[1];
ExecutableElement fooElement = foo.declaredElement;
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
List<Expression> arguments = invocation.argumentList.arguments;
_assertArgumentToParameter(arguments[0], fooElement.parameters[0]);
_assertArgumentToParameter(arguments[1], fooElement.parameters[1]);
_assertArgumentToParameter(arguments[2], fooElement.parameters[2]);
}
test_methodInvocation_notFunction_field_dynamic() async {
addTestFile(r'''
class C {
dynamic f;
foo() {
f(1);
}
}
''');
await resolveTestFile();
ClassDeclaration cDeclaration = result.unit.declarations[0];
FieldDeclaration fDeclaration = cDeclaration.members[0];
VariableDeclaration fNode = fDeclaration.fields.variables[0];
FieldElement fElement = fNode.declaredElement;
MethodDeclaration fooDeclaration = cDeclaration.members[1];
BlockFunctionBody fooBody = fooDeclaration.body;
List<Statement> fooStatements = fooBody.block.statements;
ExpressionStatement statement = fooStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.methodName.staticElement, same(fElement.getter));
expect(invocation.staticInvokeType, DynamicTypeImpl.instance);
expect(invocation.staticType, DynamicTypeImpl.instance);
List<Expression> arguments = invocation.argumentList.arguments;
expect(arguments[0].staticParameterElement, isNull);
}
test_methodInvocation_notFunction_getter_dynamic() async {
addTestFile(r'''
class C {
get f => null;
foo() {
f(1);
}
}
''');
await resolveTestFile();
ClassDeclaration cDeclaration = result.unit.declarations[0];
MethodDeclaration fDeclaration = cDeclaration.members[0];
PropertyAccessorElement fElement = fDeclaration.declaredElement;
MethodDeclaration fooDeclaration = cDeclaration.members[1];
BlockFunctionBody fooBody = fooDeclaration.body;
List<Statement> fooStatements = fooBody.block.statements;
ExpressionStatement statement = fooStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.methodName.staticElement, same(fElement));
expect(invocation.staticInvokeType, DynamicTypeImpl.instance);
expect(invocation.staticType, DynamicTypeImpl.instance);
List<Expression> arguments = invocation.argumentList.arguments;
Expression argument = arguments[0];
expect(argument.staticParameterElement, isNull);
}
test_methodInvocation_notFunction_getter_typedef() async {
addTestFile(r'''
typedef String Fun(int a, {int b});
class C {
Fun get f => null;
foo() {
f(1, b: 2);
}
}
''');
await resolveTestFile();
FunctionTypeAlias funDeclaration = result.unit.declarations[0];
FunctionTypeAliasElement funElement = funDeclaration.declaredElement;
ClassDeclaration cDeclaration = result.unit.declarations[1];
MethodDeclaration fDeclaration = cDeclaration.members[0];
PropertyAccessorElement fElement = fDeclaration.declaredElement;
MethodDeclaration fooDeclaration = cDeclaration.members[1];
BlockFunctionBody fooBody = fooDeclaration.body;
List<Statement> fooStatements = fooBody.block.statements;
ExpressionStatement statement = fooStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.methodName.staticElement, same(fElement));
expect(invocation.staticInvokeType.toString(), '(int, {b: int}) → String');
expect(invocation.staticType, typeProvider.stringType);
List<Expression> arguments = invocation.argumentList.arguments;
if (AnalysisDriver.useSummary2) {
_assertArgumentToParameter2(arguments[0], 'int');
_assertArgumentToParameter2(arguments[1], 'int');
} else {
_assertArgumentToParameter(arguments[0], funElement.parameters[0]);
_assertArgumentToParameter(arguments[1], funElement.parameters[1]);
}
}
test_methodInvocation_notFunction_local_dynamic() async {
addTestFile(r'''
main(f) {
f(1);
}
''');
await resolveTestFile();
FunctionDeclaration mainDeclaration = result.unit.declarations[0];
FunctionExpression mainFunction = mainDeclaration.functionExpression;
ParameterElement fElement =
mainFunction.parameters.parameters[0].declaredElement;
BlockFunctionBody mainBody = mainFunction.body;
List<Statement> mainStatements = mainBody.block.statements;
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.methodName.staticElement, same(fElement));
expect(invocation.staticInvokeType, isDynamicType);
expect(invocation.staticType, DynamicTypeImpl.instance);
List<Expression> arguments = invocation.argumentList.arguments;
Expression argument = arguments[0];
expect(argument.staticParameterElement, isNull);
}
test_methodInvocation_notFunction_local_functionTyped() async {
addTestFile(r'''
main(String f(int a)) {
f(1);
}
''');
await resolveTestFile();
FunctionDeclaration mainDeclaration = result.unit.declarations[0];
FunctionExpression mainFunction = mainDeclaration.functionExpression;
ParameterElement fElement =
mainFunction.parameters.parameters[0].declaredElement;
BlockFunctionBody mainBody = mainFunction.body;
List<Statement> mainStatements = mainBody.block.statements;
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.methodName.staticElement, same(fElement));
expect(invocation.staticInvokeType.toString(), '(int) → String');
expect(invocation.staticType, typeProvider.stringType);
List<Expression> arguments = invocation.argumentList.arguments;
_assertArgumentToParameter(
arguments[0], (fElement.type as FunctionType).parameters[0]);
}
test_methodInvocation_notFunction_topLevelVariable_dynamic() async {
addTestFile(r'''
dynamic f;
main() {
f(1);
}
''');
await resolveTestFile();
TopLevelVariableDeclaration fDeclaration = result.unit.declarations[0];
VariableDeclaration fNode = fDeclaration.variables.variables[0];
TopLevelVariableElement fElement = fNode.declaredElement;
List<Statement> mainStatements = _getMainStatements(result);
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.methodName.staticElement, same(fElement.getter));
expect(invocation.staticInvokeType, isDynamicType);
expect(invocation.staticType, DynamicTypeImpl.instance);
List<Expression> arguments = invocation.argumentList.arguments;
Expression argument = arguments[0];
expect(argument.staticParameterElement, isNull);
}
test_methodInvocation_staticMethod() async {
addTestFile(r'''
main() {
C.m(1);
}
class C {
static void m(int p) {}
void foo() {
m(2);
}
}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
ClassDeclaration cNode = result.unit.declarations[1];
ClassElement cElement = cNode.declaredElement;
MethodDeclaration mNode = cNode.members[0];
MethodElement mElement = mNode.declaredElement;
{
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
List<Expression> arguments = invocation.argumentList.arguments;
SimpleIdentifier target = invocation.target;
expect(target.staticElement, same(cElement));
expect(target.staticType, same(cElement.type));
var invokeTypeStr = '(int) → void';
expect(invocation.staticType.toString(), 'void');
expect(invocation.staticInvokeType.toString(), invokeTypeStr);
expect(invocation.staticInvokeType.element, same(mElement));
expect(invocation.methodName.staticElement, same(mElement));
expect(invocation.methodName.staticType.toString(), invokeTypeStr);
Expression argument = arguments[0];
_assertArgumentToParameter(argument, mElement.parameters[0]);
}
{
MethodDeclaration fooNode = cNode.members[1];
BlockFunctionBody fooBody = fooNode.body;
List<Statement> statements = fooBody.block.statements;
ExpressionStatement statement = statements[0];
MethodInvocation invocation = statement.expression;
List<Expression> arguments = invocation.argumentList.arguments;
expect(invocation.target, isNull);
var invokeTypeStr = '(int) → void';
expect(invocation.staticType.toString(), 'void');
expect(invocation.staticInvokeType.toString(), invokeTypeStr);
expect(invocation.staticInvokeType.element, same(mElement));
expect(invocation.methodName.staticElement, same(mElement));
expect(invocation.methodName.staticType.toString(), invokeTypeStr);
Expression argument = arguments[0];
_assertArgumentToParameter(argument, mElement.parameters[0]);
}
}
test_methodInvocation_staticMethod_contextTypeParameter() async {
addTestFile(r'''
class C<T> {
static E foo<E>(C<E> c) => null;
void bar() {
foo(this);
}
}
''');
await resolveTestFile();
ClassDeclaration cNode = result.unit.declarations[0];
TypeParameterElement tElement = cNode.declaredElement.typeParameters[0];
MethodDeclaration barNode = cNode.members[1];
BlockFunctionBody barBody = barNode.body;
ExpressionStatement fooStatement = barBody.block.statements[0];
MethodInvocation fooInvocation = fooStatement.expression;
expect(fooInvocation.staticInvokeType.toString(), '(C<T>) → T');
expect(fooInvocation.staticType.toString(), 'T');
expect(fooInvocation.staticType.element, same(tElement));
}
test_methodInvocation_topLevelFunction() async {
addTestFile(r'''
void main() {
f(1, '2');
}
double f(int a, String b) {}
''');
String fTypeString = '(int, String) → double';
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
InterfaceType doubleType = typeProvider.doubleType;
FunctionDeclaration fNode = result.unit.declarations[1];
FunctionElement fElement = fNode.declaredElement;
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
List<Expression> arguments = invocation.argumentList.arguments;
expect(invocation.methodName.staticElement, same(fElement));
expect(invocation.methodName.staticType.toString(), fTypeString);
expect(invocation.staticType, doubleType);
expect(invocation.staticInvokeType.toString(), fTypeString);
_assertArgumentToParameter(arguments[0], fElement.parameters[0]);
_assertArgumentToParameter(arguments[1], fElement.parameters[1]);
}
@failingTest
test_methodInvocation_topLevelFunction_generic() async {
addTestFile(r'''
void main() {
f<bool, String>(true, 'str');
f(1, 2.3);
}
void f<T, U>(T a, U b) {}
''');
await resolveTestFile();
List<Statement> mainStatements = _getMainStatements(result);
FunctionDeclaration fNode = result.unit.declarations[1];
FunctionElement fElement = fNode.declaredElement;
// f<bool, String>(true, 'str');
{
String fTypeString = '(bool, String) → void';
ExpressionStatement statement = mainStatements[0];
MethodInvocation invocation = statement.expression;
List<TypeAnnotation> typeArguments = invocation.typeArguments.arguments;
expect(typeArguments, hasLength(2));
{
TypeName typeArgument = typeArguments[0];
InterfaceType boolType = typeProvider.boolType;
expect(typeArgument.type, boolType);
expect(typeArgument.name.staticElement, boolType.element);
expect(typeArgument.name.staticType, boolType);
}
{
TypeName typeArgument = typeArguments[1];
InterfaceType stringType = typeProvider.stringType;
expect(typeArgument.type, stringType);
expect(typeArgument.name.staticElement, stringType.element);
expect(typeArgument.name.staticType, stringType);
}
List<Expression> arguments = invocation.argumentList.arguments;
expect(invocation.methodName.staticElement, same(fElement));
expect(invocation.methodName.staticType.toString(), fTypeString);
expect(invocation.staticType, VoidTypeImpl.instance);
expect(invocation.staticInvokeType.toString(), fTypeString);
_assertArgumentToParameter(arguments[0], fElement.parameters[0],
memberType: typeProvider.boolType);
_assertArgumentToParameter(arguments[1], fElement.parameters[1],
memberType: typeProvider.stringType);
}
// f(1, 2.3);
{
String fTypeString = '(int, double) → void';
ExpressionStatement statement = mainStatements[1];
MethodInvocation invocation = statement.expression;
List<Expression> arguments = invocation.argumentList.arguments;
expect(invocation.methodName.staticElement, same(fElement));
expect(invocation.methodName.staticType.toString(), fTypeString);
expect(invocation.staticType, VoidTypeImpl.instance);
expect(invocation.staticInvokeType.toString(), fTypeString);
_assertArgumentToParameter(arguments[0], fElement.parameters[0],
memberType: typeProvider.intType);
_assertArgumentToParameter(arguments[1], fElement.parameters[1],
memberType: typeProvider.doubleType);
}
}
test_optionalConst() async {
addTestFile(r'''
class C {
const C();
const C.named();
}
const a = C(); // ref
const b = C.named(); // ref
''');
await resolveTestFile();
expect(result.errors, isEmpty);
var c = findElement.class_('C');
{
var creation = findNode.instanceCreation('C(); // ref');
assertElement(creation, c.unnamedConstructor);
assertType(creation, 'C');
assertTypeName(creation.constructorName.type, c, 'C');
}
{
var creation = findNode.instanceCreation('C.named(); // ref');
var namedConstructor = c.getNamedConstructor('named');
assertElement(creation, namedConstructor);
assertType(creation, 'C');
assertTypeName(creation.constructorName.type, c, 'C');
assertElement(creation.constructorName.name, namedConstructor);
}
}
test_optionalConst_prefixed() async {
newFile('/test/lib/a.dart', content: r'''
class C {
const C();
const C.named();
}
''');
addTestFile(r'''
import 'a.dart' as p;
const a = p.C(); // ref
const b = p.C.named(); // ref
''');
await resolveTestFile();
expect(result.errors, isEmpty);
var import = findElement.import('package:test/a.dart');
var c = import.importedLibrary.getType('C');
{
var creation = findNode.instanceCreation('C(); // ref');
assertElement(creation, c.unnamedConstructor);
assertType(creation, 'C');
assertTypeName(creation.constructorName.type, c, 'C',
expectedPrefix: import.prefix);
}
{
var creation = findNode.instanceCreation('C.named(); // ref');
var namedConstructor = c.getNamedConstructor('named');
assertElement(creation, namedConstructor);
assertType(creation, 'C');
assertTypeName(creation.constructorName.type, c, 'C',
expectedPrefix: import.prefix);
assertElement(creation.constructorName.name, namedConstructor);
}
}
test_optionalConst_typeArguments() async {
addTestFile(r'''
class C<T> {
const C();
const C.named();
}
const a = C<int>(); // ref
const b = C<String>.named(); // ref
''');
await resolveTestFile();
expect(result.errors, isEmpty);
var c = findElement.class_('C');
{
var creation = findNode.instanceCreation('C<int>(); // ref');
assertMember(creation, 'C<int>', c.unnamedConstructor);
assertType(creation, 'C<int>');
assertTypeName(creation.constructorName.type, c, 'C<int>');
assertTypeName(findNode.typeName('int>'), intElement, 'int');
}
{
var creation = findNode.instanceCreation('C<String>.named(); // ref');
var namedConstructor = c.getNamedConstructor('named');
assertMember(creation, 'C<String>', namedConstructor);
assertType(creation, 'C<String>');
assertTypeName(creation.constructorName.type, c, 'C<String>');
assertTypeName(findNode.typeName('String>'), stringElement, 'String');
assertMember(
creation.constructorName.name, 'C<String>', namedConstructor);
}
}
test_outline_invalid_mixin_arguments_tooFew() async {
addTestFile(r'''
class A extends Object with Map<int> {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var mapRef = findNode.typeName('Map<');
assertTypeName(mapRef, mapElement, 'Map<dynamic, dynamic>');
var intRef = findNode.typeName('int>');
assertTypeName(intRef, intElement, 'int');
}
test_outline_invalid_mixin_arguments_tooMany() async {
addTestFile(r'''
class A extends Object with List<int, double> {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var listRef = findNode.typeName('List<');
assertTypeName(listRef, listElement, 'List<dynamic>');
var intRef = findNode.typeName('int,');
assertTypeName(intRef, intElement, 'int');
var doubleRef = findNode.typeName('double>');
assertTypeName(doubleRef, doubleElement, 'double');
}
test_outline_invalid_mixin_typeParameter() async {
addTestFile(r'''
class A<T> extends Object with T<int, double> {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var tRef = findNode.typeName('T<');
assertTypeName(tRef, findElement.typeParameter('T'), 'T');
var intRef = findNode.typeName('int,');
assertTypeName(intRef, intElement, 'int');
var doubleRef = findNode.typeName('double>');
assertTypeName(doubleRef, doubleElement, 'double');
}
test_outline_invalid_supertype_arguments_tooFew() async {
addTestFile(r'''
class A extends Map<int> {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var mapRef = findNode.typeName('Map<');
assertTypeName(mapRef, mapElement, 'Map<dynamic, dynamic>');
var intRef = findNode.typeName('int>');
assertTypeName(intRef, intElement, 'int');
}
test_outline_invalid_supertype_arguments_tooMany() async {
addTestFile(r'''
class A extends List<int, double> {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var listRef = findNode.typeName('List<');
assertTypeName(listRef, listElement, 'List<dynamic>');
var intRef = findNode.typeName('int,');
assertTypeName(intRef, intElement, 'int');
var doubleRef = findNode.typeName('double>');
assertTypeName(doubleRef, doubleElement, 'double');
}
test_outline_invalid_supertype_hasArguments() async {
addTestFile(r'''
class A extends X<int, double> {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.typeName('X<');
assertTypeName(xRef, null, 'dynamic');
var intRef = findNode.typeName('int,');
assertTypeName(intRef, intElement, 'int');
var doubleRef = findNode.typeName('double>');
assertTypeName(doubleRef, doubleElement, 'double');
}
test_outline_invalid_supertype_noArguments() async {
addTestFile(r'''
class A extends X {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var xRef = findNode.typeName('X {}');
assertTypeName(xRef, null, 'dynamic');
}
test_outline_invalid_supertype_typeParameter() async {
addTestFile(r'''
class A<T> extends T<int, double> {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var tRef = findNode.typeName('T<');
assertTypeName(tRef, findElement.typeParameter('T'), 'T');
var intRef = findNode.typeName('int,');
assertTypeName(intRef, intElement, 'int');
var doubleRef = findNode.typeName('double>');
assertTypeName(doubleRef, doubleElement, 'double');
}
test_outline_invalid_type_arguments_tooFew() async {
addTestFile(r'''
typedef Map<int> F();
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var mapRef = findNode.typeName('Map<');
assertTypeName(mapRef, mapElement, 'Map<dynamic, dynamic>');
var intRef = findNode.typeName('int>');
assertTypeName(intRef, intElement, 'int');
}
test_outline_invalid_type_arguments_tooMany() async {
addTestFile(r'''
typedef List<int, double> F();
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var listRef = findNode.typeName('List<');
assertTypeName(listRef, listElement, 'List<dynamic>');
var intRef = findNode.typeName('int,');
assertTypeName(intRef, intElement, 'int');
var doubleRef = findNode.typeName('double>');
assertTypeName(doubleRef, doubleElement, 'double');
}
test_outline_invalid_type_typeParameter() async {
addTestFile(r'''
typedef T<int> F<T>();
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var tRef = findNode.typeName('T<');
assertTypeName(tRef, findElement.typeParameter('T'), 'T');
var intRef = findNode.typeName('int>');
assertTypeName(intRef, intElement, 'int');
}
test_outline_type_genericFunction() async {
addTestFile(r'''
int Function(double) g() => null;
''');
await resolveTestFile();
expect(result.errors, isEmpty);
var intRef = findNode.typeName('int Function');
assertTypeName(intRef, intElement, 'int');
var functionRef = findNode.genericFunctionType('Function(double)');
assertType(functionRef, '(double) → int');
var doubleRef = findNode.typeName('double) g');
assertTypeName(doubleRef, doubleElement, 'double');
}
test_outline_type_topLevelVar_named() async {
addTestFile(r'''
int a;
List<double> b;
''');
await resolveTestFile();
expect(result.errors, isEmpty);
var intRef = findNode.typeName('int a');
assertTypeName(intRef, intElement, 'int');
var listRef = findNode.typeName('List<double> b');
assertTypeName(listRef, listElement, 'List<double>');
var doubleRef = findNode.typeName('double> b');
assertTypeName(doubleRef, doubleElement, 'double');
}
test_outline_type_topLevelVar_named_prefixed() async {
addTestFile(r'''
import 'dart:async' as my;
my.Future<int> a;
''');
await resolveTestFile();
ImportElement myImport = result.libraryElement.imports[0];
var intRef = findNode.typeName('int> a');
assertTypeName(intRef, intElement, 'int');
var futureRef = findNode.typeName('my.Future<int> a');
assertTypeName(futureRef, futureElement, 'Future<int>',
expectedPrefix: myImport.prefix);
}
test_postfix_increment_of_non_generator() async {
addTestFile('''
void f(int g()) {
g()++;
}
''');
await resolveTestFile();
var gRef = findNode.simple('g()++');
assertType(gRef, '() → int');
assertElement(gRef, findElement.parameter('g'));
}
test_postfix_increment_of_postfix_increment() async {
addTestFile('''
void f(int x) {
x ++ ++;
}
''');
await resolveTestFile();
var xRef = findNode.simple('x ++');
assertType(xRef, 'int');
assertElement(xRef, findElement.parameter('x'));
}
test_postfixExpression_local() async {
String content = r'''
main() {
int v = 0;
v++;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> mainStatements = _getMainStatements(result);
VariableElement v;
{
VariableDeclarationStatement statement = mainStatements[0];
v = statement.variables.variables[0].declaredElement;
expect(v.type, typeProvider.intType);
}
{
ExpressionStatement statement = mainStatements[1];
PostfixExpression postfix = statement.expression;
expect(postfix.operator.type, TokenType.PLUS_PLUS);
expect(postfix.staticElement.name, '+');
expect(postfix.staticType, typeProvider.intType);
SimpleIdentifier operand = postfix.operand;
expect(operand.staticElement, same(v));
expect(operand.staticType, typeProvider.intType);
}
}
test_postfixExpression_propertyAccess() async {
String content = r'''
main() {
new C().f++;
}
class C {
int f;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
ClassDeclaration cClassDeclaration = unit.declarations[1];
ClassElement cClassElement = cClassDeclaration.declaredElement;
FieldElement fElement = cClassElement.getField('f');
List<Statement> mainStatements = _getMainStatements(result);
{
ExpressionStatement statement = mainStatements[0];
PostfixExpression postfix = statement.expression;
expect(postfix.operator.type, TokenType.PLUS_PLUS);
expect(postfix.staticElement.name, '+');
expect(postfix.staticType, typeProvider.intType);
PropertyAccess propertyAccess = postfix.operand;
expect(propertyAccess.staticType, typeProvider.intType);
SimpleIdentifier propertyName = propertyAccess.propertyName;
expect(propertyName.staticElement, same(fElement.setter));
expect(propertyName.staticType, typeProvider.intType);
}
}
test_prefix_increment_of_non_generator() async {
addTestFile('''
void f(bool x) {
++!x;
}
''');
await resolveTestFile();
var xRef = findNode.simple('x;');
assertType(xRef, 'bool');
assertElement(xRef, findElement.parameter('x'));
}
test_prefix_increment_of_postfix_increment() async {
addTestFile('''
void f(int x) {
++x++;
}
''');
await resolveTestFile();
var xRef = findNode.simple('x++');
assertType(xRef, 'int');
assertElement(xRef, findElement.parameter('x'));
}
test_prefix_increment_of_prefix_increment() async {
addTestFile('''
void f(int x) {
++ ++ x;
}
''');
await resolveTestFile();
var xRef = findNode.simple('x;');
assertType(xRef, 'int');
assertElement(xRef, findElement.parameter('x'));
}
test_prefixedIdentifier_classInstance_instanceField() async {
String content = r'''
main() {
var c = new C();
c.f;
}
class C {
int f;
}
''';
addTestFile(content);
await resolveTestFile();
List<Statement> statements = _getMainStatements(result);
ClassDeclaration cDeclaration = result.unit.declarations[1];
ClassElement cElement = cDeclaration.declaredElement;
FieldElement fElement = cElement.fields[0];
VariableDeclarationStatement cStatement = statements[0];
VariableElement vElement =
cStatement.variables.variables[0].declaredElement;
ExpressionStatement statement = statements[1];
PrefixedIdentifier prefixed = statement.expression;
SimpleIdentifier prefix = prefixed.prefix;
expect(prefix.staticElement, same(vElement));
expect(prefix.staticType, cElement.type);
SimpleIdentifier identifier = prefixed.identifier;
expect(identifier.staticElement, same(fElement.getter));
expect(identifier.staticType, typeProvider.intType);
}
test_prefixedIdentifier_className_staticField() async {
String content = r'''
main() {
C.f;
}
class C {
static f = 0;
}
''';
addTestFile(content);
await resolveTestFile();
List<Statement> statements = _getMainStatements(result);
ClassDeclaration cDeclaration = result.unit.declarations[1];
ClassElement cElement = cDeclaration.declaredElement;
FieldElement fElement = cElement.fields[0];
ExpressionStatement statement = statements[0];
PrefixedIdentifier prefixed = statement.expression;
SimpleIdentifier prefix = prefixed.prefix;
expect(prefix.staticElement, same(cElement));
expect(prefix.staticType, cElement.type);
SimpleIdentifier identifier = prefixed.identifier;
expect(identifier.staticElement, same(fElement.getter));
expect(identifier.staticType, typeProvider.intType);
}
test_prefixedIdentifier_explicitCall() async {
addTestFile(r'''
main(double computation(int p)) {
computation.call;
}
''');
await resolveTestFile();
expect(result.errors, isEmpty);
FunctionDeclaration main = result.unit.declarations[0];
FunctionElement mainElement = main.declaredElement;
ParameterElement parameter = mainElement.parameters[0];
BlockFunctionBody mainBody = main.functionExpression.body;
List<Statement> statements = mainBody.block.statements;
ExpressionStatement statement = statements[0];
PrefixedIdentifier prefixed = statement.expression;
expect(prefixed.prefix.staticElement, same(parameter));
expect(prefixed.prefix.staticType.toString(), '(int) → double');
SimpleIdentifier methodName = prefixed.identifier;
expect(methodName.staticElement, isNull);
expect(methodName.staticType, typeProvider.dynamicType);
}
test_prefixedIdentifier_importPrefix_className() async {
newFile('/test/lib/lib.dart', content: '''
class MyClass {}
typedef void MyFunctionTypeAlias();
int myTopVariable;
int myTopFunction() => 0;
int get myGetter => 0;
void set mySetter(int _) {}
''');
addTestFile(r'''
import 'lib.dart' as my;
main() {
my.MyClass;
my.MyFunctionTypeAlias;
my.myTopVariable;
my.myTopFunction;
my.myTopFunction();
my.myGetter;
my.mySetter = 0;
}
''');
await resolveTestFile();
// TODO(scheglov) Uncomment and fix "unused imports" hint.
// expect(result.errors, isEmpty);
var unitElement = result.unit.declaredElement;
ImportElement myImport = unitElement.library.imports[0];
PrefixElement myPrefix = myImport.prefix;
var typeProvider = unitElement.context.typeProvider;
var myLibrary = myImport.importedLibrary;
var myUnit = myLibrary.definingCompilationUnit;
var myClass = myUnit.types.single;
var myFunctionTypeAlias = myUnit.functionTypeAliases.single;
var myTopVariable = myUnit.topLevelVariables[0];
var myTopFunction = myUnit.functions.single;
var myGetter = myUnit.topLevelVariables[1].getter;
var mySetter = myUnit.topLevelVariables[2].setter;
expect(myTopVariable.name, 'myTopVariable');
expect(myGetter.displayName, 'myGetter');
expect(mySetter.displayName, 'mySetter');
List<Statement> statements = _getMainStatements(result);
void assertPrefix(SimpleIdentifier identifier) {
expect(identifier.staticElement, same(myPrefix));
expect(identifier.staticType, isNull);
}
void assertPrefixedIdentifier(
int statementIndex, Element expectedElement, DartType expectedType) {
ExpressionStatement statement = statements[statementIndex];
PrefixedIdentifier prefixed = statement.expression;
assertPrefix(prefixed.prefix);
expect(prefixed.identifier.staticElement, same(expectedElement));
expect(prefixed.identifier.staticType, expectedType);
}
assertPrefixedIdentifier(0, myClass, typeProvider.typeType);
assertPrefixedIdentifier(1, myFunctionTypeAlias, typeProvider.typeType);
assertPrefixedIdentifier(2, myTopVariable.getter, typeProvider.intType);
{
ExpressionStatement statement = statements[3];
PrefixedIdentifier prefixed = statement.expression;
assertPrefix(prefixed.prefix);
expect(prefixed.identifier.staticElement, same(myTopFunction));
expect(prefixed.identifier.staticType, isNotNull);
}
{
ExpressionStatement statement = statements[4];
MethodInvocation invocation = statement.expression;
assertPrefix(invocation.target);
expect(invocation.methodName.staticElement, same(myTopFunction));
expect(invocation.methodName.staticType, isNotNull);
}
assertPrefixedIdentifier(5, myGetter, typeProvider.intType);
{
ExpressionStatement statement = statements[6];
AssignmentExpression assignment = statement.expression;
PrefixedIdentifier left = assignment.leftHandSide;
assertPrefix(left.prefix);
expect(left.identifier.staticElement, same(mySetter));
expect(left.identifier.staticType, typeProvider.intType);
}
}
test_prefixExpression_local() async {
String content = r'''
main() {
int v = 0;
++v;
~v;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> mainStatements = _getMainStatements(result);
VariableElement v;
{
VariableDeclarationStatement statement = mainStatements[0];
v = statement.variables.variables[0].declaredElement;
expect(v.type, typeProvider.intType);
}
{
ExpressionStatement statement = mainStatements[1];
PrefixExpression prefix = statement.expression;
expect(prefix.operator.type, TokenType.PLUS_PLUS);
expect(prefix.staticElement.name, '+');
expect(prefix.staticType, typeProvider.intType);
SimpleIdentifier operand = prefix.operand;
expect(operand.staticElement, same(v));
expect(operand.staticType, typeProvider.intType);
}
{
ExpressionStatement statement = mainStatements[2];
PrefixExpression prefix = statement.expression;
expect(prefix.operator.type, TokenType.TILDE);
expect(prefix.staticElement.name, '~');
expect(prefix.staticType, typeProvider.intType);
SimpleIdentifier operand = prefix.operand;
expect(operand.staticElement, same(v));
expect(operand.staticType, typeProvider.intType);
}
}
test_prefixExpression_local_not() async {
String content = r'''
main() {
bool v = true;
!v;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> mainStatements = _getMainStatements(result);
VariableElement v;
{
VariableDeclarationStatement statement = mainStatements[0];
v = statement.variables.variables[0].declaredElement;
expect(v.type, typeProvider.boolType);
}
{
ExpressionStatement statement = mainStatements[1];
PrefixExpression prefix = statement.expression;
expect(prefix.operator.type, TokenType.BANG);
expect(prefix.staticElement, isNull);
expect(prefix.staticType, typeProvider.boolType);
SimpleIdentifier operand = prefix.operand;
expect(operand.staticElement, same(v));
expect(operand.staticType, typeProvider.boolType);
}
}
test_prefixExpression_propertyAccess() async {
String content = r'''
main() {
++new C().f;
~new C().f;
}
class C {
int f;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
ClassDeclaration cClassDeclaration = unit.declarations[1];
ClassElement cClassElement = cClassDeclaration.declaredElement;
FieldElement fElement = cClassElement.getField('f');
List<Statement> mainStatements = _getMainStatements(result);
{
ExpressionStatement statement = mainStatements[0];
PrefixExpression prefix = statement.expression;
expect(prefix.operator.type, TokenType.PLUS_PLUS);
expect(prefix.staticElement.name, '+');
expect(prefix.staticType, typeProvider.intType);
PropertyAccess propertyAccess = prefix.operand;
expect(propertyAccess.staticType, typeProvider.intType);
SimpleIdentifier propertyName = propertyAccess.propertyName;
expect(propertyName.staticElement, same(fElement.setter));
expect(propertyName.staticType, typeProvider.intType);
}
{
ExpressionStatement statement = mainStatements[1];
PrefixExpression prefix = statement.expression;
expect(prefix.operator.type, TokenType.TILDE);
expect(prefix.staticElement.name, '~');
expect(prefix.staticType, typeProvider.intType);
PropertyAccess propertyAccess = prefix.operand;
expect(propertyAccess.staticType, typeProvider.intType);
SimpleIdentifier propertyName = propertyAccess.propertyName;
expect(propertyName.staticElement, same(fElement.getter));
expect(propertyName.staticType, typeProvider.intType);
}
}
test_propertyAccess_field() async {
String content = r'''
main() {
new C().f;
}
class C {
int f;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
ClassDeclaration cClassDeclaration = unit.declarations[1];
ClassElement cClassElement = cClassDeclaration.declaredElement;
FieldElement fElement = cClassElement.getField('f');
List<Statement> mainStatements = _getMainStatements(result);
{
ExpressionStatement statement = mainStatements[0];
PropertyAccess access = statement.expression;
expect(access.staticType, typeProvider.intType);
InstanceCreationExpression newC = access.target;
expect(newC.staticElement, cClassElement.unnamedConstructor);
expect(newC.staticType, cClassElement.type);
expect(access.propertyName.staticElement, same(fElement.getter));
expect(access.propertyName.staticType, typeProvider.intType);
}
}
test_propertyAccess_getter() async {
String content = r'''
main() {
new C().f;
}
class C {
int get f => 0;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
ClassDeclaration cClassDeclaration = unit.declarations[1];
ClassElement cClassElement = cClassDeclaration.declaredElement;
FieldElement fElement = cClassElement.getField('f');
List<Statement> mainStatements = _getMainStatements(result);
{
ExpressionStatement statement = mainStatements[0];
PropertyAccess access = statement.expression;
expect(access.staticType, typeProvider.intType);
InstanceCreationExpression newC = access.target;
expect(newC.staticElement, cClassElement.unnamedConstructor);
expect(newC.staticType, cClassElement.type);
expect(access.propertyName.staticElement, same(fElement.getter));
expect(access.propertyName.staticType, typeProvider.intType);
}
}
test_reference_to_class_type_parameter() async {
addTestFile('''
class C<T> {
void f() {
T x;
}
}
''');
await resolveTestFile();
var tElement = findElement.class_('C').typeParameters[0];
var tReference = findNode.simple('T x');
var tReferenceType = tReference.staticType as TypeParameterType;
expect(tReferenceType.element, same(tElement));
assertElement(tReference, tElement);
}
test_setLiteral() async {
addTestFile(r'''
main() {
var v = <int>{};
print(v);
}
''');
await resolveTestFile();
expect(result.errors, isEmpty);
var literal = findNode.setOrMapLiteral('<int>{}');
assertType(literal, 'Set<int>');
var intRef = findNode.simple('int>{}');
assertElement(intRef, intElement);
assertType(intRef, 'int');
}
test_stringInterpolation() async {
String content = r'''
void main() {
var v = 42;
'$v$v $v';
' ${v + 1} ';
}
''';
addTestFile(content);
await resolveTestFile();
expect(result.path, testFile);
expect(result.errors, isEmpty);
FunctionDeclaration main = result.unit.declarations[0];
expect(main.declaredElement, isNotNull);
expect(main.name.staticElement, isNotNull);
expect(main.name.staticType.toString(), '() → void');
BlockFunctionBody body = main.functionExpression.body;
NodeList<Statement> statements = body.block.statements;
// var v = 42;
VariableElement vElement;
{
VariableDeclarationStatement statement = statements[0];
vElement = statement.variables.variables[0].name.staticElement;
}
{
ExpressionStatement statement = statements[1];
StringInterpolation interpolation = statement.expression;
InterpolationExpression element_1 = interpolation.elements[1];
SimpleIdentifier expression_1 = element_1.expression;
expect(expression_1.staticElement, same(vElement));
expect(expression_1.staticType, typeProvider.intType);
InterpolationExpression element_3 = interpolation.elements[3];
SimpleIdentifier expression_3 = element_3.expression;
expect(expression_3.staticElement, same(vElement));
expect(expression_3.staticType, typeProvider.intType);
InterpolationExpression element_5 = interpolation.elements[5];
SimpleIdentifier expression_5 = element_5.expression;
expect(expression_5.staticElement, same(vElement));
expect(expression_5.staticType, typeProvider.intType);
}
{
ExpressionStatement statement = statements[2];
StringInterpolation interpolation = statement.expression;
InterpolationExpression element_1 = interpolation.elements[1];
BinaryExpression expression = element_1.expression;
expect(expression.staticType, typeProvider.intType);
SimpleIdentifier left = expression.leftOperand;
expect(left.staticElement, same(vElement));
expect(left.staticType, typeProvider.intType);
}
}
test_stringInterpolation_multiLine_emptyBeforeAfter() async {
addTestFile(r"""
void main() {
var v = 42;
'''$v''';
}
""");
await resolveTestFile();
expect(result.errors, isEmpty);
}
test_super() async {
String content = r'''
class A {
void method(int p) {}
int get getter => 0;
void set setter(int p) {}
int operator+(int p) => 0;
}
class B extends A {
void test() {
method(1);
super.method(2);
getter;
super.getter;
setter = 3;
super.setter = 4;
this + 5;
}
}
''';
addTestFile(content);
await resolveTestFile();
ClassDeclaration aNode = result.unit.declarations[0];
ClassDeclaration bNode = result.unit.declarations[1];
MethodElement methodElement = aNode.members[0].declaredElement;
PropertyAccessorElement getterElement = aNode.members[1].declaredElement;
PropertyAccessorElement setterElement = aNode.members[2].declaredElement;
MethodElement operatorElement = aNode.members[3].declaredElement;
MethodDeclaration testNode = bNode.members[0];
BlockFunctionBody testBody = testNode.body;
List<Statement> testStatements = testBody.block.statements;
// method(1);
{
ExpressionStatement statement = testStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.target, isNull);
expect(invocation.methodName.staticElement, same(methodElement));
}
// super.method(2);
{
ExpressionStatement statement = testStatements[1];
MethodInvocation invocation = statement.expression;
SuperExpression target = invocation.target;
expect(target.staticType, bNode.declaredElement.type); // raw
expect(invocation.methodName.staticElement, same(methodElement));
}
// getter;
{
ExpressionStatement statement = testStatements[2];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, same(getterElement));
expect(identifier.staticType, typeProvider.intType);
}
// super.getter;
{
ExpressionStatement statement = testStatements[3];
PropertyAccess propertyAccess = statement.expression;
expect(propertyAccess.staticType, typeProvider.intType);
SuperExpression target = propertyAccess.target;
expect(target.staticType, bNode.declaredElement.type); // raw
expect(propertyAccess.propertyName.staticElement, same(getterElement));
expect(propertyAccess.propertyName.staticType, typeProvider.intType);
}
// setter = 3;
{
ExpressionStatement statement = testStatements[4];
AssignmentExpression assignment = statement.expression;
SimpleIdentifier identifier = assignment.leftHandSide;
expect(identifier.staticElement, same(setterElement));
expect(identifier.staticType, typeProvider.intType);
}
// this.setter = 4;
{
ExpressionStatement statement = testStatements[5];
AssignmentExpression assignment = statement.expression;
PropertyAccess propertyAccess = assignment.leftHandSide;
SuperExpression target = propertyAccess.target;
expect(target.staticType, bNode.declaredElement.type); // raw
expect(propertyAccess.propertyName.staticElement, same(setterElement));
expect(propertyAccess.propertyName.staticType, typeProvider.intType);
}
// super + 5;
{
ExpressionStatement statement = testStatements[6];
BinaryExpression binary = statement.expression;
ThisExpression target = binary.leftOperand;
expect(target.staticType, bNode.declaredElement.type); // raw
expect(binary.staticElement, same(operatorElement));
expect(binary.staticType, typeProvider.intType);
}
}
test_this() async {
String content = r'''
class A {
void method(int p) {}
int get getter => 0;
void set setter(int p) {}
int operator+(int p) => 0;
void test() {
method(1);
this.method(2);
getter;
this.getter;
setter = 3;
this.setter = 4;
this + 5;
}
}
''';
addTestFile(content);
await resolveTestFile();
ClassDeclaration aNode = result.unit.declarations[0];
MethodElement methodElement = aNode.members[0].declaredElement;
PropertyAccessorElement getterElement = aNode.members[1].declaredElement;
PropertyAccessorElement setterElement = aNode.members[2].declaredElement;
MethodElement operatorElement = aNode.members[3].declaredElement;
MethodDeclaration testNode = aNode.members[4];
BlockFunctionBody testBody = testNode.body;
List<Statement> testStatements = testBody.block.statements;
// method(1);
{
ExpressionStatement statement = testStatements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.target, isNull);
expect(invocation.methodName.staticElement, same(methodElement));
}
// this.method(2);
{
ExpressionStatement statement = testStatements[1];
MethodInvocation invocation = statement.expression;
ThisExpression target = invocation.target;
expect(target.staticType, aNode.declaredElement.type); // raw
expect(invocation.methodName.staticElement, same(methodElement));
}
// getter;
{
ExpressionStatement statement = testStatements[2];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, same(getterElement));
expect(identifier.staticType, typeProvider.intType);
}
// this.getter;
{
ExpressionStatement statement = testStatements[3];
PropertyAccess propertyAccess = statement.expression;
expect(propertyAccess.staticType, typeProvider.intType);
ThisExpression target = propertyAccess.target;
expect(target.staticType, aNode.declaredElement.type); // raw
expect(propertyAccess.propertyName.staticElement, same(getterElement));
expect(propertyAccess.propertyName.staticType, typeProvider.intType);
}
// setter = 3;
{
ExpressionStatement statement = testStatements[4];
AssignmentExpression assignment = statement.expression;
SimpleIdentifier identifier = assignment.leftHandSide;
expect(identifier.staticElement, same(setterElement));
expect(identifier.staticType, typeProvider.intType);
}
// this.setter = 4;
{
ExpressionStatement statement = testStatements[5];
AssignmentExpression assignment = statement.expression;
PropertyAccess propertyAccess = assignment.leftHandSide;
ThisExpression target = propertyAccess.target;
expect(target.staticType, aNode.declaredElement.type); // raw
expect(propertyAccess.propertyName.staticElement, same(setterElement));
expect(propertyAccess.propertyName.staticType, typeProvider.intType);
}
// this + 5;
{
ExpressionStatement statement = testStatements[6];
BinaryExpression binary = statement.expression;
ThisExpression target = binary.leftOperand;
expect(target.staticType, aNode.declaredElement.type); // raw
expect(binary.staticElement, same(operatorElement));
expect(binary.staticType, typeProvider.intType);
}
}
test_top_class_constructor_parameter_defaultValue() async {
String content = r'''
class C {
double f;
C([int a: 1 + 2]) : f = 3.4;
}
''';
addTestFile(content);
await resolveTestFile();
ClassDeclaration cNode = result.unit.declarations[0];
ClassElement cElement = cNode.declaredElement;
ConstructorDeclaration constructorNode = cNode.members[1];
DefaultFormalParameter aNode = constructorNode.parameters.parameters[0];
_assertDefaultParameter(aNode, cElement.unnamedConstructor.parameters[0],
name: 'a',
offset: 31,
kind: ParameterKind.POSITIONAL,
type: typeProvider.intType);
BinaryExpression binary = aNode.defaultValue;
expect(binary.staticElement, isNotNull);
expect(binary.staticType, typeProvider.intType);
expect(binary.leftOperand.staticType, typeProvider.intType);
expect(binary.rightOperand.staticType, typeProvider.intType);
}
test_top_class_extends() async {
String content = r'''
class A<T> {}
class B extends A<int> {}
''';
addTestFile(content);
await resolveTestFile();
var aRef = findNode.typeName('A<int>');
assertTypeName(aRef, findElement.class_('A'), 'A<int>');
var intRef = findNode.typeName('int>');
assertTypeName(intRef, intElement, 'int');
}
test_top_class_full() async {
String content = r'''
class A<T> {}
class B<T> {}
class C<T> {}
class D extends A<bool> with B<int> implements C<double> {}
''';
addTestFile(content);
await resolveTestFile();
ClassDeclaration aNode = result.unit.declarations[0];
ClassElement aElement = aNode.declaredElement;
ClassDeclaration bNode = result.unit.declarations[1];
ClassElement bElement = bNode.declaredElement;
ClassDeclaration cNode = result.unit.declarations[2];
ClassElement cElement = cNode.declaredElement;
ClassDeclaration dNode = result.unit.declarations[3];
Element dElement = dNode.declaredElement;
SimpleIdentifier dName = dNode.name;
expect(dName.staticElement, same(dElement));
expect(dName.staticType, typeProvider.typeType);
{
var aRawType = aElement.type;
var expectedType = aRawType.instantiate([typeProvider.boolType]);
TypeName superClass = dNode.extendsClause.superclass;
expect(superClass.type, expectedType);
SimpleIdentifier identifier = superClass.name;
expect(identifier.staticElement, aElement);
expect(identifier.staticType, expectedType);
}
{
var bRawType = bElement.type;
var expectedType = bRawType.instantiate([typeProvider.intType]);
TypeName mixinType = dNode.withClause.mixinTypes[0];
expect(mixinType.type, expectedType);
SimpleIdentifier identifier = mixinType.name;
expect(identifier.staticElement, bElement);
expect(identifier.staticType, expectedType);
}
{
var cRawType = cElement.type;
var expectedType = cRawType.instantiate([typeProvider.doubleType]);
TypeName implementedType = dNode.implementsClause.interfaces[0];
expect(implementedType.type, expectedType);
SimpleIdentifier identifier = implementedType.name;
expect(identifier.staticElement, cElement);
expect(identifier.staticType, expectedType);
}
}
test_top_classTypeAlias() async {
String content = r'''
class A<T> {}
class B<T> {}
class C<T> {}
class D = A<bool> with B<int> implements C<double>;
''';
addTestFile(content);
await resolveTestFile();
ClassDeclaration aNode = result.unit.declarations[0];
ClassElement aElement = aNode.declaredElement;
ClassDeclaration bNode = result.unit.declarations[1];
ClassElement bElement = bNode.declaredElement;
ClassDeclaration cNode = result.unit.declarations[2];
ClassElement cElement = cNode.declaredElement;
ClassTypeAlias dNode = result.unit.declarations[3];
Element dElement = dNode.declaredElement;
SimpleIdentifier dName = dNode.name;
expect(dName.staticElement, same(dElement));
expect(dName.staticType, typeProvider.typeType);
{
var aRawType = aElement.type;
var expectedType = aRawType.instantiate([typeProvider.boolType]);
TypeName superClass = dNode.superclass;
expect(superClass.type, expectedType);
SimpleIdentifier identifier = superClass.name;
expect(identifier.staticElement, same(aElement));
expect(identifier.staticType, expectedType);
}
{
var bRawType = bElement.type;
var expectedType = bRawType.instantiate([typeProvider.intType]);
TypeName mixinType = dNode.withClause.mixinTypes[0];
expect(mixinType.type, expectedType);
SimpleIdentifier identifier = mixinType.name;
expect(identifier.staticElement, same(bElement));
expect(identifier.staticType, expectedType);
}
{
var cRawType = cElement.type;
var expectedType = cRawType.instantiate([typeProvider.doubleType]);
TypeName interfaceType = dNode.implementsClause.interfaces[0];
expect(interfaceType.type, expectedType);
SimpleIdentifier identifier = interfaceType.name;
expect(identifier.staticElement, same(cElement));
expect(identifier.staticType, expectedType);
}
}
test_top_enum() async {
String content = r'''
enum MyEnum {
A, B
}
''';
addTestFile(content);
await resolveTestFile();
EnumDeclaration enumNode = result.unit.declarations[0];
ClassElement enumElement = enumNode.declaredElement;
SimpleIdentifier dName = enumNode.name;
expect(dName.staticElement, same(enumElement));
expect(dName.staticType, isNull);
{
var aElement = enumElement.getField('A');
var aNode = enumNode.constants[0];
expect(aNode.declaredElement, same(aElement));
expect(aNode.name.staticElement, same(aElement));
expect(aNode.name.staticType, same(enumElement.type));
}
{
var bElement = enumElement.getField('B');
var bNode = enumNode.constants[1];
expect(bNode.declaredElement, same(bElement));
expect(bNode.name.staticElement, same(bElement));
expect(bNode.name.staticType, same(enumElement.type));
}
}
test_top_executables_class() async {
String content = r'''
class C {
C(int p);
C.named(int p);
int publicMethod(double p) => 0;
int get publicGetter => 0;
void set publicSetter(double p) {}
}
''';
addTestFile(content);
await resolveTestFile();
expect(result.path, testFile);
InterfaceType typeType = typeProvider.typeType;
InterfaceType doubleType = typeProvider.doubleType;
InterfaceType intType = typeProvider.intType;
ClassElement doubleElement = doubleType.element;
ClassElement intElement = intType.element;
ClassDeclaration cNode = result.unit.declarations[0];
ClassElement cElement = cNode.declaredElement;
// The class name identifier.
expect(cNode.name.staticElement, same(cElement));
expect(cNode.name.staticType, typeType);
// unnamed constructor
{
ConstructorDeclaration node = cNode.members[0];
expect(node.declaredElement, isNotNull);
expect(node.declaredElement.type.toString(), '(int) → C');
expect(node.returnType.staticElement, same(cElement));
expect(node.returnType.staticType, typeType);
expect(node.name, isNull);
}
// named constructor
{
ConstructorDeclaration node = cNode.members[1];
expect(node.declaredElement, isNotNull);
expect(node.declaredElement.type.toString(), '(int) → C');
expect(node.returnType.staticElement, same(cElement));
expect(node.returnType.staticType, typeType);
expect(node.name.staticElement, same(node.declaredElement));
expect(node.name.staticType.toString(), '(int) → C');
}
// publicMethod()
{
MethodDeclaration node = cNode.members[2];
expect(node.declaredElement, isNotNull);
expect(node.declaredElement.type.toString(), '(double) → int');
// method return type
TypeName returnType = node.returnType;
SimpleIdentifier returnTypeName = returnType.name;
expect(returnType.type, intType);
expect(returnTypeName.staticElement, intElement);
expect(returnTypeName.staticType, intType);
// method name
expect(node.name.staticElement, same(node.declaredElement));
expect(node.name.staticType, same(node.declaredElement.type));
// method parameter
{
SimpleFormalParameter pNode = node.parameters.parameters[0];
expect(pNode.declaredElement, isNotNull);
expect(pNode.declaredElement.type, doubleType);
TypeName pType = pNode.type;
expect(pType.name.staticElement, doubleElement);
expect(pType.name.staticType, doubleType);
expect(pNode.identifier.staticElement, pNode.declaredElement);
expect(pNode.identifier.staticType, doubleType);
}
}
// publicGetter()
{
MethodDeclaration node = cNode.members[3];
expect(node.declaredElement, isNotNull);
expect(node.declaredElement.type.toString(), '() → int');
// getter return type
TypeName returnType = node.returnType;
SimpleIdentifier returnTypeName = returnType.name;
expect(returnType.type, intType);
expect(returnTypeName.staticElement, intElement);
expect(returnTypeName.staticType, intType);
// getter name
expect(node.name.staticElement, same(node.declaredElement));
expect(node.name.staticType, intType);
}
// publicSetter()
{
MethodDeclaration node = cNode.members[4];
expect(node.declaredElement, isNotNull);
expect(node.declaredElement.type.toString(), '(double) → void');
// setter return type
TypeName returnType = node.returnType;
SimpleIdentifier returnTypeName = returnType.name;
expect(returnType.type, VoidTypeImpl.instance);
expect(returnTypeName.staticElement, isNull);
expect(returnTypeName.staticType, VoidTypeImpl.instance);
// setter name
expect(node.name.staticElement, same(node.declaredElement));
expect(node.name.staticType, doubleType);
// setter parameter
{
SimpleFormalParameter pNode = node.parameters.parameters[0];
expect(pNode.declaredElement, isNotNull);
expect(pNode.declaredElement.type, doubleType);
TypeName pType = pNode.type;
expect(pType.name.staticElement, doubleElement);
expect(pType.name.staticType, doubleType);
expect(pNode.identifier.staticElement, pNode.declaredElement);
expect(pNode.identifier.staticType, doubleType);
}
}
}
test_top_executables_top() async {
String content = r'''
int topFunction(double p) => 0;
int get topGetter => 0;
void set topSetter(double p) {}
''';
addTestFile(content);
await resolveTestFile();
expect(result.path, testFile);
InterfaceType doubleType = typeProvider.doubleType;
InterfaceType intType = typeProvider.intType;
ClassElement doubleElement = doubleType.element;
ClassElement intElement = intType.element;
// topFunction()
{
FunctionDeclaration node = result.unit.declarations[0];
expect(node.declaredElement, isNotNull);
expect(node.declaredElement.type.toString(), '(double) → int');
// function return type
TypeName returnType = node.returnType;
SimpleIdentifier returnTypeName = returnType.name;
expect(returnType.type, intType);
expect(returnTypeName.staticElement, intElement);
expect(returnTypeName.staticType, intType);
// function name
expect(node.name.staticElement, same(node.declaredElement));
expect(node.name.staticType, same(node.declaredElement.type));
// function parameter
{
SimpleFormalParameter pNode =
node.functionExpression.parameters.parameters[0];
expect(pNode.declaredElement, isNotNull);
expect(pNode.declaredElement.type, doubleType);
TypeName pType = pNode.type;
expect(pType.name.staticElement, doubleElement);
expect(pType.name.staticType, doubleType);
expect(pNode.identifier.staticElement, pNode.declaredElement);
expect(pNode.identifier.staticType, doubleType);
}
}
// topGetter()
{
FunctionDeclaration node = result.unit.declarations[1];
expect(node.declaredElement, isNotNull);
expect(node.declaredElement.type.toString(), '() → int');
// getter return type
TypeName returnType = node.returnType;
SimpleIdentifier returnTypeName = returnType.name;
expect(returnType.type, intType);
expect(returnTypeName.staticElement, intElement);
expect(returnTypeName.staticType, intType);
// getter name
expect(node.name.staticElement, same(node.declaredElement));
expect(node.name.staticType, intType);
}
// topSetter()
{
FunctionDeclaration node = result.unit.declarations[2];
expect(node.declaredElement, isNotNull);
expect(node.declaredElement.type.toString(), '(double) → void');
// setter return type
TypeName returnType = node.returnType;
SimpleIdentifier returnTypeName = returnType.name;
expect(returnType.type, VoidTypeImpl.instance);
expect(returnTypeName.staticElement, isNull);
expect(returnTypeName.staticType, VoidTypeImpl.instance);
// setter name
expect(node.name.staticElement, same(node.declaredElement));
expect(node.name.staticType, doubleType);
// setter parameter
{
SimpleFormalParameter pNode =
node.functionExpression.parameters.parameters[0];
expect(pNode.declaredElement, isNotNull);
expect(pNode.declaredElement.type, doubleType);
TypeName pType = pNode.type;
expect(pType.name.staticElement, doubleElement);
expect(pType.name.staticType, doubleType);
expect(pNode.identifier.staticElement, pNode.declaredElement);
expect(pNode.identifier.staticType, doubleType);
}
}
}
test_top_field_class() async {
String content = r'''
class C<T> {
var a = 1;
T b;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
var typeProvider = unitElement.context.typeProvider;
ClassDeclaration cNode = unit.declarations[0];
ClassElement cElement = cNode.declaredElement;
TypeParameterElement tElement = cElement.typeParameters[0];
expect(cElement, same(unitElement.types[0]));
{
FieldElement aElement = cElement.getField('a');
FieldDeclaration aDeclaration = cNode.members[0];
VariableDeclaration aNode = aDeclaration.fields.variables[0];
expect(aNode.declaredElement, same(aElement));
expect(aElement.type, typeProvider.intType);
expect(aNode.name.staticElement, same(aElement));
expect(aNode.name.staticType, same(aElement.type));
Expression aValue = aNode.initializer;
expect(aValue.staticType, typeProvider.intType);
}
{
FieldElement bElement = cElement.getField('b');
FieldDeclaration bDeclaration = cNode.members[1];
TypeName typeName = bDeclaration.fields.type;
SimpleIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, same(tElement));
expect(typeIdentifier.staticType, tElement.type);
VariableDeclaration bNode = bDeclaration.fields.variables[0];
expect(bNode.declaredElement, same(bElement));
expect(bElement.type, tElement.type);
expect(bNode.name.staticElement, same(bElement));
expect(bNode.name.staticType, same(bElement.type));
}
}
test_top_field_class_multiple() async {
String content = r'''
class C {
var a = 1, b = 2.3;
}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
var typeProvider = unitElement.context.typeProvider;
ClassDeclaration cNode = unit.declarations[0];
ClassElement cElement = cNode.declaredElement;
FieldDeclaration fieldDeclaration = cNode.members[0];
{
FieldElement aElement = cElement.getField('a');
VariableDeclaration aNode = fieldDeclaration.fields.variables[0];
expect(aNode.declaredElement, same(aElement));
expect(aElement.type, typeProvider.intType);
expect(aNode.name.staticElement, same(aElement));
expect(aNode.name.staticType, same(aElement.type));
Expression aValue = aNode.initializer;
expect(aValue.staticType, typeProvider.intType);
}
{
FieldElement bElement = cElement.getField('b');
VariableDeclaration bNode = fieldDeclaration.fields.variables[1];
expect(bNode.declaredElement, same(bElement));
expect(bElement.type, typeProvider.doubleType);
expect(bNode.name.staticElement, same(bElement));
expect(bNode.name.staticType, same(bElement.type));
Expression aValue = bNode.initializer;
expect(aValue.staticType, typeProvider.doubleType);
}
}
test_top_field_top() async {
String content = r'''
var a = 1;
double b = 2.3;
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
var typeProvider = unitElement.context.typeProvider;
{
TopLevelVariableDeclaration aDeclaration = unit.declarations[0];
VariableDeclaration aNode = aDeclaration.variables.variables[0];
TopLevelVariableElement aElement = aNode.declaredElement;
expect(aElement, same(unitElement.topLevelVariables[0]));
expect(aElement.type, typeProvider.intType);
expect(aNode.name.staticElement, same(aElement));
expect(aNode.name.staticType, same(aElement.type));
Expression aValue = aNode.initializer;
expect(aValue.staticType, typeProvider.intType);
}
{
TopLevelVariableDeclaration bDeclaration = unit.declarations[1];
VariableDeclaration bNode = bDeclaration.variables.variables[0];
TopLevelVariableElement bElement = bNode.declaredElement;
expect(bElement, same(unitElement.topLevelVariables[1]));
expect(bElement.type, typeProvider.doubleType);
TypeName typeName = bDeclaration.variables.type;
_assertTypeNameSimple(typeName, typeProvider.doubleType);
expect(bNode.name.staticElement, same(bElement));
expect(bNode.name.staticType, same(bElement.type));
Expression aValue = bNode.initializer;
expect(aValue.staticType, typeProvider.doubleType);
}
}
test_top_field_top_multiple() async {
String content = r'''
var a = 1, b = 2.3;
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
var typeProvider = unitElement.context.typeProvider;
TopLevelVariableDeclaration variableDeclaration = unit.declarations[0];
expect(variableDeclaration.variables.type, isNull);
{
VariableDeclaration aNode = variableDeclaration.variables.variables[0];
TopLevelVariableElement aElement = aNode.declaredElement;
expect(aElement, same(unitElement.topLevelVariables[0]));
expect(aElement.type, typeProvider.intType);
expect(aNode.name.staticElement, same(aElement));
expect(aNode.name.staticType, aElement.type);
Expression aValue = aNode.initializer;
expect(aValue.staticType, typeProvider.intType);
}
{
VariableDeclaration bNode = variableDeclaration.variables.variables[1];
TopLevelVariableElement bElement = bNode.declaredElement;
expect(bElement, same(unitElement.topLevelVariables[1]));
expect(bElement.type, typeProvider.doubleType);
expect(bNode.name.staticElement, same(bElement));
expect(bNode.name.staticType, bElement.type);
Expression aValue = bNode.initializer;
expect(aValue.staticType, typeProvider.doubleType);
}
}
test_top_function_namedParameters() async {
addTestFile(r'''
double f(int a, {String b, bool c: 1 == 2}) {}
void main() {
f(1, b: '2', c: true);
}
''');
String fTypeString = '(int, {b: String, c: bool}) → double';
await resolveTestFile();
FunctionDeclaration fDeclaration = result.unit.declarations[0];
FunctionElement fElement = fDeclaration.declaredElement;
InterfaceType doubleType = typeProvider.doubleType;
expect(fElement, isNotNull);
expect(fElement.type.toString(), fTypeString);
expect(fDeclaration.name.staticElement, same(fElement));
expect(fDeclaration.name.staticType, fElement.type);
TypeName fReturnTypeNode = fDeclaration.returnType;
expect(fReturnTypeNode.name.staticElement, same(doubleType.element));
expect(fReturnTypeNode.type, doubleType);
//
// Validate the parameters at the declaration site.
//
List<ParameterElement> elements = fElement.parameters;
expect(elements, hasLength(3));
List<FormalParameter> nodes =
fDeclaration.functionExpression.parameters.parameters;
expect(nodes, hasLength(3));
_assertSimpleParameter(nodes[0], elements[0],
name: 'a',
offset: 13,
kind: ParameterKind.REQUIRED,
type: typeProvider.intType);
DefaultFormalParameter bNode = nodes[1];
_assertDefaultParameter(bNode, elements[1],
name: 'b',
offset: 24,
kind: ParameterKind.NAMED,
type: typeProvider.stringType);
expect(bNode.defaultValue, isNull);
DefaultFormalParameter cNode = nodes[2];
_assertDefaultParameter(cNode, elements[2],
name: 'c',
offset: 32,
kind: ParameterKind.NAMED,
type: typeProvider.boolType);
{
BinaryExpression defaultValue = cNode.defaultValue;
expect(defaultValue.staticElement, isNotNull);
expect(defaultValue.staticType, typeProvider.boolType);
}
//
// Validate the arguments at the call site.
//
FunctionDeclaration mainDeclaration = result.unit.declarations[1];
BlockFunctionBody body = mainDeclaration.functionExpression.body;
ExpressionStatement statement = body.block.statements[0];
MethodInvocation invocation = statement.expression;
List<Expression> arguments = invocation.argumentList.arguments;
_assertArgumentToParameter(arguments[0], fElement.parameters[0]);
_assertArgumentToParameter(arguments[1], fElement.parameters[1]);
_assertArgumentToParameter(arguments[2], fElement.parameters[2]);
}
test_top_functionTypeAlias() async {
String content = r'''
typedef int F<T>(bool a, T b);
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
var typeProvider = unitElement.context.typeProvider;
FunctionTypeAlias alias = unit.declarations[0];
FunctionTypeAliasElement aliasElement = alias.declaredElement;
expect(aliasElement, same(unitElement.functionTypeAliases[0]));
expect(aliasElement.returnType, typeProvider.intType);
_assertTypeNameSimple(alias.returnType, typeProvider.intType);
_assertSimpleParameter(
alias.parameters.parameters[0], aliasElement.parameters[0],
name: 'a',
offset: 22,
kind: ParameterKind.REQUIRED,
type: typeProvider.boolType);
_assertSimpleParameter(
alias.parameters.parameters[1], aliasElement.parameters[1],
name: 'b',
offset: 27,
kind: ParameterKind.REQUIRED,
type: aliasElement.typeParameters[0].type);
}
test_top_typeParameter() async {
String content = r'''
class A {}
class C<T extends A, U extends List<A>, V> {}
''';
addTestFile(content);
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
var typeProvider = unitElement.context.typeProvider;
ClassDeclaration aNode = unit.declarations[0];
ClassElement aElement = aNode.declaredElement;
expect(aElement, same(unitElement.types[0]));
ClassDeclaration cNode = unit.declarations[1];
ClassElement cElement = cNode.declaredElement;
expect(cElement, same(unitElement.types[1]));
{
TypeParameter tNode = cNode.typeParameters.typeParameters[0];
expect(tNode.declaredElement, same(cElement.typeParameters[0]));
TypeName bound = tNode.bound;
expect(bound.type, aElement.type);
SimpleIdentifier boundIdentifier = bound.name;
expect(boundIdentifier.staticElement, same(aElement));
expect(boundIdentifier.staticType, aElement.type);
}
{
var listElement = typeProvider.listType.element;
var listOfA = typeProvider.listType.instantiate([aElement.type]);
TypeParameter uNode = cNode.typeParameters.typeParameters[1];
expect(uNode.declaredElement, same(cElement.typeParameters[1]));
TypeName bound = uNode.bound;
expect(bound.type, listOfA);
SimpleIdentifier listIdentifier = bound.name;
expect(listIdentifier.staticElement, same(listElement));
expect(listIdentifier.staticType, listOfA);
TypeName aTypeName = bound.typeArguments.arguments[0];
expect(aTypeName.type, aElement.type);
SimpleIdentifier aIdentifier = aTypeName.name;
expect(aIdentifier.staticElement, same(aElement));
expect(aIdentifier.staticType, aElement.type);
}
{
TypeParameter vNode = cNode.typeParameters.typeParameters[2];
expect(vNode.declaredElement, same(cElement.typeParameters[2]));
expect(vNode.bound, isNull);
}
}
test_tryCatch() async {
addTestFile(r'''
void main() {
try {} catch (e, st) {
e;
st;
}
try {} on int catch (e, st) {
e;
st;
}
try {} catch (e) {
e;
}
try {} on int catch (e) {
e;
}
try {} on int {}
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
List<Statement> statements = _getMainStatements(result);
// catch (e, st)
{
TryStatement statement = statements[0];
CatchClause catchClause = statement.catchClauses[0];
expect(catchClause.exceptionType, isNull);
SimpleIdentifier exceptionNode = catchClause.exceptionParameter;
LocalVariableElement exceptionElement = exceptionNode.staticElement;
expect(exceptionElement.type, DynamicTypeImpl.instance);
SimpleIdentifier stackNode = catchClause.stackTraceParameter;
LocalVariableElement stackElement = stackNode.staticElement;
expect(stackElement.type, typeProvider.stackTraceType);
List<Statement> catchStatements = catchClause.body.statements;
ExpressionStatement exceptionStatement = catchStatements[0];
SimpleIdentifier exceptionIdentifier = exceptionStatement.expression;
expect(exceptionIdentifier.staticElement, same(exceptionElement));
expect(exceptionIdentifier.staticType, DynamicTypeImpl.instance);
ExpressionStatement stackStatement = catchStatements[1];
SimpleIdentifier stackIdentifier = stackStatement.expression;
expect(stackIdentifier.staticElement, same(stackElement));
expect(stackIdentifier.staticType, typeProvider.stackTraceType);
}
// on int catch (e, st)
{
TryStatement statement = statements[1];
CatchClause catchClause = statement.catchClauses[0];
_assertTypeNameSimple(catchClause.exceptionType, typeProvider.intType);
SimpleIdentifier exceptionNode = catchClause.exceptionParameter;
LocalVariableElement exceptionElement = exceptionNode.staticElement;
expect(exceptionElement.type, typeProvider.intType);
SimpleIdentifier stackNode = catchClause.stackTraceParameter;
LocalVariableElement stackElement = stackNode.staticElement;
expect(stackElement.type, typeProvider.stackTraceType);
List<Statement> catchStatements = catchClause.body.statements;
ExpressionStatement exceptionStatement = catchStatements[0];
SimpleIdentifier exceptionIdentifier = exceptionStatement.expression;
expect(exceptionIdentifier.staticElement, same(exceptionElement));
expect(exceptionIdentifier.staticType, typeProvider.intType);
ExpressionStatement stackStatement = catchStatements[1];
SimpleIdentifier stackIdentifier = stackStatement.expression;
expect(stackIdentifier.staticElement, same(stackElement));
expect(stackIdentifier.staticType, typeProvider.stackTraceType);
}
// catch (e)
{
TryStatement statement = statements[2];
CatchClause catchClause = statement.catchClauses[0];
expect(catchClause.exceptionType, isNull);
expect(catchClause.stackTraceParameter, isNull);
SimpleIdentifier exceptionNode = catchClause.exceptionParameter;
LocalVariableElement exceptionElement = exceptionNode.staticElement;
expect(exceptionElement.type, DynamicTypeImpl.instance);
}
// on int catch (e)
{
TryStatement statement = statements[3];
CatchClause catchClause = statement.catchClauses[0];
_assertTypeNameSimple(catchClause.exceptionType, typeProvider.intType);
expect(catchClause.stackTraceParameter, isNull);
SimpleIdentifier exceptionNode = catchClause.exceptionParameter;
LocalVariableElement exceptionElement = exceptionNode.staticElement;
expect(exceptionElement.type, typeProvider.intType);
}
// on int catch (e)
{
TryStatement statement = statements[4];
CatchClause catchClause = statement.catchClauses[0];
_assertTypeNameSimple(catchClause.exceptionType, typeProvider.intType);
expect(catchClause.exceptionParameter, isNull);
expect(catchClause.stackTraceParameter, isNull);
}
}
test_type_dynamic() async {
addTestFile('''
main() {
dynamic d;
}
''');
await resolveTestFile();
var statements = _getMainStatements(result);
var variableDeclarationStatement =
statements[0] as VariableDeclarationStatement;
var type = variableDeclarationStatement.variables.type as TypeName;
expect(type.type, isDynamicType);
var typeName = type.name;
assertTypeDynamic(typeName);
expect(typeName.staticElement, same(typeProvider.dynamicType.element));
}
test_type_functionTypeAlias() async {
addTestFile(r'''
typedef T F<T>(bool a);
class C {
F<int> f;
}
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
CompilationUnitElement unitElement = unit.declaredElement;
var typeProvider = unitElement.context.typeProvider;
FunctionTypeAlias alias = unit.declarations[0];
GenericTypeAliasElement aliasElement = alias.declaredElement;
FunctionType aliasType = aliasElement.type;
ClassDeclaration cNode = unit.declarations[1];
FieldDeclaration fDeclaration = cNode.members[0];
FunctionType instantiatedAliasType =
aliasType.instantiate([typeProvider.intType]);
TypeName typeName = fDeclaration.fields.type;
expect(typeName.type, instantiatedAliasType);
SimpleIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, same(aliasElement));
expect(typeIdentifier.staticType, instantiatedAliasType);
List<TypeAnnotation> typeArguments = typeName.typeArguments.arguments;
expect(typeArguments, hasLength(1));
_assertTypeNameSimple(typeArguments[0], typeProvider.intType);
}
test_type_void() async {
addTestFile('''
main() {
void v;
}
''');
await resolveTestFile();
var statements = _getMainStatements(result);
var variableDeclarationStatement =
statements[0] as VariableDeclarationStatement;
var type = variableDeclarationStatement.variables.type as TypeName;
expect(type.type, isVoidType);
var typeName = type.name;
expect(typeName.staticType, isVoidType);
expect(typeName.staticElement, isNull);
}
test_typeAnnotation_prefixed() async {
newFile('/test/lib/a.dart', content: 'class A {}');
newFile('/test/lib/b.dart', content: "export 'a.dart';");
newFile('/test/lib/c.dart', content: "export 'a.dart';");
addTestFile(r'''
import 'b.dart' as b;
import 'c.dart' as c;
b.A a1;
c.A a2;
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
ImportElement bImport = unit.declaredElement.library.imports[0];
ImportElement cImport = unit.declaredElement.library.imports[1];
LibraryElement bLibrary = bImport.importedLibrary;
LibraryElement aLibrary = bLibrary.exports[0].exportedLibrary;
ClassElement aClass = aLibrary.getType('A');
{
TopLevelVariableDeclaration declaration = unit.declarations[0];
TypeName typeName = declaration.variables.type;
PrefixedIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, aClass);
expect(typeIdentifier.prefix.name, 'b');
expect(typeIdentifier.prefix.staticElement, same(bImport.prefix));
expect(typeIdentifier.identifier.staticElement, aClass);
}
{
TopLevelVariableDeclaration declaration = unit.declarations[1];
TypeName typeName = declaration.variables.type;
PrefixedIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, aClass);
expect(typeIdentifier.prefix.name, 'c');
expect(typeIdentifier.prefix.staticElement, same(cImport.prefix));
expect(typeIdentifier.identifier.staticElement, aClass);
}
}
test_typeLiteral() async {
addTestFile(r'''
void main() {
int;
F;
}
typedef void F(int p);
''');
await resolveTestFile();
CompilationUnit unit = result.unit;
var typeProvider = unit.declaredElement.context.typeProvider;
FunctionTypeAlias fNode = unit.declarations[1];
FunctionTypeAliasElement fElement = fNode.declaredElement;
var statements = _getMainStatements(result);
{
ExpressionStatement statement = statements[0];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, same(typeProvider.intType.element));
expect(identifier.staticType, typeProvider.typeType);
}
{
ExpressionStatement statement = statements[1];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, same(fElement));
expect(identifier.staticType, typeProvider.typeType);
}
}
test_typeParameter() async {
addTestFile(r'''
class C<T> {
get t => T;
}
''');
await resolveTestFile();
var identifier = findNode.simple('T;');
assertElement(identifier, findElement.typeParameter('T'));
assertType(identifier, 'Type');
}
test_unresolved_instanceCreation_name_11() async {
addTestFile(r'''
int arg1, arg2;
main() {
new Foo<int, double>(arg1, p2: arg2);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticType, isDynamicType);
ConstructorName constructorName = creation.constructorName;
expect(constructorName.name, isNull);
TypeName typeName = constructorName.type;
expect(typeName.type, isDynamicType);
SimpleIdentifier typeIdentifier = typeName.name;
expect(typeIdentifier.staticElement, isNull);
expect(typeIdentifier.staticType, isDynamicType);
assertTypeArguments(typeName.typeArguments, [intType, doubleType]);
_assertInvocationArguments(creation.argumentList,
[checkTopVarRef('arg1'), checkTopVarUndefinedNamedRef('arg2')]);
}
@failingTest
test_unresolved_instanceCreation_name_21() async {
addTestFile(r'''
int arg1, arg2;
main() {
new foo.Bar<int, double>(arg1, p2: arg2);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticType, isDynamicType);
ConstructorName constructorName = creation.constructorName;
expect(constructorName.name, isNull);
TypeName typeName = constructorName.type;
expect(typeName.type, isDynamicType);
PrefixedIdentifier typePrefixed = typeName.name;
expect(typePrefixed.staticElement, isNull);
expect(typePrefixed.staticType, isDynamicType);
SimpleIdentifier typePrefix = typePrefixed.prefix;
expect(typePrefix.staticElement, isNull);
expect(typePrefix.staticType, isNull);
SimpleIdentifier typeIdentifier = typePrefixed.identifier;
expect(typeIdentifier.staticElement, isNull);
expect(typeIdentifier.staticType, isDynamicType);
assertTypeArguments(typeName.typeArguments, [intType, doubleType]);
_assertInvocationArguments(creation.argumentList,
[checkTopVarRef('arg1'), checkTopVarUndefinedNamedRef('arg2')]);
}
@failingTest
test_unresolved_instanceCreation_name_22() async {
addTestFile(r'''
import 'dart:math' as foo;
int arg1, arg2;
main() {
new foo.Bar<int, double>(arg1, p2: arg2);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var unitElement = result.unit.declaredElement;
var foo = unitElement.library.imports[0].prefix;
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticType, isDynamicType);
ConstructorName constructorName = creation.constructorName;
expect(constructorName.name, isNull);
TypeName typeName = constructorName.type;
expect(typeName.type, isDynamicType);
PrefixedIdentifier typePrefixed = typeName.name;
expect(typePrefixed.staticElement, isNull);
expect(typePrefixed.staticType, isDynamicType);
SimpleIdentifier typePrefix = typePrefixed.prefix;
expect(typePrefix.staticElement, same(foo));
expect(typePrefix.staticType, isNull);
SimpleIdentifier typeIdentifier = typePrefixed.identifier;
expect(typeIdentifier.staticElement, isNull);
expect(typeIdentifier.staticType, isDynamicType);
assertTypeArguments(typeName.typeArguments, [intType, doubleType]);
_assertInvocationArguments(creation.argumentList,
[checkTopVarRef('arg1'), checkTopVarUndefinedNamedRef('arg2')]);
}
@failingTest
test_unresolved_instanceCreation_name_31() async {
addTestFile(r'''
int arg1, arg2;
main() {
new foo.Bar<int, double>.baz(arg1, p2: arg2);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticType, isDynamicType);
ConstructorName constructorName = creation.constructorName;
TypeName typeName = constructorName.type;
expect(typeName.type, isDynamicType);
PrefixedIdentifier typePrefixed = typeName.name;
assertElementNull(typePrefixed);
assertTypeNull(typePrefixed);
SimpleIdentifier typePrefix = typePrefixed.prefix;
assertElementNull(typePrefix);
assertTypeNull(typePrefix);
SimpleIdentifier typeIdentifier = typePrefixed.identifier;
assertElementNull(typeIdentifier);
assertTypeNull(typeIdentifier);
assertElementNull(constructorName.name);
assertTypeNull(constructorName.name);
assertTypeArguments(typeName.typeArguments, [intType, doubleType]);
_assertInvocationArguments(creation.argumentList,
[checkTopVarRef('arg1'), checkTopVarUndefinedNamedRef('arg2')]);
}
@failingTest
test_unresolved_instanceCreation_name_32() async {
addTestFile(r'''
import 'dart:math' as foo;
int arg1, arg2;
main() {
new foo.Bar<int, double>.baz(arg1, p2: arg2);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var unitElement = result.unit.declaredElement;
var mathImport = unitElement.library.imports[0];
var foo = mathImport.prefix;
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticType, isDynamicType);
ConstructorName constructorName = creation.constructorName;
TypeName typeName = constructorName.type;
expect(typeName.type, isDynamicType);
PrefixedIdentifier typePrefixed = typeName.name;
assertElementNull(typePrefixed);
assertTypeNull(typePrefixed);
SimpleIdentifier typePrefix = typePrefixed.prefix;
assertElement(typePrefix, foo);
assertTypeNull(typePrefix);
SimpleIdentifier typeIdentifier = typePrefixed.identifier;
assertElementNull(typeIdentifier);
assertTypeNull(typeIdentifier);
assertElementNull(constructorName.name);
assertTypeNull(constructorName.name);
assertTypeArguments(typeName.typeArguments, [intType, doubleType]);
_assertInvocationArguments(creation.argumentList,
[checkTopVarRef('arg1'), checkTopVarUndefinedNamedRef('arg2')]);
}
@failingTest
test_unresolved_instanceCreation_name_33() async {
addTestFile(r'''
import 'dart:math' as foo;
int arg1, arg2;
main() {
new foo.Random<int, double>.baz(arg1, p2: arg2);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var unitElement = result.unit.declaredElement;
var mathImport = unitElement.library.imports[0];
var foo = mathImport.prefix;
var randomElement = mathImport.importedLibrary.getType('Random');
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
InstanceCreationExpression creation = statement.expression;
expect(creation.staticType, randomElement.type);
ConstructorName constructorName = creation.constructorName;
TypeName typeName = constructorName.type;
assertType(typeName, 'Random');
PrefixedIdentifier typePrefixed = typeName.name;
assertElement(typePrefixed, randomElement);
assertTypeNull(typePrefixed);
SimpleIdentifier typePrefix = typePrefixed.prefix;
assertElement(typePrefix, foo);
assertTypeNull(typePrefix);
SimpleIdentifier typeIdentifier = typePrefixed.identifier;
assertElement(typeIdentifier, randomElement);
assertTypeNull(typeIdentifier);
assertElementNull(constructorName.name);
assertTypeNull(constructorName.name);
assertTypeArguments(typeName.typeArguments, [intType, doubleType]);
_assertInvocationArguments(creation.argumentList,
[checkTopVarRef('arg1'), checkTopVarUndefinedNamedRef('arg2')]);
}
test_unresolved_methodInvocation_noTarget() async {
addTestFile(r'''
int arg1, arg2;
main() {
bar<int, double>(arg1, p2: arg2);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.target, isNull);
expect(invocation.staticType, isDynamicType);
expect(invocation.staticInvokeType, isDynamicType);
SimpleIdentifier name = invocation.methodName;
expect(name.staticElement, isNull);
expect(name.staticType, isDynamicType);
assertTypeArguments(invocation.typeArguments, [intType, doubleType]);
_assertInvocationArguments(invocation.argumentList,
[checkTopVarRef('arg1'), checkTopVarUndefinedNamedRef('arg2')]);
}
test_unresolved_methodInvocation_target_resolved() async {
addTestFile(r'''
Object foo;
int arg1, arg2;
main() {
foo.bar<int, double>(arg1, p2: arg2);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
TopLevelVariableElement foo = _getTopLevelVariable(result, 'foo');
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
MethodInvocation invocation = statement.expression;
expect(invocation.staticType, isDynamicType);
assertUnresolvedInvokeType(invocation.staticInvokeType);
SimpleIdentifier target = invocation.target;
expect(target.staticElement, same(foo.getter));
expect(target.staticType, typeProvider.objectType);
SimpleIdentifier name = invocation.methodName;
expect(name.staticElement, isNull);
assertUnresolvedInvokeType(name.staticType);
assertTypeArguments(invocation.typeArguments, [intType, doubleType]);
_assertInvocationArguments(invocation.argumentList,
[checkTopVarRef('arg1'), checkTopVarUndefinedNamedRef('arg2')]);
}
test_unresolved_methodInvocation_target_unresolved() async {
addTestFile(r'''
int arg1, arg2;
main() {
foo.bar<int, double>(arg1, p2: arg2);
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var invocation = findNode.methodInvocation('foo.bar');
assertTypeDynamic(invocation);
assertUnresolvedInvokeType(invocation.staticInvokeType);
SimpleIdentifier target = invocation.target;
assertElementNull(target);
assertTypeDynamic(target);
SimpleIdentifier name = invocation.methodName;
assertElementNull(name);
assertUnresolvedInvokeType(name.staticType);
assertTypeArguments(invocation.typeArguments, [intType, doubleType]);
_assertInvocationArguments(invocation.argumentList,
[checkTopVarRef('arg1'), checkTopVarUndefinedNamedRef('arg2')]);
}
test_unresolved_postfix_operand() async {
addTestFile(r'''
main() {
a++;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var postfix = findNode.postfix('a++');
assertElementNull(postfix);
assertTypeDynamic(postfix);
SimpleIdentifier aRef = postfix.operand;
assertElementNull(aRef);
assertTypeDynamic(aRef);
}
test_unresolved_postfix_operator() async {
addTestFile(r'''
A a;
main() {
a++;
}
class A {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var postfix = findNode.postfix('a++');
assertElementNull(postfix);
assertType(postfix, 'A');
SimpleIdentifier aRef = postfix.operand;
assertElement(aRef, findElement.topSet('a'));
assertType(aRef, 'A');
}
test_unresolved_prefix_operand() async {
addTestFile(r'''
main() {
++a;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var prefix = findNode.prefix('++a');
assertElementNull(prefix);
assertTypeDynamic(prefix);
SimpleIdentifier aRef = prefix.operand;
assertElementNull(aRef);
assertTypeDynamic(aRef);
}
test_unresolved_prefix_operator() async {
addTestFile(r'''
A a;
main() {
++a;
}
class A {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var prefix = findNode.prefix('++a');
assertElementNull(prefix);
assertTypeDynamic(prefix);
SimpleIdentifier aRef = prefix.operand;
assertElement(aRef, findElement.topSet('a'));
assertType(aRef, 'A');
}
test_unresolved_prefixedIdentifier_identifier() async {
addTestFile(r'''
Object foo;
main() {
foo.bar;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
TopLevelVariableElement foo = _getTopLevelVariable(result, 'foo');
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
PrefixedIdentifier prefixed = statement.expression;
expect(prefixed.staticElement, isNull);
expect(prefixed.staticType, isDynamicType);
SimpleIdentifier prefix = prefixed.prefix;
expect(prefix.staticElement, same(foo.getter));
expect(prefix.staticType, typeProvider.objectType);
SimpleIdentifier identifier = prefixed.identifier;
expect(identifier.staticElement, isNull);
expect(identifier.staticType, isDynamicType);
}
test_unresolved_prefixedIdentifier_prefix() async {
addTestFile(r'''
main() {
foo.bar;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
PrefixedIdentifier prefixed = statement.expression;
expect(prefixed.staticElement, isNull);
expect(prefixed.staticType, isDynamicType);
SimpleIdentifier prefix = prefixed.prefix;
expect(prefix.staticElement, isNull);
expect(prefix.staticType, isDynamicType);
SimpleIdentifier identifier = prefixed.identifier;
expect(identifier.staticElement, isNull);
expect(identifier.staticType, isDynamicType);
}
test_unresolved_propertyAccess_1() async {
addTestFile(r'''
main() {
foo.bar.baz;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
PropertyAccess propertyAccess = statement.expression;
expect(propertyAccess.staticType, isDynamicType);
{
PrefixedIdentifier prefixed = propertyAccess.target;
expect(prefixed.staticElement, isNull);
expect(prefixed.staticType, isDynamicType);
SimpleIdentifier prefix = prefixed.prefix;
expect(prefix.staticElement, isNull);
expect(prefix.staticType, isDynamicType);
SimpleIdentifier identifier = prefixed.identifier;
expect(identifier.staticElement, isNull);
expect(identifier.staticType, isDynamicType);
}
SimpleIdentifier property = propertyAccess.propertyName;
expect(property.staticElement, isNull);
expect(property.staticType, isDynamicType);
}
test_unresolved_propertyAccess_2() async {
addTestFile(r'''
Object foo;
main() {
foo.bar.baz;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
TopLevelVariableElement foo = _getTopLevelVariable(result, 'foo');
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
PropertyAccess propertyAccess = statement.expression;
expect(propertyAccess.staticType, isDynamicType);
{
PrefixedIdentifier prefixed = propertyAccess.target;
expect(prefixed.staticElement, isNull);
expect(prefixed.staticType, isDynamicType);
SimpleIdentifier prefix = prefixed.prefix;
expect(prefix.staticElement, same(foo.getter));
expect(prefix.staticType, typeProvider.objectType);
SimpleIdentifier identifier = prefixed.identifier;
expect(identifier.staticElement, isNull);
expect(identifier.staticType, isDynamicType);
}
SimpleIdentifier property = propertyAccess.propertyName;
expect(property.staticElement, isNull);
expect(property.staticType, isDynamicType);
}
test_unresolved_propertyAccess_3() async {
addTestFile(r'''
Object foo;
main() {
foo.hashCode.baz;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
PropertyAccessorElement objectHashCode =
typeProvider.objectType.getGetter('hashCode');
TopLevelVariableElement foo = _getTopLevelVariable(result, 'foo');
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
PropertyAccess propertyAccess = statement.expression;
expect(propertyAccess.staticType, isDynamicType);
{
PrefixedIdentifier prefixed = propertyAccess.target;
expect(prefixed.staticElement, same(objectHashCode));
expect(prefixed.staticType, typeProvider.intType);
SimpleIdentifier prefix = prefixed.prefix;
expect(prefix.staticElement, same(foo.getter));
expect(prefix.staticType, typeProvider.objectType);
SimpleIdentifier identifier = prefixed.identifier;
expect(identifier.staticElement, same(objectHashCode));
expect(identifier.staticType, typeProvider.intType);
}
SimpleIdentifier property = propertyAccess.propertyName;
expect(property.staticElement, isNull);
expect(property.staticType, isDynamicType);
}
test_unresolved_redirectingFactory_1() async {
addTestFile(r'''
class A {
factory A() = B;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
}
test_unresolved_redirectingFactory_22() async {
addTestFile(r'''
class A {
factory A() = B.named;
}
class B {}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
var bRef = findNode.simple('B.');
assertElement(bRef, findElement.class_('B'));
assertType(bRef, 'B');
var namedRef = findNode.simple('named;');
assertElementNull(namedRef);
assertTypeNull(namedRef);
}
test_unresolved_simpleIdentifier() async {
addTestFile(r'''
main() {
foo;
}
''');
await resolveTestFile();
expect(result.errors, isNotEmpty);
List<Statement> statements = _getMainStatements(result);
ExpressionStatement statement = statements[0];
SimpleIdentifier identifier = statement.expression;
expect(identifier.staticElement, isNull);
expect(identifier.staticType, isDynamicType);
}
/// Assert that the [argument] is associated with the [expected]. If the
/// [argument] is a [NamedExpression], the name must be resolved to the
/// parameter.
void _assertArgumentToParameter(
Expression argument, ParameterElement expected,
{DartType memberType}) {
ParameterElement actual = argument.staticParameterElement;
if (memberType != null) {
expect(actual.type, memberType);
}
ParameterElement base = actual;
if (actual is ParameterMember) {
base = actual.baseElement;
// Unwrap ParameterMember one more time.
// By some reason we wrap in twice.
if (base is ParameterMember) {
ParameterMember member = base;
base = member.baseElement;
}
}
ParameterElement baseActual = base;
expect(baseActual, same(expected));
if (argument is NamedExpression) {
SimpleIdentifier name = argument.name.label;
expect(name.staticElement, same(actual));
expect(name.staticType, isNull);
}
}
/// Assert that the [argument] has the [expectedType]. If the [argument] is
/// a [NamedExpression], the name must be resolved to the same parameter.
void _assertArgumentToParameter2(Expression argument, String expectedType,
{DartType memberType}) {
ParameterElement actual = argument.staticParameterElement;
var actualType = actual.type;
expect('$actualType', expectedType);
if (argument is NamedExpression) {
SimpleIdentifier name = argument.name.label;
expect(name.staticElement, same(actual));
expect(name.staticType, isNull);
}
}
/// Assert that the given [creation] creates instance of the [classElement].
/// Limitations: no import prefix, no type arguments, unnamed constructor.
void _assertConstructorInvocation(
InstanceCreationExpression creation, ClassElement classElement) {
assertType(creation, classElement.name);
var constructorName = creation.constructorName;
var constructorElement = classElement.unnamedConstructor;
expect(constructorName.staticElement, constructorElement);
var typeName = constructorName.type;
expect(typeName.typeArguments, isNull);
SimpleIdentifier typeIdentifier = typeName.name;
assertElement(typeIdentifier, classElement);
assertType(typeIdentifier, classElement.name);
// Only unnamed constructors are supported now.
expect(constructorName.name, isNull);
}
void _assertDefaultParameter(
DefaultFormalParameter node, ParameterElement element,
{String name, int offset, ParameterKind kind, DartType type}) {
expect(node, isNotNull);
NormalFormalParameter normalNode = node.parameter;
_assertSimpleParameter(normalNode, element,
name: name, offset: offset, kind: kind, type: type);
}
/// Test that [argumentList] has exactly two arguments - required `arg1`, and
/// unresolved named `arg2`, both are the reference to top-level variables.
void _assertInvocationArguments(ArgumentList argumentList,
List<void Function(Expression)> argumentCheckers) {
expect(argumentList.arguments, hasLength(argumentCheckers.length));
for (int i = 0; i < argumentCheckers.length; i++) {
argumentCheckers[i](argumentList.arguments[i]);
}
}
void _assertParameterElement(ParameterElement element,
{String name, int offset, ParameterKind kind, DartType type}) {
expect(element, isNotNull);
expect(name, isNotNull);
expect(offset, isNotNull);
expect(kind, isNotNull);
expect(type, isNotNull);
expect(element.name, name);
expect(element.nameOffset, offset);
// ignore: deprecated_member_use_from_same_package
expect(element.parameterKind, kind);
expect(element.type, type);
}
void _assertSimpleParameter(
SimpleFormalParameter node, ParameterElement element,
{String name, int offset, ParameterKind kind, DartType type}) {
_assertParameterElement(element,
name: name, offset: offset, kind: kind, type: type);
expect(node, isNotNull);
expect(node.declaredElement, same(element));
expect(node.identifier.staticElement, same(element));
TypeName typeName = node.type;
if (typeName != null) {
expect(typeName.type, type);
expect(typeName.name.staticElement, same(type.element));
}
}
void _assertTypeNameSimple(TypeName typeName, DartType type) {
expect(typeName.type, type);
SimpleIdentifier identifier = typeName.name;
expect(identifier.staticElement, same(type.element));
expect(identifier.staticType, type);
}
List<Statement> _getMainStatements(ResolvedUnitResult result) {
for (var declaration in result.unit.declarations) {
if (declaration is FunctionDeclaration &&
declaration.name.name == 'main') {
BlockFunctionBody body = declaration.functionExpression.body;
return body.block.statements;
}
}
fail('Not found main() in ${result.unit}');
}
TopLevelVariableElement _getTopLevelVariable(
ResolvedUnitResult result, String name) {
for (var variable in result.unit.declaredElement.topLevelVariables) {
if (variable.name == name) {
return variable;
}
}
fail('Not found $name');
}
// String _p(String path) => convertPath(path);
}
/**
* Resolution tests that are run with all of the experiments enabled.
*/
@reflectiveTest
class DriverResolutionWithExperimentsTest extends BaseAnalysisDriverTest {
AnalysisOptionsImpl createAnalysisOptions() => super.createAnalysisOptions()
..enabledExperiments = _computeActiveExperimentNames();
test_binaryExpression_gtGtGt() async {
addTestFile('''
class A {
A operator >>>(int amount) => this;
}
f(A a) {
a >>> 3;
}
''');
var result = await driver.getResult(testFile);
CompilationUnit unit = result.unit;
MethodDeclaration declaration =
(unit.declarations[0] as ClassDeclaration).members[0];
ExecutableElement operatorElement = declaration.declaredElement;
expect(operatorElement.name, '>>>');
ExpressionStatement statement =
((unit.declarations[1] as FunctionDeclaration).functionExpression.body
as BlockFunctionBody)
.block
.statements[0];
BinaryExpression binary = statement.expression;
expect(binary.operator.type, TokenType.GT_GT_GT);
expect(binary.staticElement, operatorElement);
}
List<String> _computeActiveExperimentNames() {
var result = <String>[];
for (var feature in ExperimentStatus.knownFeatures.values) {
if (!feature.isExpired) {
result.add(feature.enableString);
}
}
return result;
}
}