| // Copyright (c) 2017, 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:analysis_server/src/computer/computer_outline.dart'; |
| import 'package:analyzer/src/dart/analysis/driver.dart'; |
| import 'package:analyzer_plugin/protocol/protocol_common.dart'; |
| import 'package:meta/meta.dart'; |
| import 'package:test/test.dart'; |
| import 'package:test_reflective_loader/test_reflective_loader.dart'; |
| |
| import '../../abstract_context.dart'; |
| |
| main() { |
| defineReflectiveSuite(() { |
| defineReflectiveTests(FlutterOutlineComputerTest); |
| defineReflectiveTests(OutlineComputerTest); |
| }); |
| } |
| |
| class AbstractOutlineComputerTest extends AbstractContextTest { |
| String testPath; |
| String testCode; |
| |
| @override |
| void setUp() { |
| super.setUp(); |
| testPath = resourceProvider.convertPath('/test.dart'); |
| } |
| |
| Future<Outline> _computeOutline(String code) async { |
| testCode = code; |
| newFile(testPath, content: code); |
| AnalysisResult analysisResult = await driver.getResult(testPath); |
| return new DartUnitOutlineComputer( |
| testPath, analysisResult.lineInfo, analysisResult.unit, |
| withBasicFlutter: true) |
| .compute(); |
| } |
| } |
| |
| @reflectiveTest |
| class FlutterOutlineComputerTest extends AbstractOutlineComputerTest { |
| @override |
| void setUp() { |
| super.setUp(); |
| addFlutterPackage(); |
| } |
| |
| test_columnWithChildren() async { |
| Outline unitOutline = await _computeOutline(''' |
| import 'package:flutter/widgets.dart'; |
| |
| class MyWidget extends StatelessWidget { |
| @override |
| Widget build(BuildContext context) { |
| return new Column(children: [ |
| const Text('aaa'), |
| const Text('bbb'), |
| ]); // Column |
| } |
| } |
| '''); |
| expect(_toText(unitOutline), r''' |
| MyWidget |
| build |
| Column |
| Text('aaa') |
| Text('bbb') |
| '''); |
| var myWidget = unitOutline.children[0]; |
| var build = myWidget.children[0]; |
| |
| var columnOutline = build.children[0]; |
| { |
| int offset = testCode.indexOf('new Column'); |
| int length = testCode.indexOf('; // Column') - offset; |
| _expect(columnOutline, |
| name: 'Column', |
| elementOffset: offset, |
| offset: offset, |
| length: length); |
| } |
| |
| { |
| var textOutline = columnOutline.children[0]; |
| String text = "const Text('aaa')"; |
| int offset = testCode.indexOf(text); |
| _expect(textOutline, |
| name: "Text('aaa')", |
| elementOffset: offset, |
| offset: offset, |
| length: text.length); |
| } |
| |
| { |
| var textOutline = columnOutline.children[1]; |
| String text = "const Text('bbb')"; |
| int offset = testCode.indexOf(text); |
| _expect(textOutline, |
| name: "Text('bbb')", |
| elementOffset: offset, |
| offset: offset, |
| length: text.length); |
| } |
| } |
| |
| void _expect(Outline outline, |
| {@required String name, |
| @required int elementOffset, |
| @required int offset, |
| @required int length}) { |
| Element element = outline.element; |
| expect(element.name, name); |
| expect(element.location.offset, elementOffset); |
| expect(outline.offset, offset); |
| expect(outline.length, length); |
| } |
| |
| static String _toText(Outline outline) { |
| var buffer = new StringBuffer(); |
| |
| void writeOutline(Outline outline, String indent) { |
| buffer.write(indent); |
| buffer.writeln(outline.element.name); |
| for (var child in outline.children ?? const []) { |
| writeOutline(child, '$indent '); |
| } |
| } |
| |
| for (var child in outline.children) { |
| writeOutline(child, ''); |
| } |
| return buffer.toString(); |
| } |
| } |
| |
| @reflectiveTest |
| class OutlineComputerTest extends AbstractOutlineComputerTest { |
| test_class() async { |
| Outline unitOutline = await _computeOutline(''' |
| abstract class A<K, V> { |
| int fa, fb; |
| String fc; |
| A(int i, String s); |
| A.name(num p); |
| A._privateName(num p); |
| static String ma(int pa) => null; |
| _mb(int pb); |
| R mc<R, P>(P p) {} |
| String get propA => null; |
| set propB(int v) {} |
| } |
| class B { |
| B(int p); |
| } |
| String fa(int pa) => null; |
| R fb<R, P>(P p) {} |
| '''); |
| List<Outline> topOutlines = unitOutline.children; |
| expect(topOutlines, hasLength(4)); |
| // A |
| { |
| Outline outline_A = topOutlines[0]; |
| Element element_A = outline_A.element; |
| expect(element_A.kind, ElementKind.CLASS); |
| expect(element_A.name, "A"); |
| expect(element_A.typeParameters, "<K, V>"); |
| { |
| Location location = element_A.location; |
| expect(location.offset, testCode.indexOf("A<K, V> {")); |
| expect(location.length, 1); |
| } |
| expect(element_A.parameters, null); |
| expect(element_A.returnType, null); |
| // A children |
| List<Outline> outlines_A = outline_A.children; |
| expect(outlines_A, hasLength(11)); |
| { |
| Outline outline = outlines_A[0]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FIELD); |
| expect(element.name, "fa"); |
| expect(element.parameters, isNull); |
| expect(element.returnType, "int"); |
| } |
| { |
| Outline outline = outlines_A[1]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FIELD); |
| expect(element.name, "fb"); |
| expect(element.parameters, isNull); |
| expect(element.returnType, "int"); |
| } |
| { |
| Outline outline = outlines_A[2]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FIELD); |
| expect(element.name, "fc"); |
| expect(element.parameters, isNull); |
| expect(element.returnType, "String"); |
| } |
| { |
| Outline outline = outlines_A[3]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.CONSTRUCTOR); |
| expect(element.name, "A"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("A(int i, String s);")); |
| expect(location.length, "A".length); |
| } |
| expect(element.parameters, "(int i, String s)"); |
| expect(element.returnType, isNull); |
| expect(element.isAbstract, isFalse); |
| expect(element.isStatic, isFalse); |
| } |
| { |
| Outline outline = outlines_A[4]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.CONSTRUCTOR); |
| expect(element.name, "A.name"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("name(num p);")); |
| expect(location.length, "name".length); |
| } |
| expect(element.parameters, "(num p)"); |
| expect(element.returnType, isNull); |
| expect(element.isAbstract, isFalse); |
| expect(element.isStatic, isFalse); |
| } |
| { |
| Outline outline = outlines_A[5]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.CONSTRUCTOR); |
| expect(element.name, "A._privateName"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("_privateName(num p);")); |
| expect(location.length, "_privateName".length); |
| } |
| expect(element.parameters, "(num p)"); |
| expect(element.returnType, isNull); |
| expect(element.isAbstract, isFalse); |
| expect(element.isStatic, isFalse); |
| } |
| { |
| Outline outline = outlines_A[6]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.METHOD); |
| expect(element.name, "ma"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("ma(int pa) => null;")); |
| expect(location.length, "ma".length); |
| } |
| expect(element.parameters, "(int pa)"); |
| expect(element.returnType, "String"); |
| expect(element.isAbstract, isFalse); |
| expect(element.isStatic, isTrue); |
| } |
| { |
| Outline outline = outlines_A[7]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.METHOD); |
| expect(element.name, "_mb"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("_mb(int pb);")); |
| expect(location.length, "_mb".length); |
| } |
| expect(element.parameters, "(int pb)"); |
| expect(element.returnType, ""); |
| expect(element.isAbstract, isTrue); |
| expect(element.isStatic, isFalse); |
| } |
| { |
| Outline outline = outlines_A[8]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.METHOD); |
| expect(element.name, "mc"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("mc<R, P>")); |
| expect(location.length, "mc".length); |
| } |
| expect(element.parameters, "(P p)"); |
| expect(element.returnType, "R"); |
| expect(element.typeParameters, "<R, P>"); |
| expect(element.isAbstract, isFalse); |
| expect(element.isStatic, isFalse); |
| } |
| { |
| Outline outline = outlines_A[9]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.GETTER); |
| expect(element.name, "propA"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("propA => null;")); |
| expect(location.length, "propA".length); |
| } |
| expect(element.parameters, isNull); |
| expect(element.returnType, "String"); |
| } |
| { |
| Outline outline = outlines_A[10]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.SETTER); |
| expect(element.name, "propB"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("propB(int v) {}")); |
| expect(location.length, "propB".length); |
| } |
| expect(element.parameters, "(int v)"); |
| expect(element.returnType, ""); |
| } |
| } |
| // B |
| { |
| Outline outline_B = topOutlines[1]; |
| Element element_B = outline_B.element; |
| expect(element_B.kind, ElementKind.CLASS); |
| expect(element_B.name, "B"); |
| expect(element_B.typeParameters, isNull); |
| { |
| Location location = element_B.location; |
| expect(location.offset, testCode.indexOf("B {")); |
| expect(location.length, 1); |
| } |
| expect(element_B.parameters, null); |
| expect(element_B.returnType, null); |
| // B children |
| List<Outline> outlines_B = outline_B.children; |
| expect(outlines_B, hasLength(1)); |
| { |
| Outline outline = outlines_B[0]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.CONSTRUCTOR); |
| expect(element.name, "B"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("B(int p);")); |
| expect(location.length, "B".length); |
| } |
| expect(element.parameters, "(int p)"); |
| expect(element.returnType, isNull); |
| } |
| } |
| { |
| Outline outline = topOutlines[2]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FUNCTION); |
| expect(element.name, "fa"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("fa(int pa)")); |
| expect(location.length, "ma".length); |
| } |
| expect(element.parameters, "(int pa)"); |
| expect(element.returnType, "String"); |
| expect(element.isAbstract, isFalse); |
| expect(element.isStatic, isTrue); |
| } |
| { |
| Outline outline = topOutlines[3]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FUNCTION); |
| expect(element.name, "fb"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("fb<R, P>")); |
| expect(location.length, "fb".length); |
| } |
| expect(element.parameters, "(P p)"); |
| expect(element.returnType, "R"); |
| expect(element.typeParameters, "<R, P>"); |
| expect(element.isAbstract, isFalse); |
| expect(element.isStatic, isTrue); |
| } |
| } |
| |
| test_enum() async { |
| Outline unitOutline = await _computeOutline(''' |
| enum MyEnum { |
| A, B, C |
| } |
| '''); |
| List<Outline> topOutlines = unitOutline.children; |
| expect(topOutlines, hasLength(1)); |
| // MyEnum |
| { |
| Outline outline_MyEnum = topOutlines[0]; |
| Element element_MyEnum = outline_MyEnum.element; |
| expect(element_MyEnum.kind, ElementKind.ENUM); |
| expect(element_MyEnum.name, "MyEnum"); |
| { |
| Location location = element_MyEnum.location; |
| expect(location.offset, testCode.indexOf("MyEnum {")); |
| expect(location.length, 'MyEnum'.length); |
| } |
| expect(element_MyEnum.parameters, null); |
| expect(element_MyEnum.returnType, null); |
| // MyEnum children |
| List<Outline> outlines_MyEnum = outline_MyEnum.children; |
| expect(outlines_MyEnum, hasLength(3)); |
| _isEnumConstant(outlines_MyEnum[0], 'A'); |
| _isEnumConstant(outlines_MyEnum[1], 'B'); |
| _isEnumConstant(outlines_MyEnum[2], 'C'); |
| } |
| } |
| |
| test_groupAndTest() async { |
| Outline outline = await _computeOutline(''' |
| void group(name, closure) {} |
| void test(name) {} |
| void main() { |
| group('group1', () { |
| group('group1_1', () { |
| test('test1_1_1'); |
| test('test1_1_2'); |
| }); |
| group('group1_2', () { |
| test('test1_2_1'); |
| }); |
| }); |
| group('group2', () { |
| test('test2_1'); |
| test('test2_2'); |
| }); |
| } |
| '''); |
| // unit |
| List<Outline> unit_children = outline.children; |
| expect(unit_children, hasLength(3)); |
| // main |
| Outline main_outline = unit_children[2]; |
| _expect(main_outline, |
| kind: ElementKind.FUNCTION, |
| name: 'main', |
| offset: testCode.indexOf("main() {"), |
| parameters: '()', |
| returnType: 'void'); |
| List<Outline> main_children = main_outline.children; |
| expect(main_children, hasLength(2)); |
| // group1 |
| Outline group1_outline = main_children[0]; |
| _expect(group1_outline, |
| kind: ElementKind.UNIT_TEST_GROUP, |
| length: 5, |
| name: 'group("group1")', |
| offset: testCode.indexOf("group('group1'")); |
| List<Outline> group1_children = group1_outline.children; |
| expect(group1_children, hasLength(2)); |
| // group1_1 |
| Outline group1_1_outline = group1_children[0]; |
| _expect(group1_1_outline, |
| kind: ElementKind.UNIT_TEST_GROUP, |
| length: 5, |
| name: 'group("group1_1")', |
| offset: testCode.indexOf("group('group1_1'")); |
| List<Outline> group1_1_children = group1_1_outline.children; |
| expect(group1_1_children, hasLength(2)); |
| // test1_1_1 |
| Outline test1_1_1_outline = group1_1_children[0]; |
| _expect(test1_1_1_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 4, |
| name: 'test("test1_1_1")', |
| offset: testCode.indexOf("test('test1_1_1'")); |
| // test1_1_1 |
| Outline test1_1_2_outline = group1_1_children[1]; |
| _expect(test1_1_2_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 4, |
| name: 'test("test1_1_2")', |
| offset: testCode.indexOf("test('test1_1_2'")); |
| // group1_2 |
| Outline group1_2_outline = group1_children[1]; |
| _expect(group1_2_outline, |
| kind: ElementKind.UNIT_TEST_GROUP, |
| length: 5, |
| name: 'group("group1_2")', |
| offset: testCode.indexOf("group('group1_2'")); |
| List<Outline> group1_2_children = group1_2_outline.children; |
| expect(group1_2_children, hasLength(1)); |
| // test2_1 |
| Outline test1_2_1_outline = group1_2_children[0]; |
| _expect(test1_2_1_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 4, |
| name: 'test("test1_2_1")', |
| offset: testCode.indexOf("test('test1_2_1'")); |
| // group2 |
| Outline group2_outline = main_children[1]; |
| _expect(group2_outline, |
| kind: ElementKind.UNIT_TEST_GROUP, |
| length: 5, |
| name: 'group("group2")', |
| offset: testCode.indexOf("group('group2'")); |
| List<Outline> group2_children = group2_outline.children; |
| expect(group2_children, hasLength(2)); |
| // test2_1 |
| Outline test2_1_outline = group2_children[0]; |
| _expect(test2_1_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 4, |
| name: 'test("test2_1")', |
| offset: testCode.indexOf("test('test2_1'")); |
| // test2_2 |
| Outline test2_2_outline = group2_children[1]; |
| _expect(test2_2_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 4, |
| name: 'test("test2_2")', |
| offset: testCode.indexOf("test('test2_2'")); |
| } |
| |
| /** |
| * Code like this caused NPE in the past. |
| * |
| * https://code.google.com/p/dart/issues/detail?id=21373 |
| */ |
| test_invalidGetterInConstructor() async { |
| Outline outline = await _computeOutline(''' |
| class A { |
| A() { |
| get badGetter { |
| const int CONST = 0; |
| } |
| } |
| } |
| '''); |
| expect(outline, isNotNull); |
| } |
| |
| /** |
| * Code like this caused Dart2 failure in the past. |
| * |
| * https://github.com/dart-lang/sdk/issues/33228 |
| */ |
| test_invocation_ofParameter() async { |
| Outline outline = await _computeOutline(''' |
| main(p()) { |
| p(); |
| } |
| '''); |
| expect(outline, isNotNull); |
| } |
| |
| test_isTest_isTestGroup() async { |
| addMetaPackageSource(); |
| Outline outline = await _computeOutline(''' |
| import 'package:meta/meta.dart'; |
| |
| @isTestGroup |
| void myGroup(name, body()) {} |
| |
| @isTest |
| void myTest(name) {} |
| |
| void main() { |
| myGroup('group1', () { |
| myGroup('group1_1', () { |
| myTest('test1_1_1'); |
| myTest('test1_1_2'); |
| }); |
| myGroup('group1_2', () { |
| myTest('test1_2_1'); |
| }); |
| }); |
| myGroup('group2', () { |
| myTest('test2_1'); |
| myTest('test2_2'); |
| }); |
| } |
| '''); |
| // unit |
| List<Outline> unit_children = outline.children; |
| expect(unit_children, hasLength(3)); |
| // main |
| Outline main_outline = unit_children[2]; |
| _expect(main_outline, |
| kind: ElementKind.FUNCTION, |
| name: 'main', |
| offset: testCode.indexOf("main() {"), |
| parameters: '()', |
| returnType: 'void'); |
| List<Outline> main_children = main_outline.children; |
| expect(main_children, hasLength(2)); |
| // group1 |
| Outline group1_outline = main_children[0]; |
| _expect(group1_outline, |
| kind: ElementKind.UNIT_TEST_GROUP, |
| length: 7, |
| name: 'myGroup("group1")', |
| offset: testCode.indexOf("myGroup('group1'")); |
| List<Outline> group1_children = group1_outline.children; |
| expect(group1_children, hasLength(2)); |
| // group1_1 |
| Outline group1_1_outline = group1_children[0]; |
| _expect(group1_1_outline, |
| kind: ElementKind.UNIT_TEST_GROUP, |
| length: 7, |
| name: 'myGroup("group1_1")', |
| offset: testCode.indexOf("myGroup('group1_1'")); |
| List<Outline> group1_1_children = group1_1_outline.children; |
| expect(group1_1_children, hasLength(2)); |
| // test1_1_1 |
| Outline test1_1_1_outline = group1_1_children[0]; |
| _expect(test1_1_1_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 6, |
| name: 'myTest("test1_1_1")', |
| offset: testCode.indexOf("myTest('test1_1_1'")); |
| // test1_1_1 |
| Outline test1_1_2_outline = group1_1_children[1]; |
| _expect(test1_1_2_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 6, |
| name: 'myTest("test1_1_2")', |
| offset: testCode.indexOf("myTest('test1_1_2'")); |
| // group1_2 |
| Outline group1_2_outline = group1_children[1]; |
| _expect(group1_2_outline, |
| kind: ElementKind.UNIT_TEST_GROUP, |
| length: 7, |
| name: 'myGroup("group1_2")', |
| offset: testCode.indexOf("myGroup('group1_2'")); |
| List<Outline> group1_2_children = group1_2_outline.children; |
| expect(group1_2_children, hasLength(1)); |
| // test2_1 |
| Outline test1_2_1_outline = group1_2_children[0]; |
| _expect(test1_2_1_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 6, |
| name: 'myTest("test1_2_1")', |
| offset: testCode.indexOf("myTest('test1_2_1'")); |
| // group2 |
| Outline group2_outline = main_children[1]; |
| _expect(group2_outline, |
| kind: ElementKind.UNIT_TEST_GROUP, |
| length: 7, |
| name: 'myGroup("group2")', |
| offset: testCode.indexOf("myGroup('group2'")); |
| List<Outline> group2_children = group2_outline.children; |
| expect(group2_children, hasLength(2)); |
| // test2_1 |
| Outline test2_1_outline = group2_children[0]; |
| _expect(test2_1_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 6, |
| name: 'myTest("test2_1")', |
| offset: testCode.indexOf("myTest('test2_1'")); |
| // test2_2 |
| Outline test2_2_outline = group2_children[1]; |
| _expect(test2_2_outline, |
| kind: ElementKind.UNIT_TEST_TEST, |
| leaf: true, |
| length: 6, |
| name: 'myTest("test2_2")', |
| offset: testCode.indexOf("myTest('test2_2'")); |
| } |
| |
| test_localFunctions() async { |
| Outline unitOutline = await _computeOutline(''' |
| class A { |
| A() { |
| int local_A() {} |
| } |
| m() { |
| local_m() {} |
| } |
| } |
| f() { |
| local_f1(int i) {} |
| local_f2(String s) { |
| local_f21(int p) {} |
| } |
| } |
| '''); |
| List<Outline> topOutlines = unitOutline.children; |
| expect(topOutlines, hasLength(2)); |
| // A |
| { |
| Outline outline_A = topOutlines[0]; |
| Element element_A = outline_A.element; |
| expect(element_A.kind, ElementKind.CLASS); |
| expect(element_A.name, "A"); |
| { |
| Location location = element_A.location; |
| expect(location.offset, testCode.indexOf("A {")); |
| expect(location.length, "A".length); |
| } |
| expect(element_A.parameters, null); |
| expect(element_A.returnType, null); |
| // A children |
| List<Outline> outlines_A = outline_A.children; |
| expect(outlines_A, hasLength(2)); |
| { |
| Outline constructorOutline = outlines_A[0]; |
| Element constructorElement = constructorOutline.element; |
| expect(constructorElement.kind, ElementKind.CONSTRUCTOR); |
| expect(constructorElement.name, "A"); |
| { |
| Location location = constructorElement.location; |
| expect(location.offset, testCode.indexOf("A() {")); |
| expect(location.length, "A".length); |
| } |
| expect(constructorElement.parameters, "()"); |
| expect(constructorElement.returnType, isNull); |
| // local function |
| List<Outline> outlines_constructor = constructorOutline.children; |
| expect(outlines_constructor, hasLength(1)); |
| { |
| Outline outline = outlines_constructor[0]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FUNCTION); |
| expect(element.name, "local_A"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("local_A() {}")); |
| expect(location.length, "local_A".length); |
| } |
| expect(element.parameters, "()"); |
| expect(element.returnType, "int"); |
| } |
| } |
| { |
| Outline outline_m = outlines_A[1]; |
| Element element_m = outline_m.element; |
| expect(element_m.kind, ElementKind.METHOD); |
| expect(element_m.name, "m"); |
| { |
| Location location = element_m.location; |
| expect(location.offset, testCode.indexOf("m() {")); |
| expect(location.length, "m".length); |
| } |
| expect(element_m.parameters, "()"); |
| expect(element_m.returnType, ""); |
| // local function |
| List<Outline> methodChildren = outline_m.children; |
| expect(methodChildren, hasLength(1)); |
| { |
| Outline outline = methodChildren[0]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FUNCTION); |
| expect(element.name, "local_m"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("local_m() {}")); |
| expect(location.length, "local_m".length); |
| } |
| expect(element.parameters, "()"); |
| expect(element.returnType, ""); |
| } |
| } |
| } |
| // f() |
| { |
| Outline outline_f = topOutlines[1]; |
| Element element_f = outline_f.element; |
| expect(element_f.kind, ElementKind.FUNCTION); |
| expect(element_f.name, "f"); |
| { |
| Location location = element_f.location; |
| expect(location.offset, testCode.indexOf("f() {")); |
| expect(location.length, "f".length); |
| } |
| expect(element_f.parameters, "()"); |
| expect(element_f.returnType, ""); |
| // f() children |
| List<Outline> outlines_f = outline_f.children; |
| expect(outlines_f, hasLength(2)); |
| { |
| Outline outline_f1 = outlines_f[0]; |
| Element element_f1 = outline_f1.element; |
| expect(element_f1.kind, ElementKind.FUNCTION); |
| expect(element_f1.name, "local_f1"); |
| { |
| Location location = element_f1.location; |
| expect(location.offset, testCode.indexOf("local_f1(int i) {}")); |
| expect(location.length, "local_f1".length); |
| } |
| expect(element_f1.parameters, "(int i)"); |
| expect(element_f1.returnType, ""); |
| } |
| { |
| Outline outline_f2 = outlines_f[1]; |
| Element element_f2 = outline_f2.element; |
| expect(element_f2.kind, ElementKind.FUNCTION); |
| expect(element_f2.name, "local_f2"); |
| { |
| Location location = element_f2.location; |
| expect(location.offset, testCode.indexOf("local_f2(String s) {")); |
| expect(location.length, "local_f2".length); |
| } |
| expect(element_f2.parameters, "(String s)"); |
| expect(element_f2.returnType, ""); |
| // local_f2() local function |
| List<Outline> outlines_f2 = outline_f2.children; |
| expect(outlines_f2, hasLength(1)); |
| { |
| Outline outline_f21 = outlines_f2[0]; |
| Element element_f21 = outline_f21.element; |
| expect(element_f21.kind, ElementKind.FUNCTION); |
| expect(element_f21.name, "local_f21"); |
| { |
| Location location = element_f21.location; |
| expect(location.offset, testCode.indexOf("local_f21(int p) {")); |
| expect(location.length, "local_f21".length); |
| } |
| expect(element_f21.parameters, "(int p)"); |
| expect(element_f21.returnType, ""); |
| } |
| } |
| } |
| } |
| |
| test_mixin() async { |
| Outline unitOutline = await _computeOutline(''' |
| mixin M<N> { |
| c(int d) {} |
| String get e => null; |
| set f(int g) {} |
| } |
| '''); |
| List<Outline> topOutlines = unitOutline.children; |
| expect(topOutlines, hasLength(1)); |
| // M |
| { |
| Outline outline_M = topOutlines[0]; |
| Element element_M = outline_M.element; |
| expect(element_M.kind, ElementKind.MIXIN); |
| expect(element_M.name, "M"); |
| expect(element_M.typeParameters, "<N>"); |
| { |
| Location location = element_M.location; |
| expect(location.offset, testCode.indexOf("M<N>")); |
| expect(location.length, 1); |
| } |
| expect(element_M.parameters, isNull); |
| expect(element_M.returnType, isNull); |
| // M children |
| List<Outline> outlines_M = outline_M.children; |
| expect(outlines_M, hasLength(3)); |
| { |
| Outline outline = outlines_M[0]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.METHOD); |
| expect(element.name, "c"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("c(int d)")); |
| expect(location.length, 1); |
| } |
| expect(element.parameters, "(int d)"); |
| expect(element.returnType, ""); |
| expect(element.isAbstract, isFalse); |
| expect(element.isStatic, isFalse); |
| } |
| { |
| Outline outline = outlines_M[1]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.GETTER); |
| expect(element.name, "e"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("e => null")); |
| expect(location.length, 1); |
| } |
| expect(element.parameters, isNull); |
| expect(element.returnType, "String"); |
| } |
| { |
| Outline outline = outlines_M[2]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.SETTER); |
| expect(element.name, "f"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("f(int g)")); |
| expect(location.length, 1); |
| } |
| expect(element.parameters, "(int g)"); |
| expect(element.returnType, ""); |
| } |
| } |
| } |
| |
| test_sourceRanges_fields() async { |
| Outline unitOutline = await _computeOutline(''' |
| class A { |
| int fieldA, fieldB = 2; |
| |
| int fieldC; |
| |
| /// Documentation. |
| int fieldD; |
| } |
| '''); |
| List<Outline> outlines = unitOutline.children[0].children; |
| expect(outlines, hasLength(4)); |
| |
| // fieldA |
| { |
| Outline outline = outlines[0]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FIELD); |
| expect(element.name, "fieldA"); |
| |
| expect(outline.offset, 12); |
| expect(outline.length, 10); |
| |
| expect(outline.codeOffset, 16); |
| expect(outline.codeLength, 6); |
| } |
| |
| // fieldB |
| { |
| Outline outline = outlines[1]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FIELD); |
| expect(element.name, "fieldB"); |
| |
| expect(outline.offset, 24); |
| expect(outline.length, 11); |
| |
| expect(outline.codeOffset, 24); |
| expect(outline.codeLength, 10); |
| } |
| |
| // fieldC |
| { |
| Outline outline = outlines[2]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FIELD); |
| expect(element.name, "fieldC"); |
| |
| expect(outline.offset, 41); |
| expect(outline.length, 11); |
| |
| expect(outline.codeOffset, 45); |
| expect(outline.codeLength, 6); |
| } |
| |
| // fieldD |
| { |
| Outline outline = outlines[3]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FIELD); |
| expect(element.name, "fieldD"); |
| |
| expect(outline.offset, 58); |
| expect(outline.length, 32); |
| |
| expect(outline.codeOffset, 83); |
| expect(outline.codeLength, 6); |
| } |
| } |
| |
| test_sourceRanges_inUnit() async { |
| Outline unitOutline = await _computeOutline(''' |
| /// My first class. |
| class A {} |
| |
| class B {} |
| '''); |
| List<Outline> topOutlines = unitOutline.children; |
| expect(topOutlines, hasLength(2)); |
| |
| // A |
| { |
| Outline outline = topOutlines[0]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.CLASS); |
| expect(element.name, "A"); |
| |
| expect(outline.offset, 0); |
| expect(outline.length, 30); |
| |
| expect(outline.codeOffset, 20); |
| expect(outline.codeLength, 10); |
| } |
| |
| // B |
| { |
| Outline outline = topOutlines[1]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.CLASS); |
| expect(element.name, "B"); |
| |
| expect(outline.offset, 32); |
| expect(outline.length, 10); |
| |
| expect(outline.codeOffset, 32); |
| expect(outline.codeLength, 10); |
| } |
| } |
| |
| test_sourceRanges_method() async { |
| Outline unitOutline = await _computeOutline(''' |
| class A { |
| int methodA() {} |
| |
| /// Documentation. |
| @override |
| int methodB() {} |
| } |
| '''); |
| List<Outline> outlines = unitOutline.children[0].children; |
| expect(outlines, hasLength(2)); |
| |
| // methodA |
| { |
| Outline outline = outlines[0]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.METHOD); |
| expect(element.name, "methodA"); |
| |
| expect(outline.offset, 12); |
| expect(outline.length, 16); |
| |
| expect(outline.codeOffset, 12); |
| expect(outline.codeLength, 16); |
| } |
| |
| // methodB |
| { |
| Outline outline = outlines[1]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.METHOD); |
| expect(element.name, "methodB"); |
| |
| expect(outline.offset, 34); |
| expect(outline.length, 49); |
| |
| expect(outline.codeOffset, 67); |
| expect(outline.codeLength, 16); |
| } |
| } |
| |
| test_topLevel() async { |
| Outline unitOutline = await _computeOutline(''' |
| typedef String FTA<K, V>(int i, String s); |
| typedef FTB(int p); |
| class A<T> {} |
| class B {} |
| class CTA<T> = A<T> with B; |
| class CTB = A with B; |
| String fA(int i, String s) => null; |
| fB(int p) => null; |
| String get propA => null; |
| set propB(int v) {} |
| '''); |
| List<Outline> topOutlines = unitOutline.children; |
| expect(topOutlines, hasLength(10)); |
| // FTA |
| { |
| Outline outline = topOutlines[0]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FUNCTION_TYPE_ALIAS); |
| expect(element.name, "FTA"); |
| expect(element.typeParameters, "<K, V>"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("FTA<K, V>(")); |
| expect(location.length, "FTA".length); |
| } |
| expect(element.parameters, "(int i, String s)"); |
| expect(element.returnType, "String"); |
| } |
| // FTB |
| { |
| Outline outline = topOutlines[1]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FUNCTION_TYPE_ALIAS); |
| expect(element.name, "FTB"); |
| expect(element.typeParameters, isNull); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("FTB(")); |
| expect(location.length, "FTB".length); |
| } |
| expect(element.parameters, "(int p)"); |
| expect(element.returnType, ""); |
| } |
| // CTA |
| { |
| Outline outline = topOutlines[4]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.CLASS_TYPE_ALIAS); |
| expect(element.name, "CTA"); |
| expect(element.typeParameters, '<T>'); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("CTA<T> =")); |
| expect(location.length, "CTA".length); |
| } |
| expect(element.parameters, isNull); |
| expect(element.returnType, isNull); |
| } |
| // CTB |
| { |
| Outline outline = topOutlines[5]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.CLASS_TYPE_ALIAS); |
| expect(element.name, 'CTB'); |
| expect(element.typeParameters, isNull); |
| expect(element.returnType, isNull); |
| } |
| // fA |
| { |
| Outline outline = topOutlines[6]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FUNCTION); |
| expect(element.name, "fA"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("fA(")); |
| expect(location.length, "fA".length); |
| } |
| expect(element.parameters, "(int i, String s)"); |
| expect(element.returnType, "String"); |
| } |
| // fB |
| { |
| Outline outline = topOutlines[7]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.FUNCTION); |
| expect(element.name, "fB"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("fB(")); |
| expect(location.length, "fB".length); |
| } |
| expect(element.parameters, "(int p)"); |
| expect(element.returnType, ""); |
| } |
| // propA |
| { |
| Outline outline = topOutlines[8]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.GETTER); |
| expect(element.name, "propA"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("propA => null;")); |
| expect(location.length, "propA".length); |
| } |
| expect(element.parameters, ""); |
| expect(element.returnType, "String"); |
| } |
| // propB |
| { |
| Outline outline = topOutlines[9]; |
| Element element = outline.element; |
| expect(element.kind, ElementKind.SETTER); |
| expect(element.name, "propB"); |
| { |
| Location location = element.location; |
| expect(location.offset, testCode.indexOf("propB(int v) {}")); |
| expect(location.length, "propB".length); |
| } |
| expect(element.parameters, "(int v)"); |
| expect(element.returnType, ""); |
| } |
| } |
| |
| void _expect(Outline outline, |
| {ElementKind kind, |
| bool leaf: false, |
| int length, |
| String name, |
| int offset, |
| String parameters, |
| String returnType}) { |
| Element element = outline.element; |
| Location location = element.location; |
| |
| if (kind != null) { |
| expect(element.kind, kind); |
| } |
| if (leaf) { |
| expect(outline.children, isNull); |
| } |
| length ??= name?.length; |
| if (length != null) { |
| expect(location.length, length); |
| } |
| if (name != null) { |
| expect(element.name, name); |
| } |
| if (offset != null) { |
| expect(location.offset, offset); |
| } |
| if (parameters != null) { |
| expect(element.parameters, parameters); |
| } |
| if (returnType != null) { |
| expect(element.returnType, returnType); |
| } |
| } |
| |
| void _isEnumConstant(Outline outline, String name) { |
| Element element = outline.element; |
| expect(element.kind, ElementKind.ENUM_CONSTANT); |
| expect(element.name, name); |
| expect(element.parameters, isNull); |
| expect(element.returnType, isNull); |
| } |
| } |