blob: eab4b0dd1d1869eaea01a76d643feba9d1840b63 [file] [log] [blame]
// Copyright (c) 2019, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'dart:io';
import 'package:_fe_analyzer_shared/src/testing/id.dart' show ActualData, Id;
import 'package:_fe_analyzer_shared/src/testing/id_testing.dart';
import 'package:_fe_analyzer_shared/src/type_inference/assigned_variables.dart';
import 'package:analyzer/dart/ast/ast.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/src/dart/analysis/testing_data.dart';
import 'package:analyzer/src/dart/resolver/flow_analysis_visitor.dart';
import 'package:analyzer/src/util/ast_data_extractor.dart';
import '../util/id_testing_helper.dart';
main(List<String> args) {
Directory dataDir = Directory.fromUri(
Platform.script.resolve(
'../../../_fe_analyzer_shared/test/flow_analysis/assigned_variables/'
'data',
),
);
return runTests<_Data>(
dataDir,
args: args,
createUriForFileName: createUriForFileName,
onFailure: onFailure,
runTest: runTestFor(const _AssignedVariablesDataComputer(), [
analyzerDefaultConfig,
]),
);
}
class _AssignedVariablesDataComputer extends DataComputer<_Data> {
const _AssignedVariablesDataComputer();
@override
DataInterpreter<_Data> get dataValidator =>
const _AssignedVariablesDataInterpreter();
@override
void computeUnitData(
TestingData testingData,
CompilationUnit unit,
Map<Id, ActualData<_Data>> actualMap,
) {
var unitElement = unit.declaredFragment!.element;
var uri = unitElement.firstFragment.source.uri;
var flowResult = testingData.uriToFlowAnalysisData[uri]!;
_AssignedVariablesDataExtractor(uri, actualMap, flowResult).run(unit);
}
}
class _AssignedVariablesDataExtractor extends AstDataExtractor<_Data> {
final FlowAnalysisDataForTesting _flowResult;
Declaration? _currentDeclaration;
AssignedVariablesForTesting<AstNode, PromotableElement2>?
_currentAssignedVariables;
_AssignedVariablesDataExtractor(super.uri, super.actualMap, this._flowResult);
@override
_Data? computeNodeValue(Id id, AstNode node) {
if (node is FunctionDeclarationStatement) {
node = node.functionDeclaration;
}
var currentAssignedVariables = _currentAssignedVariables;
if (currentAssignedVariables == null) return null;
if (node == _currentDeclaration) {
return _Data(
_convertVars(currentAssignedVariables.declaredAtTopLevel),
_convertVars(currentAssignedVariables.readAnywhere),
_convertVars(currentAssignedVariables.readCapturedAnywhere),
_convertVars(currentAssignedVariables.writtenAnywhere),
_convertVars(currentAssignedVariables.capturedAnywhere),
);
}
if (!currentAssignedVariables.isTracked(node)) return null;
return _Data(
_convertVars(currentAssignedVariables.declaredInNode(node)),
_convertVars(currentAssignedVariables.readInNode(node)),
_convertVars(currentAssignedVariables.readCapturedInNode(node)),
_convertVars(currentAssignedVariables.writtenInNode(node)),
_convertVars(currentAssignedVariables.capturedInNode(node)),
);
}
@override
visitConstructorDeclaration(ConstructorDeclaration node) {
_handlePossibleTopLevelDeclaration(
node,
() => super.visitConstructorDeclaration(node),
);
}
@override
void visitFunctionDeclaration(FunctionDeclaration node) {
_handlePossibleTopLevelDeclaration(
node,
() => super.visitFunctionDeclaration(node),
);
}
@override
void visitVariableDeclaration(VariableDeclaration node) {
_handlePossibleTopLevelDeclaration(
node,
() => super.visitVariableDeclaration(node),
);
}
Set<String> _convertVars(Iterable<int> x) =>
x.map((e) => _currentAssignedVariables!.variableForKey(e).name3!).toSet();
void _handlePossibleTopLevelDeclaration(
AstNode node,
void Function() callback,
) {
if (_currentDeclaration == null) {
_currentDeclaration = node as Declaration;
_currentAssignedVariables = _flowResult.assignedVariables[node];
callback();
_currentDeclaration = null;
_currentAssignedVariables = null;
} else {
callback();
}
}
}
class _AssignedVariablesDataInterpreter implements DataInterpreter<_Data> {
const _AssignedVariablesDataInterpreter();
@override
String getText(_Data actualData, [String? indentation]) {
var parts = <String>[];
if (actualData.declared.isNotEmpty) {
parts.add('declared=${_setToString(actualData.declared)}');
}
if (actualData.read.isNotEmpty) {
parts.add('read=${_setToString(actualData.read)}');
}
if (actualData.readCaptured.isNotEmpty) {
parts.add('read=${_setToString(actualData.readCaptured)}');
}
if (actualData.assigned.isNotEmpty) {
parts.add('assigned=${_setToString(actualData.assigned)}');
}
if (actualData.captured.isNotEmpty) {
parts.add('captured=${_setToString(actualData.captured)}');
}
if (parts.isEmpty) return 'none';
return parts.join(', ');
}
@override
String? isAsExpected(_Data actualData, String? expectedData) {
var actualDataText = getText(actualData);
if (actualDataText == expectedData) {
return null;
} else {
return 'Expected "$expectedData", got "$actualDataText"';
}
}
@override
bool isEmpty(_Data actualData) =>
actualData.assigned.isEmpty && actualData.captured.isEmpty;
String _setToString(Set<String> values) {
List<String> sortedValues = values.toList()..sort();
return '{${sortedValues.join(', ')}}';
}
}
class _Data {
final Set<String> declared;
final Set<String> read;
final Set<String> readCaptured;
final Set<String> assigned;
final Set<String> captured;
_Data(
this.declared,
this.read,
this.readCaptured,
this.assigned,
this.captured,
);
}