| // Copyright (c) 2014, 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 'dart:core'; |
| |
| import 'package:analysis_server/protocol/protocol_constants.dart'; |
| import 'package:analysis_server/src/analysis_server.dart'; |
| import 'package:analysis_server/src/computer/computer_hover.dart'; |
| import 'package:analysis_server/src/computer/computer_signature.dart'; |
| import 'package:analysis_server/src/computer/imported_elements_computer.dart'; |
| import 'package:analysis_server/src/domain_abstract.dart'; |
| import 'package:analysis_server/src/domains/analysis/navigation_dart.dart'; |
| import 'package:analysis_server/src/plugin/plugin_manager.dart'; |
| import 'package:analysis_server/src/plugin/request_converter.dart'; |
| import 'package:analysis_server/src/plugin/result_merger.dart'; |
| import 'package:analysis_server/src/protocol/protocol_internal.dart'; |
| import 'package:analysis_server/src/protocol_server.dart'; |
| import 'package:analyzer/dart/ast/ast.dart'; |
| import 'package:analyzer/error/error.dart' as engine; |
| import 'package:analyzer/file_system/file_system.dart'; |
| import 'package:analyzer/src/dart/analysis/driver.dart'; |
| import 'package:analyzer/src/generated/engine.dart' as engine; |
| import 'package:analyzer/src/generated/source.dart'; |
| import 'package:analyzer_plugin/protocol/protocol.dart' as plugin; |
| import 'package:analyzer_plugin/protocol/protocol_common.dart'; |
| import 'package:analyzer_plugin/protocol/protocol_constants.dart' as plugin; |
| import 'package:analyzer_plugin/protocol/protocol_generated.dart' as plugin; |
| import 'package:analyzer_plugin/src/utilities/navigation/navigation.dart'; |
| import 'package:path/path.dart'; |
| |
| // TODO(devoncarew): See #31456 for the tracking issue to remove this flag. |
| final bool disableManageImportsOnPaste = true; |
| |
| /** |
| * Instances of the class [AnalysisDomainHandler] implement a [RequestHandler] |
| * that handles requests in the `analysis` domain. |
| */ |
| class AnalysisDomainHandler extends AbstractRequestHandler { |
| /** |
| * Initialize a newly created handler to handle requests for the given [server]. |
| */ |
| AnalysisDomainHandler(AnalysisServer server) : super(server); |
| |
| /** |
| * Implement the `analysis.getErrors` request. |
| */ |
| Future<void> getErrors(Request request) async { |
| // TODO(brianwilkerson) Determine whether this await is necessary. |
| await null; |
| String file = new AnalysisGetErrorsParams.fromRequest(request).file; |
| |
| void send(engine.AnalysisOptions analysisOptions, LineInfo lineInfo, |
| List<engine.AnalysisError> errors) { |
| if (lineInfo == null) { |
| server.sendResponse(new Response.getErrorsInvalidFile(request)); |
| } else { |
| List<AnalysisError> protocolErrors = |
| doAnalysisError_listFromEngine(analysisOptions, lineInfo, errors); |
| server.sendResponse( |
| new AnalysisGetErrorsResult(protocolErrors).toResponse(request.id)); |
| } |
| } |
| |
| AnalysisResult result = await server.getAnalysisResult(file); |
| send(result?.driver?.analysisOptions, result?.lineInfo, result?.errors); |
| } |
| |
| /** |
| * Implement the `analysis.getHover` request. |
| */ |
| Future<void> getHover(Request request) async { |
| // TODO(brianwilkerson) Determine whether this await is necessary. |
| await null; |
| var params = new AnalysisGetHoverParams.fromRequest(request); |
| |
| // Prepare the resolved units. |
| AnalysisResult result = await server.getAnalysisResult(params.file); |
| CompilationUnit unit = result?.unit; |
| |
| // Prepare the hovers. |
| List<HoverInformation> hovers = <HoverInformation>[]; |
| if (unit != null) { |
| HoverInformation hoverInformation = |
| new DartUnitHoverComputer(unit, params.offset).compute(); |
| if (hoverInformation != null) { |
| hovers.add(hoverInformation); |
| } |
| } |
| |
| // Send the response. |
| server.sendResponse( |
| new AnalysisGetHoverResult(hovers).toResponse(request.id)); |
| } |
| |
| /** |
| * Implement the `analysis.getImportedElements` request. |
| */ |
| Future<void> getImportedElements(Request request) async { |
| // TODO(brianwilkerson) Determine whether this await is necessary. |
| await null; |
| AnalysisGetImportedElementsParams params = |
| new AnalysisGetImportedElementsParams.fromRequest(request); |
| // |
| // Prepare the resolved unit. |
| // |
| AnalysisResult result = await server.getAnalysisResult(params.file); |
| if (result == null) { |
| server.sendResponse(new Response.getImportedElementsInvalidFile(request)); |
| } |
| |
| List<ImportedElements> elements; |
| |
| // |
| // Compute the list of imported elements. |
| // |
| if (disableManageImportsOnPaste) { |
| elements = <ImportedElements>[]; |
| } else { |
| elements = new ImportedElementsComputer( |
| result.unit, params.offset, params.length) |
| .compute(); |
| } |
| |
| // |
| // Send the response. |
| // |
| server.sendResponse( |
| new AnalysisGetImportedElementsResult(elements).toResponse(request.id)); |
| } |
| |
| /** |
| * Implement the `analysis.getLibraryDependencies` request. |
| */ |
| Response getLibraryDependencies(Request request) { |
| return new Response.unsupportedFeature(request.id, |
| 'Please contact the Dart analyzer team if you need this request.'); |
| // server.onAnalysisComplete.then((_) { |
| // LibraryDependencyCollector collector = |
| // new LibraryDependencyCollector(server.analysisContexts); |
| // Set<String> libraries = collector.collectLibraryDependencies(); |
| // Map<String, Map<String, List<String>>> packageMap = |
| // collector.calculatePackageMap(server.folderMap); |
| // server.sendResponse(new AnalysisGetLibraryDependenciesResult( |
| // libraries.toList(growable: false), packageMap) |
| // .toResponse(request.id)); |
| // }).catchError((error, st) { |
| // server.sendResponse(new Response.serverError(request, error, st)); |
| // }); |
| // // delay response |
| // return Response.DELAYED_RESPONSE; |
| } |
| |
| /** |
| * Implement the `analysis.getNavigation` request. |
| */ |
| Future<void> getNavigation(Request request) async { |
| // TODO(brianwilkerson) Determine whether this await is necessary. |
| await null; |
| var params = new AnalysisGetNavigationParams.fromRequest(request); |
| String file = params.file; |
| int offset = params.offset; |
| int length = params.length; |
| |
| AnalysisDriver driver = server.getAnalysisDriver(file); |
| if (driver == null) { |
| server.sendResponse(new Response.getNavigationInvalidFile(request)); |
| } else { |
| // |
| // Allow plugins to start computing navigation data. |
| // |
| plugin.AnalysisGetNavigationParams requestParams = |
| new plugin.AnalysisGetNavigationParams(file, offset, length); |
| Map<PluginInfo, Future<plugin.Response>> pluginFutures = server |
| .pluginManager |
| .broadcastRequest(requestParams, contextRoot: driver.contextRoot); |
| // |
| // Compute navigation data generated by server. |
| // |
| List<AnalysisNavigationParams> allResults = <AnalysisNavigationParams>[]; |
| AnalysisResult result = await server.getAnalysisResult(file); |
| CompilationUnit unit = result?.unit; |
| if (unit != null && result.exists) { |
| NavigationCollectorImpl collector = new NavigationCollectorImpl(); |
| computeDartNavigation(collector, unit, offset, length); |
| collector.createRegions(); |
| allResults.add(new AnalysisNavigationParams( |
| file, collector.regions, collector.targets, collector.files)); |
| } |
| // |
| // Add the navigation data produced by plugins to the server-generated |
| // navigation data. |
| // |
| if (pluginFutures != null) { |
| List<plugin.Response> responses = await waitForResponses(pluginFutures, |
| requestParameters: requestParams); |
| for (plugin.Response response in responses) { |
| plugin.AnalysisGetNavigationResult result = |
| new plugin.AnalysisGetNavigationResult.fromResponse(response); |
| allResults.add(new AnalysisNavigationParams( |
| file, result.regions, result.targets, result.files)); |
| } |
| } |
| // |
| // Return the result. |
| // |
| ResultMerger merger = new ResultMerger(); |
| AnalysisNavigationParams mergedResults = |
| merger.mergeNavigation(allResults); |
| if (mergedResults == null) { |
| server.sendResponse(new AnalysisGetNavigationResult( |
| <String>[], <NavigationTarget>[], <NavigationRegion>[]) |
| .toResponse(request.id)); |
| } else { |
| server.sendResponse(new AnalysisGetNavigationResult(mergedResults.files, |
| mergedResults.targets, mergedResults.regions) |
| .toResponse(request.id)); |
| } |
| } |
| } |
| |
| /** |
| * Implement the `analysis.getReachableSources` request. |
| */ |
| Response getReachableSources(Request request) { |
| return new Response.unsupportedFeature(request.id, |
| 'Please contact the Dart analyzer team if you need this request.'); |
| // AnalysisGetReachableSourcesParams params = |
| // new AnalysisGetReachableSourcesParams.fromRequest(request); |
| // ContextSourcePair pair = server.getContextSourcePair(params.file); |
| // if (pair.context == null || pair.source == null) { |
| // return new Response.getReachableSourcesInvalidFile(request); |
| // } |
| // Map<String, List<String>> sources = |
| // new ReachableSourceCollector(pair.source, pair.context) |
| // .collectSources(); |
| // return new AnalysisGetReachableSourcesResult(sources) |
| // .toResponse(request.id); |
| } |
| |
| /** |
| * Implement the `analysis.getSignature` request. |
| */ |
| Future<void> getSignature(Request request) async { |
| var params = new AnalysisGetSignatureParams.fromRequest(request); |
| |
| // Prepare the resolved units. |
| AnalysisResult result = await server.getAnalysisResult(params.file); |
| CompilationUnit unit = result?.unit; |
| if (unit == null) { |
| server.sendResponse(new Response.getSignatureInvalidFile(request)); |
| return; |
| } |
| |
| // Ensure the offset provided is a valid location in the file. |
| final computer = new DartUnitSignatureComputer(unit, params.offset); |
| if (!computer.offsetIsValid) { |
| server.sendResponse(new Response.getSignatureInvalidOffset(request)); |
| return; |
| } |
| |
| // Try to get a signature. |
| final signature = computer.compute(); |
| if (signature == null) { |
| server.sendResponse(new Response.getSignatureUnknownFunction(request)); |
| return; |
| } |
| |
| server.sendResponse(signature.toResponse(request.id)); |
| } |
| |
| @override |
| Response handleRequest(Request request) { |
| try { |
| String requestName = request.method; |
| if (requestName == ANALYSIS_REQUEST_GET_ERRORS) { |
| getErrors(request); |
| return Response.DELAYED_RESPONSE; |
| } else if (requestName == ANALYSIS_REQUEST_GET_HOVER) { |
| getHover(request); |
| return Response.DELAYED_RESPONSE; |
| } else if (requestName == ANALYSIS_REQUEST_GET_IMPORTED_ELEMENTS) { |
| getImportedElements(request); |
| return Response.DELAYED_RESPONSE; |
| } else if (requestName == ANALYSIS_REQUEST_GET_LIBRARY_DEPENDENCIES) { |
| return getLibraryDependencies(request); |
| } else if (requestName == ANALYSIS_REQUEST_GET_NAVIGATION) { |
| getNavigation(request); |
| return Response.DELAYED_RESPONSE; |
| } else if (requestName == ANALYSIS_REQUEST_GET_REACHABLE_SOURCES) { |
| return getReachableSources(request); |
| } else if (requestName == ANALYSIS_REQUEST_GET_SIGNATURE) { |
| getSignature(request); |
| return Response.DELAYED_RESPONSE; |
| } else if (requestName == ANALYSIS_REQUEST_REANALYZE) { |
| return reanalyze(request); |
| } else if (requestName == ANALYSIS_REQUEST_SET_ANALYSIS_ROOTS) { |
| return setAnalysisRoots(request); |
| } else if (requestName == ANALYSIS_REQUEST_SET_GENERAL_SUBSCRIPTIONS) { |
| return setGeneralSubscriptions(request); |
| } else if (requestName == ANALYSIS_REQUEST_SET_PRIORITY_FILES) { |
| return setPriorityFiles(request); |
| } else if (requestName == ANALYSIS_REQUEST_SET_SUBSCRIPTIONS) { |
| return setSubscriptions(request); |
| } else if (requestName == ANALYSIS_REQUEST_UPDATE_CONTENT) { |
| return updateContent(request); |
| } else if (requestName == ANALYSIS_REQUEST_UPDATE_OPTIONS) { |
| return updateOptions(request); |
| } |
| } on RequestFailure catch (exception) { |
| return exception.response; |
| } |
| return null; |
| } |
| |
| /** |
| * Implement the 'analysis.reanalyze' request. |
| */ |
| Response reanalyze(Request request) { |
| server.options.analytics?.sendEvent('analysis', 'reanalyze'); |
| |
| AnalysisReanalyzeParams params = |
| new AnalysisReanalyzeParams.fromRequest(request); |
| List<String> roots = params.roots; |
| if (roots == null || roots.isNotEmpty) { |
| List<String> includedPaths = server.contextManager.includedPaths; |
| List<Resource> rootResources = null; |
| if (roots != null) { |
| rootResources = <Resource>[]; |
| for (String rootPath in roots) { |
| if (!includedPaths.contains(rootPath)) { |
| return new Response.invalidAnalysisRoot(request, rootPath); |
| } |
| rootResources.add(server.resourceProvider.getResource(rootPath)); |
| } |
| } |
| server.reanalyze(rootResources); |
| } |
| // |
| // Restart all of the plugins. This is an async operation that will happen |
| // in the background. |
| // |
| server.pluginManager.restartPlugins(); |
| // |
| // Send the response. |
| // |
| return new AnalysisReanalyzeResult().toResponse(request.id); |
| } |
| |
| /** |
| * Implement the 'analysis.setAnalysisRoots' request. |
| */ |
| Response setAnalysisRoots(Request request) { |
| if (server.options.enableUXExperiment1) { |
| return new AnalysisSetAnalysisRootsResult().toResponse(request.id); |
| } |
| |
| var params = new AnalysisSetAnalysisRootsParams.fromRequest(request); |
| List<String> includedPathList = params.included; |
| List<String> excludedPathList = params.excluded; |
| |
| server.options.analytics?.sendEvent('analysis', 'setAnalysisRoots', |
| value: includedPathList.length); |
| |
| // validate |
| for (String path in includedPathList) { |
| if (!server.isValidFilePath(path)) { |
| return new Response.invalidFilePathFormat(request, path); |
| } |
| } |
| for (String path in excludedPathList) { |
| if (!server.isValidFilePath(path)) { |
| return new Response.invalidFilePathFormat(request, path); |
| } |
| } |
| Map<String, String> packageRoots = |
| params.packageRoots ?? <String, String>{}; |
| |
| if (server.options.enableUXExperiment2 && |
| server.detachableFileSystemManager != null) { |
| server.detachableFileSystemManager.setAnalysisRoots( |
| request.id, includedPathList, excludedPathList, packageRoots); |
| } else { |
| server.setAnalysisRoots( |
| request.id, includedPathList, excludedPathList, packageRoots); |
| } |
| return new AnalysisSetAnalysisRootsResult().toResponse(request.id); |
| } |
| |
| /** |
| * Implement the 'analysis.setGeneralSubscriptions' request. |
| */ |
| Response setGeneralSubscriptions(Request request) { |
| AnalysisSetGeneralSubscriptionsParams params = |
| new AnalysisSetGeneralSubscriptionsParams.fromRequest(request); |
| server.setGeneralAnalysisSubscriptions(params.subscriptions); |
| return new AnalysisSetGeneralSubscriptionsResult().toResponse(request.id); |
| } |
| |
| /** |
| * Implement the 'analysis.setPriorityFiles' request. |
| */ |
| Response setPriorityFiles(Request request) { |
| var params = new AnalysisSetPriorityFilesParams.fromRequest(request); |
| |
| if (server.options.enableUXExperiment1) { |
| // If this experiment is enabled, set the analysis root to be the |
| // containing directory. |
| |
| List<String> includedPathList = new List<String>(); |
| |
| // Reference the priority files, remove files that don't end in dart, yaml |
| // or html suffixes and sort from shortest to longest file paths. |
| List<String> priorityFiles = params.files; |
| priorityFiles.removeWhere((s) => |
| !s.endsWith('.dart') && !s.endsWith('.yaml') && !s.endsWith('.html')); |
| |
| Context pathContext = server.resourceProvider.pathContext; |
| List<String> containingDirectories = <String>[]; |
| for (String filePath in priorityFiles) { |
| containingDirectories.add(pathContext.dirname(filePath)); |
| } |
| containingDirectories.sort(); |
| |
| // For each file, add the contained directory to includedPathList iff |
| // some other parent containing directory has not already been added. |
| for (String containedDir in containingDirectories) { |
| // Check that no parent directories have already been added (we have |
| // guarantees here as the list was sorted above.) |
| bool parentDirectoryInListAlready = false; |
| for (int i = 0; i < includedPathList.length; i++) { |
| if (containedDir.startsWith(includedPathList[i])) { |
| parentDirectoryInListAlready = true; |
| } |
| } |
| if (!parentDirectoryInListAlready) { |
| includedPathList.add(containedDir); |
| } |
| } |
| |
| server.setAnalysisRoots( |
| request.id, includedPathList, <String>[], <String, String>{}); |
| } |
| |
| server.setPriorityFiles(request.id, params.files); |
| // |
| // Forward the request to the plugins. |
| // |
| RequestConverter converter = new RequestConverter(); |
| server.pluginManager.setAnalysisSetPriorityFilesParams( |
| converter.convertAnalysisSetPriorityFilesParams(params)); |
| // |
| // Send the response. |
| // |
| return new AnalysisSetPriorityFilesResult().toResponse(request.id); |
| } |
| |
| /** |
| * Implement the 'analysis.setSubscriptions' request. |
| */ |
| Response setSubscriptions(Request request) { |
| var params = new AnalysisSetSubscriptionsParams.fromRequest(request); |
| // parse subscriptions |
| Map<AnalysisService, Set<String>> subMap = mapMap(params.subscriptions, |
| valueCallback: (List<String> subscriptions) => subscriptions.toSet()); |
| server.setAnalysisSubscriptions(subMap); |
| // |
| // Forward the request to the plugins. |
| // |
| RequestConverter converter = new RequestConverter(); |
| server.pluginManager.setAnalysisSetSubscriptionsParams( |
| converter.convertAnalysisSetSubscriptionsParams(params)); |
| // |
| // Send the response. |
| // |
| return new AnalysisSetSubscriptionsResult().toResponse(request.id); |
| } |
| |
| /** |
| * Implement the 'analysis.updateContent' request. |
| */ |
| Response updateContent(Request request) { |
| var params = new AnalysisUpdateContentParams.fromRequest(request); |
| server.updateContent(request.id, params.files); |
| // |
| // Forward the request to the plugins. |
| // |
| RequestConverter converter = new RequestConverter(); |
| server.pluginManager.setAnalysisUpdateContentParams( |
| converter.convertAnalysisUpdateContentParams(params)); |
| // |
| // Send the response. |
| // |
| return new AnalysisUpdateContentResult().toResponse(request.id); |
| } |
| |
| /** |
| * Implement the 'analysis.updateOptions' request. |
| */ |
| Response updateOptions(Request request) { |
| // options |
| var params = new AnalysisUpdateOptionsParams.fromRequest(request); |
| AnalysisOptions newOptions = params.options; |
| List<OptionUpdater> updaters = new List<OptionUpdater>(); |
| if (newOptions.generateDart2jsHints != null) { |
| updaters.add((engine.AnalysisOptionsImpl options) { |
| options.dart2jsHint = newOptions.generateDart2jsHints; |
| }); |
| } |
| if (newOptions.generateHints != null) { |
| updaters.add((engine.AnalysisOptionsImpl options) { |
| options.hint = newOptions.generateHints; |
| }); |
| } |
| if (newOptions.generateLints != null) { |
| updaters.add((engine.AnalysisOptionsImpl options) { |
| options.lint = newOptions.generateLints; |
| }); |
| } |
| server.updateOptions(updaters); |
| return new AnalysisUpdateOptionsResult().toResponse(request.id); |
| } |
| } |