blob: ce9c9aa4538ecef2d6822055c45363f1f3971102 [file] [log] [blame]
// Copyright (c) 2016, 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/services/search/search_engine.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/src/dart/analysis/driver.dart';
import 'package:analyzer/src/dart/analysis/search.dart';
import 'package:analyzer/src/generated/source.dart' show Source, SourceRange;
/**
* A [SearchEngine] implementation.
*/
class SearchEngineImpl implements SearchEngine {
final Iterable<AnalysisDriver> _drivers;
SearchEngineImpl(this._drivers);
@override
Future<Set<String>> membersOfSubtypes(ClassElement type) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
List<AnalysisDriver> drivers = _drivers.toList();
SearchedFiles searchedFiles = _createSearchedFiles(drivers);
String libraryUriStr = type.librarySource.uri.toString();
bool hasSubtypes = false;
Set<String> visitedIds = new Set<String>();
Set<String> members = new Set<String>();
Future<void> addMembers(ClassElement type, SubtypeResult subtype) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
if (subtype != null && !visitedIds.add(subtype.id)) {
return;
}
for (AnalysisDriver driver in drivers) {
List<SubtypeResult> subtypes = await driver.search
.subtypes(searchedFiles, type: type, subtype: subtype);
for (SubtypeResult subtype in subtypes) {
hasSubtypes = true;
members.addAll(subtype.libraryUri == libraryUriStr
? subtype.members
: subtype.members.where((name) => !name.startsWith('_')));
await addMembers(null, subtype);
}
}
}
await addMembers(type, null);
if (!hasSubtypes) {
return null;
}
return members;
}
@override
Future<Set<ClassElement>> searchAllSubtypes(ClassElement type) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
Set<ClassElement> allSubtypes = new Set<ClassElement>();
Future<void> addSubtypes(ClassElement type) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
List<SearchResult> directResults = await _searchDirectSubtypes(type);
for (SearchResult directResult in directResults) {
var directSubtype = directResult.enclosingElement as ClassElement;
if (allSubtypes.add(directSubtype)) {
await addSubtypes(directSubtype);
}
}
}
await addSubtypes(type);
return allSubtypes;
}
@override
Future<List<SearchMatch>> searchMemberDeclarations(String name) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
List<SearchMatch> allDeclarations = [];
List<AnalysisDriver> drivers = _drivers.toList();
for (AnalysisDriver driver in drivers) {
List<Element> elements = await driver.search.classMembers(name);
allDeclarations.addAll(elements.map(SearchMatchImpl.forElement));
}
return allDeclarations;
}
@override
Future<List<SearchMatch>> searchMemberReferences(String name) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
List<SearchResult> allResults = [];
List<AnalysisDriver> drivers = _drivers.toList();
SearchedFiles searchedFiles = _createSearchedFiles(drivers);
for (AnalysisDriver driver in drivers) {
List<SearchResult> results =
await driver.search.unresolvedMemberReferences(name, searchedFiles);
allResults.addAll(results);
}
return allResults.map(SearchMatchImpl.forSearchResult).toList();
}
@override
Future<List<SearchMatch>> searchReferences(Element element) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
List<SearchResult> allResults = [];
List<AnalysisDriver> drivers = _drivers.toList();
SearchedFiles searchedFiles = _createSearchedFiles(drivers);
for (AnalysisDriver driver in drivers) {
List<SearchResult> results =
await driver.search.references(element, searchedFiles);
allResults.addAll(results);
}
return allResults.map(SearchMatchImpl.forSearchResult).toList();
}
@override
Future<List<SearchMatch>> searchSubtypes(ClassElement type) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
List<SearchResult> results = await _searchDirectSubtypes(type);
return results.map(SearchMatchImpl.forSearchResult).toList();
}
@override
Future<List<SearchMatch>> searchTopLevelDeclarations(String pattern) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
Set<Element> allElements = new Set<Element>();
RegExp regExp = new RegExp(pattern);
List<AnalysisDriver> drivers = _drivers.toList();
for (AnalysisDriver driver in drivers) {
List<Element> elements = await driver.search.topLevelElements(regExp);
allElements.addAll(elements);
}
return allElements.map(SearchMatchImpl.forElement).toList();
}
/**
* Create a new [SearchedFiles] instance in which added files are owned
* by the drivers that have them added.
*/
SearchedFiles _createSearchedFiles(List<AnalysisDriver> drivers) {
var searchedFiles = new SearchedFiles();
for (AnalysisDriver driver in drivers) {
searchedFiles.ownAdded(driver.search);
}
return searchedFiles;
}
Future<List<SearchResult>> _searchDirectSubtypes(ClassElement type) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
List<SearchResult> allResults = [];
List<AnalysisDriver> drivers = _drivers.toList();
SearchedFiles searchedFiles = _createSearchedFiles(drivers);
for (AnalysisDriver driver in drivers) {
List<SearchResult> results =
await driver.search.subTypes(type, searchedFiles);
allResults.addAll(results);
}
return allResults;
}
}
class SearchMatchImpl implements SearchMatch {
@override
final String file;
@override
final Source librarySource;
@override
final Source unitSource;
@override
final LibraryElement libraryElement;
@override
final Element element;
@override
final bool isResolved;
@override
final bool isQualified;
@override
final MatchKind kind;
@override
final SourceRange sourceRange;
SearchMatchImpl(
this.file,
this.librarySource,
this.unitSource,
this.libraryElement,
this.element,
this.isResolved,
this.isQualified,
this.kind,
this.sourceRange);
@override
String toString() {
StringBuffer buffer = new StringBuffer();
buffer.write("SearchMatch(kind=");
buffer.write(kind);
buffer.write(", libraryUri=");
buffer.write(librarySource.uri);
buffer.write(", unitUri=");
buffer.write(unitSource.uri);
buffer.write(", range=");
buffer.write(sourceRange);
buffer.write(", isResolved=");
buffer.write(isResolved);
buffer.write(", isQualified=");
buffer.write(isQualified);
buffer.write(")");
return buffer.toString();
}
static SearchMatchImpl forElement(Element element) {
return new SearchMatchImpl(
element.source.fullName,
element.librarySource,
element.source,
element.library,
element,
true,
true,
MatchKind.DECLARATION,
new SourceRange(element.nameOffset, element.nameLength));
}
static SearchMatchImpl forSearchResult(SearchResult result) {
Element enclosingElement = result.enclosingElement;
return new SearchMatchImpl(
enclosingElement.source.fullName,
enclosingElement.librarySource,
enclosingElement.source,
enclosingElement.library,
enclosingElement,
result.isResolved,
result.isQualified,
toMatchKind(result.kind),
new SourceRange(result.offset, result.length));
}
static MatchKind toMatchKind(SearchResultKind kind) {
if (kind == SearchResultKind.READ) {
return MatchKind.READ;
}
if (kind == SearchResultKind.READ_WRITE) {
return MatchKind.READ_WRITE;
}
if (kind == SearchResultKind.WRITE) {
return MatchKind.WRITE;
}
if (kind == SearchResultKind.INVOCATION) {
return MatchKind.INVOCATION;
}
return MatchKind.REFERENCE;
}
}