blob: 8f047474330958795d2e78bf95ce8eded094b790 [file] [log] [blame]
// 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:developer' as developer;
import 'dart:io';
import 'package:analysis_server/protocol/protocol_constants.dart'
show PROTOCOL_VERSION;
import 'package:analysis_server/protocol/protocol_generated.dart';
import 'package:analysis_server/src/analysis_server.dart';
import 'package:analysis_server/src/analysis_server_abstract.dart';
import 'package:analysis_server/src/domain_completion.dart';
import 'package:analysis_server/src/lsp/lsp_analysis_server.dart'
show LspAnalysisServer;
import 'package:analysis_server/src/plugin/plugin_manager.dart';
import 'package:analysis_server/src/server/http_server.dart';
import 'package:analysis_server/src/services/completion/completion_performance.dart';
import 'package:analysis_server/src/socket_server.dart';
import 'package:analysis_server/src/status/ast_writer.dart';
import 'package:analysis_server/src/status/element_writer.dart';
import 'package:analysis_server/src/status/pages.dart';
import 'package:analysis_server/src/utilities/profiling.dart';
import 'package:analyzer/src/context/context_root.dart';
import 'package:analyzer/src/context/source.dart';
import 'package:analyzer/src/dart/sdk/sdk.dart';
import 'package:analyzer/src/dartdoc/dartdoc_directive_info.dart';
import 'package:analyzer/src/generated/engine.dart';
import 'package:analyzer/src/generated/source.dart';
import 'package:analyzer/src/source/package_map_resolver.dart';
import 'package:path/path.dart' as path;
final String kCustomCss = '''
.lead, .page-title+.markdown-body>p:first-child {
margin-bottom: 30px;
font-size: 20px;
font-weight: 300;
color: #555;
}
.container {
width: 1160px;
}
.masthead {
padding-top: 1rem;
padding-bottom: 1rem;
margin-bottom: 1.5rem;
text-align: center;
background-color: #4078c0;
}
.masthead .masthead-logo {
display: inline-block;
font-size: 1.5rem;
color: #fff;
float: left;
}
.masthead .mega-octicon {
font-size: 1.5rem;
}
.masthead-nav {
float: right;
margin-top: .5rem;
}
.masthead-nav a:not(:last-child) {
margin-right: 1.25rem;
}
.masthead a {
color: rgba(255,255,255,0.5);
font-size: 1rem;
}
.masthead a:hover {
color: #fff;
text-decoration: none;
}
.masthead-nav .active {
color: #fff;
font-weight: 500;
}
.counter {
display: inline-block;
padding: 2px 5px;
font-size: 11px;
font-weight: bold;
line-height: 1;
color: #666;
background-color: #eee;
border-radius: 20px;
}
.menu-item .counter {
float: right;
margin-left: 5px;
}
td.right {
text-align: right;
}
table td {
max-width: 600px;
vertical-align: text-top;
}
td.pre {
white-space: pre;
}
.nowrap {
white-space: nowrap;
}
.scroll-table {
max-height: 190px;
overflow-x: auto;
}
.footer {
padding-top: 3rem;
padding-bottom: 3rem;
margin-top: 3rem;
line-height: 1.75;
color: #7a7a7a;
border-top: 1px solid #eee;
}
.footer strong {
color: #333;
}
''';
String get _sdkVersion {
var version = Platform.version;
if (version.contains(' ')) {
version = version.substring(0, version.indexOf(' '));
}
return version;
}
String writeOption(String name, dynamic value) {
return '$name: <code>$value</code><br> ';
}
abstract class AbstractCompletionPage extends DiagnosticPageWithNav {
AbstractCompletionPage(DiagnosticsSite site)
: super(site, 'completion', 'Code Completion',
description: 'Latency statistics for code completion.');
path.Context get pathContext;
List<CompletionPerformance> get performanceItems;
@override
Future generateContent(Map<String, String> params) async {
var completions = performanceItems;
if (completions.isEmpty) {
blankslate('No completions recorded.');
return;
}
var fastCount =
completions.where((c) => c.elapsedInMilliseconds <= 100).length;
p('${completions.length} results; ${printPercentage(fastCount / completions.length)} within 100ms.');
// draw a chart
buf.writeln(
'<div id="chart-div" style="width: 700px; height: 300px;"></div>');
var rowData = StringBuffer();
for (var i = completions.length - 1; i >= 0; i--) {
// [' ', 101.5]
if (rowData.isNotEmpty) {
rowData.write(',');
}
rowData.write("[' ', ${completions[i].elapsedInMilliseconds}]");
}
buf.writeln('''
<script type="text/javascript">
google.charts.load('current', {'packages':['bar']});
google.charts.setOnLoadCallback(drawChart);
function drawChart() {
var data = google.visualization.arrayToDataTable([
['Completions', 'Time'],
$rowData
]);
var options = { bars: 'vertical', vAxis: {format: 'decimal'}, height: 300 };
var chart = new google.charts.Bar(document.getElementById('chart-div'));
chart.draw(data, google.charts.Bar.convertOptions(options));
}
</script>
''');
// emit the data as a table
buf.writeln('<table>');
buf.writeln(
'<tr><th>Time</th><th>Results</th><th>Source</th><th>Snippet</th></tr>');
for (var completion in completions) {
var shortName = pathContext.basename(completion.path);
buf.writeln('<tr>'
'<td class="pre right">${printMilliseconds(completion.elapsedInMilliseconds)}</td>'
'<td class="right">${completion.suggestionCountStr}</td>'
'<td>${escape(shortName)}</td>'
'<td><code>${escape(completion.snippet)}</code></td>'
'</tr>');
}
buf.writeln('</table>');
}
}
class AstPage extends DiagnosticPageWithNav {
String _description;
AstPage(DiagnosticsSite site)
: super(site, 'ast', 'AST', description: 'The AST for a file.');
@override
String get description => _description ?? super.description;
@override
bool get showInNav => false;
@override
Future<void> generateContent(Map<String, String> params) async {
var filePath = params['file'];
if (filePath == null) {
p('No file path provided.');
return;
}
var driver = server.getAnalysisDriver(filePath);
if (driver == null) {
p('The file <code>${escape(filePath)}</code> is not being analyzed.',
raw: true);
return;
}
var result = await driver.getResult(filePath);
if (result == null) {
p(
'An AST could not be produced for the file '
'<code>${escape(filePath)}</code>.',
raw: true);
return;
}
var writer = AstWriter(buf);
result.unit.accept(writer);
}
@override
Future<void> generatePage(Map<String, String> params) async {
try {
_description = params['file'];
await super.generatePage(params);
} finally {
_description = null;
}
}
}
class CommunicationsPage extends DiagnosticPageWithNav {
CommunicationsPage(DiagnosticsSite site)
: super(site, 'communications', 'Communications',
description:
'Latency statistics for analysis server communications.');
@override
Future generateContent(Map<String, String> params) async {
void writeRow(List<String> data, {List<String> classes}) {
buf.write('<tr>');
for (var i = 0; i < data.length; i++) {
var c = classes == null ? null : classes[i];
if (c != null) {
buf.write('<td class="$c">${escape(data[i])}</td>');
} else {
buf.write('<td>${escape(data[i])}</td>');
}
}
buf.writeln('</tr>');
}
buf.writeln('<div class="columns">');
if (server.performanceAfterStartup != null) {
buf.writeln('<div class="column one-half">');
h3('Current');
_writePerformanceTable(server.performanceAfterStartup, writeRow);
var time = server.uptime.toString();
if (time.contains('.')) {
time = time.substring(0, time.indexOf('.'));
}
buf.writeln(writeOption('Uptime', time));
buf.write('</div>');
}
buf.writeln('<div class="column one-half">');
h3('Startup');
_writePerformanceTable(server.performanceDuringStartup, writeRow);
if (server.performanceAfterStartup != null) {
var startupTime = server.performanceAfterStartup.startTime -
server.performanceDuringStartup.startTime;
buf.writeln(
writeOption('Initial analysis time', printMilliseconds(startupTime)));
}
buf.write('</div>');
buf.write('</div>');
}
void _writePerformanceTable(ServerPerformance perf,
void Function(List<String> data, {List<String> classes}) writeRow) {
var requestCount = perf.requestCount;
var latencyCount = perf.latencyCount;
var averageLatency =
latencyCount > 0 ? (perf.requestLatency ~/ latencyCount) : 0;
var maximumLatency = perf.maxLatency;
var slowRequestPercent =
latencyCount > 0 ? (perf.slowRequestCount / latencyCount) : 0.0;
buf.write('<table>');
writeRow([printInteger(requestCount), 'requests'],
classes: ['right', null]);
writeRow([printInteger(latencyCount), 'requests with latency information'],
classes: ['right', null]);
if (latencyCount > 0) {
writeRow([printMilliseconds(averageLatency), 'average latency'],
classes: ['right', null]);
writeRow([printMilliseconds(maximumLatency), 'maximum latency'],
classes: ['right', null]);
writeRow([printPercentage(slowRequestPercent), '> 150 ms latency'],
classes: ['right', null]);
}
buf.write('</table>');
}
}
class CompletionPage extends AbstractCompletionPage {
@override
AnalysisServer server;
CompletionPage(DiagnosticsSite site, this.server) : super(site);
CompletionDomainHandler get completionDomain => server.handlers
.firstWhere((handler) => handler is CompletionDomainHandler);
@override
path.Context get pathContext =>
completionDomain.server.resourceProvider.pathContext;
@override
List<CompletionPerformance> get performanceItems =>
completionDomain.performanceList.items.toList();
}
class ContentsPage extends DiagnosticPageWithNav {
String _description;
ContentsPage(DiagnosticsSite site)
: super(site, 'contents', 'Contents',
description: 'The Contents/Overlay of a file.');
@override
String get description => _description ?? super.description;
@override
bool get showInNav => false;
@override
Future<void> generateContent(Map<String, String> params) async {
var filePath = params['file'];
if (filePath == null) {
p('No file path provided.');
return;
}
var driver = server.getAnalysisDriver(filePath);
if (driver == null) {
p('The file <code>${escape(filePath)}</code> is not being analyzed.',
raw: true);
return;
}
var file = server.resourceProvider.getFile(filePath);
if (!file.exists) {
p('The file <code>${escape(filePath)}</code> does not exist.', raw: true);
return;
}
if (server.resourceProvider.hasOverlay(filePath)) {
p('Showing overlay for file.');
} else {
p('Showing file system contents for file.');
}
pre(() {
buf.write('<code>');
buf.write(escape(file.readAsStringSync()));
buf.writeln('</code>');
});
}
@override
Future<void> generatePage(Map<String, String> params) async {
try {
_description = params['file'];
await super.generatePage(params);
} finally {
_description = null;
}
}
}
class ContextsPage extends DiagnosticPageWithNav {
ContextsPage(DiagnosticsSite site)
: super(site, 'contexts', 'Contexts',
description:
'An analysis context defines the options and the set of sources being analyzed.');
@override
String get navDetail => printInteger(server.driverMap.length);
String describe(AnalysisOptionsImpl options) {
var b = StringBuffer();
b.write(writeOption('Implicit dynamic', options.implicitDynamic));
b.write(writeOption('Implicit casts', options.implicitCasts));
b.write(writeOption('Feature set', options.contextFeatures.toString()));
b.write('<br>');
b.write(writeOption('Generate hints', options.hint));
return b.toString();
}
@override
Future generateContent(Map<String, String> params) async {
var driverMap = server.driverMap;
if (driverMap.isEmpty) {
blankslate('No contexts.');
return;
}
var contextPath = params['context'];
var folders = driverMap.keys.toList();
folders
.sort((first, second) => first.shortName.compareTo(second.shortName));
var folder =
folders.firstWhere((f) => f.path == contextPath, orElse: () => null);
if (folder == null) {
folder = folders.first;
contextPath = folder.path;
}
var driver = driverMap[folder];
buf.writeln('<div class="tabnav">');
buf.writeln('<nav class="tabnav-tabs">');
for (var f in folders) {
if (f == folder) {
buf.writeln(
'<a class="tabnav-tab selected">${escape(f.shortName)}</a>');
} else {
var p = '${this.path}?context=${Uri.encodeQueryComponent(f.path)}';
buf.writeln(
'<a href="$p" class="tabnav-tab">${escape(f.shortName)}</a>');
}
}
buf.writeln('</nav>');
buf.writeln('</div>');
buf.writeln(writeOption('Context location', escape(contextPath)));
buf.writeln(writeOption('Analysis options path',
escape(driver.contextRoot.optionsFilePath ?? 'none')));
buf.writeln('<div class="columns">');
buf.writeln('<div class="column one-half">');
h3('Analysis options');
p(describe(driver.analysisOptions), raw: true);
h3('Pub files');
buf.writeln('<p>');
buf.writeln(
writeOption('Has .packages file', folder.getChild('.packages').exists));
buf.writeln(writeOption(
'Has pubspec.yaml file', folder.getChild('pubspec.yaml').exists));
buf.writeln('</p>');
buf.writeln('</div>');
buf.writeln('</div>');
h3('Lints');
var lints = driver.analysisOptions.lintRules.map((l) => l.name).toList()
..sort();
ul(lints, (String lint) => buf.write(lint), classes: 'scroll-table');
h3('Error processors');
p(driver.analysisOptions.errorProcessors
.map((e) => e.description)
.join(', '));
h3('Plugins');
p(driver.analysisOptions.enabledPluginNames.join(', '));
var priorityFiles = driver.priorityFiles;
var addedFiles = driver.addedFiles.toList();
var implicitFiles =
driver.knownFiles.difference(driver.addedFiles).toList();
addedFiles.sort();
implicitFiles.sort();
String lenCounter(List list) {
return '<span class="counter" style="float: right;">${list.length}</span>';
}
h3('Context files');
void writeFile(String file) {
var astPath = '/ast?file=${Uri.encodeQueryComponent(file)}';
var elementPath = '/element?file=${Uri.encodeQueryComponent(file)}';
var contentsPath = '/contents?file=${Uri.encodeQueryComponent(file)}';
var hasOverlay = server.resourceProvider.hasOverlay(file);
buf.write(file);
buf.writeln(' <a href="$astPath">ast</a>');
buf.writeln(' <a href="$elementPath">element</a>');
buf.writeln(
' <a href="$contentsPath">contents${hasOverlay ? '*' : ''}</a>');
}
h4('Priority files ${lenCounter(priorityFiles)}', raw: true);
ul(priorityFiles, writeFile, classes: 'scroll-table');
h4('Added files ${lenCounter(addedFiles)}', raw: true);
ul(addedFiles, writeFile, classes: 'scroll-table');
h4('Implicit files ${lenCounter(implicitFiles)}', raw: true);
ul(implicitFiles, writeFile, classes: 'scroll-table');
var sourceFactory = driver.sourceFactory;
if (sourceFactory is SourceFactoryImpl) {
h3('Resolvers');
for (var resolver in sourceFactory.resolvers) {
h4(resolver.runtimeType.toString());
buf.write('<p class="scroll-table">');
if (resolver is DartUriResolver) {
var sdk = resolver.dartSdk;
buf.write(' (sdk = ');
buf.write(sdk.runtimeType);
if (sdk is FolderBasedDartSdk) {
buf.write(' (path = ');
buf.write(sdk.directory.path);
buf.write(')');
} else if (sdk is EmbedderSdk) {
buf.write(' (map = ');
writeMap(sdk.urlMappings);
buf.write(')');
}
buf.write(')');
} else if (resolver is PackageMapUriResolver) {
writeMap(resolver.packageMap);
}
buf.write('</p>');
}
}
h3('Dartdoc template info');
var info = server.declarationsTracker
?.getContext(driver.analysisContext)
?.dartdocDirectiveInfo ??
DartdocDirectiveInfo();
buf.write('<p class="scroll-table">');
writeMap(info.templateMap);
buf.write('</p>');
}
void writeList<E>(List<E> list) {
buf.writeln('[${list.join(', ')}]');
}
void writeMap<V>(Map<String, V> map) {
var keys = map.keys.toList();
keys.sort();
var length = keys.length;
buf.write('{');
for (var i = 0; i < length; i++) {
buf.write('<br>');
var key = keys[i];
var value = map[key];
buf.write(key);
buf.write(' = ');
if (value is List) {
writeList(value);
} else {
buf.write(value);
}
buf.write(',');
}
buf.write('<br>}');
}
}
/// A page with a proscriptive notion of layout.
abstract class DiagnosticPage extends Page {
final DiagnosticsSite site;
DiagnosticPage(this.site, String id, String title, {String description})
: super(id, title, description: description);
bool get isNavPage => false;
AbstractAnalysisServer get server => site.socketServer.analysisServer;
Future<void> generateContainer(Map<String, String> params) async {
buf.writeln('<div class="columns docs-layout">');
buf.writeln('<div class="three-fourths column markdown-body">');
h1(title, classes: 'page-title');
await asyncDiv(() async {
p(description);
await generateContent(params);
}, classes: 'markdown-body');
buf.writeln('</div>');
buf.writeln('</div>');
}
Future<void> generateContent(Map<String, String> params);
void generateFooter() {
buf.writeln('''
<footer class="footer">
Dart ${site.title} <span style="float:right">SDK $_sdkVersion</span>
</footer>
''');
}
void generateHeader() {
buf.writeln('''
<header class="masthead">
<div class="container">
<span class="masthead-logo">
<span class="mega-octicon octicon-dashboard"></span>
${site.title} Diagnostics
</span>
<nav class="masthead-nav">
<a href="/status" ${isNavPage ? ' class="active"' : ''}>Diagnostics</a>
<a href="/feedback" ${isCurrentPage('/feedback') ? ' class="active"' : ''}>Feedback</a>
<a href="https://dart.dev/tools/dartanalyzer" target="_blank">Docs</a>
<a href="https://htmlpreview.github.io/?https://github.com/dart-lang/sdk/blob/master/pkg/analysis_server/doc/api.html" target="_blank">Spec</a>
</nav>
</div>
</header>
''');
}
@override
Future<void> generatePage(Map<String, String> params) async {
buf.writeln('<!DOCTYPE html><html lang="en">');
buf.write('<head>');
buf.write('<meta charset="utf-8">');
buf.write('<meta name="viewport" content="width=device-width, '
'initial-scale=1.0">');
buf.writeln('<title>${site.title}</title>');
buf.writeln('<link rel="stylesheet" '
'href="https://cdnjs.cloudflare.com/ajax/libs/Primer/6.0.0/build.css">');
buf.writeln('<link rel="stylesheet" '
'href="https://cdnjs.cloudflare.com/ajax/libs/octicons/4.4.0/font/octicons.css">');
buf.writeln('<script type="text/javascript" '
'src="https://www.gstatic.com/charts/loader.js"></script>');
buf.writeln('<style>${site.customCss}</style>');
buf.writeln('</head>');
buf.writeln('<body>');
generateHeader();
buf.writeln('<div class="container">');
await generateContainer(params);
generateFooter();
buf.writeln('</div>'); // div.container
buf.writeln('</body>');
buf.writeln('</html>');
}
}
abstract class DiagnosticPageWithNav extends DiagnosticPage {
DiagnosticPageWithNav(DiagnosticsSite site, String id, String title,
{String description})
: super(site, id, title, description: description);
@override
bool get isNavPage => true;
String get navDetail => null;
bool get showInNav => true;
@override
Future<void> generateContainer(Map<String, String> params) async {
buf.writeln('<div class="columns docs-layout">');
bool shouldShowInNav(Page page) {
return page is DiagnosticPageWithNav && page.showInNav;
}
buf.writeln('<div class="one-fifth column">');
buf.writeln('<nav class="menu docs-menu">');
for (var page in site.pages.where(shouldShowInNav)) {
buf.write('<a class="menu-item ${page == this ? ' selected' : ''}" '
'href="${page.path}">${escape(page.title)}');
var detail = (page as DiagnosticPageWithNav).navDetail;
if (detail != null) {
buf.write('<span class="counter">$detail</span>');
}
buf.writeln('</a>');
}
buf.writeln('</nav>');
buf.writeln('</div>');
buf.writeln('<div class="four-fifths column markdown-body">');
h1(title, classes: 'page-title');
await asyncDiv(() async {
p(description);
await generateContent(params);
}, classes: 'markdown-body');
buf.writeln('</div>');
buf.writeln('</div>');
}
}
class DiagnosticsSite extends Site implements AbstractGetHandler {
/// An object that can handle either a WebSocket connection or a connection
/// to the client over stdio.
AbstractSocketServer socketServer;
/// The last few lines printed.
final List<String> lastPrintedLines;
DiagnosticsSite(this.socketServer, this.lastPrintedLines)
: super('Analysis Server') {
pages.add(CommunicationsPage(this));
pages.add(ContextsPage(this));
pages.add(EnvironmentVariablesPage(this));
pages.add(ExceptionsPage(this));
//pages.add(new InstrumentationPage(this));
// Add server-specific pages. Ordering doesn't matter as the items are
// sorted later.
var server = socketServer.analysisServer;
pages.add(PluginsPage(this, server));
if (server is AnalysisServer) {
pages.add(CompletionPage(this, server));
pages.add(SubscriptionsPage(this, server));
} else if (server is LspAnalysisServer) {
pages.add(LspCompletionPage(this, server));
pages.add(LspCapabilitiesPage(this, server));
}
var profiler = ProcessProfiler.getProfilerForPlatform();
if (profiler != null) {
pages.add(MemoryAndCpuPage(this, profiler));
}
pages.sort(((Page a, Page b) =>
a.title.toLowerCase().compareTo(b.title.toLowerCase())));
// Add the status page at the beginning.
pages.insert(0, StatusPage(this));
// Add non-nav pages.
pages.add(FeedbackPage(this));
pages.add(AstPage(this));
pages.add(ElementModelPage(this));
pages.add(ContentsPage(this));
}
@override
String get customCss => kCustomCss;
@override
Page createExceptionPage(String message, StackTrace trace) =>
ExceptionPage(this, message, trace);
@override
Page createUnknownPage(String unknownPath) => NotFoundPage(this, unknownPath);
}
class ElementModelPage extends DiagnosticPageWithNav {
String _description;
ElementModelPage(DiagnosticsSite site)
: super(site, 'element', 'Element model',
description: 'The element model for a file.');
@override
String get description => _description ?? super.description;
@override
bool get showInNav => false;
@override
Future<void> generateContent(Map<String, String> params) async {
var filePath = params['file'];
if (filePath == null) {
p('No file path provided.');
return;
}
var driver = server.getAnalysisDriver(filePath);
if (driver == null) {
p('The file <code>${escape(filePath)}</code> is not being analyzed.',
raw: true);
return;
}
var result = await driver.getResult(filePath);
if (result == null) {
p(
'An element model could not be produced for the file '
'<code>${escape(filePath)}</code>.',
raw: true);
return;
}
var writer = ElementWriter(buf);
result.unit.declaredElement.accept(writer);
}
@override
Future<void> generatePage(Map<String, String> params) async {
try {
_description = params['file'];
await super.generatePage(params);
} finally {
_description = null;
}
}
}
class EnvironmentVariablesPage extends DiagnosticPageWithNav {
EnvironmentVariablesPage(DiagnosticsSite site)
: super(site, 'environment', 'Environment Variables',
description:
'System environment variables as seen from the analysis server.');
@override
Future generateContent(Map<String, String> params) async {
buf.writeln('<table>');
buf.writeln('<tr><th>Variable</th><th>Value</th></tr>');
for (var key in Platform.environment.keys.toList()..sort()) {
var value = Platform.environment[key];
buf.writeln('<tr><td>${escape(key)}</td><td>${escape(value)}</td></tr>');
}
buf.writeln('</table>');
}
}
class ExceptionPage extends DiagnosticPage {
final StackTrace trace;
ExceptionPage(DiagnosticsSite site, String message, this.trace)
: super(site, '', '500 Oops', description: message);
@override
Future generateContent(Map<String, String> params) async {
p(trace.toString(), style: 'white-space: pre');
}
}
class ExceptionsPage extends DiagnosticPageWithNav {
ExceptionsPage(DiagnosticsSite site)
: super(site, 'exceptions', 'Exceptions',
description: 'Exceptions from the analysis server.');
Iterable<ServerException> get exceptions => server.exceptions.items;
@override
String get navDetail => printInteger(exceptions.length);
@override
Future generateContent(Map<String, String> params) async {
if (exceptions.isEmpty) {
blankslate('No exceptions encountered!');
} else {
for (var ex in exceptions) {
h3('Exception ${ex.exception}');
p('${escape(ex.message)}<br>${writeOption('fatal', ex.fatal)}',
raw: true);
pre(() {
buf.writeln('<code>${escape(ex.stackTrace.toString())}</code>');
}, classes: 'scroll-table');
}
}
}
}
class FeedbackPage extends DiagnosticPage {
FeedbackPage(DiagnosticsSite site)
: super(site, 'feedback', 'Feedback',
description: 'Providing feedback and filing issues.');
@override
Future generateContent(Map<String, String> params) async {
var issuesUrl = 'https://github.com/dart-lang/sdk/issues';
p(
'To file issues or feature requests, see our '
'<a href="$issuesUrl">bug tracker</a>. When filing an issue, please describe:',
raw: true,
);
ul([
'what you were doing',
'what occurred',
'what you think the expected behavior should have been',
], (line) => buf.writeln(line));
var ideInfo = <String>[];
if (server.options.clientId != null) {
ideInfo.add(server.options.clientId);
}
if (server.options.clientVersion != null) {
ideInfo.add(server.options.clientVersion);
}
var ideText = ideInfo.map((str) => '<code>$str</code>').join(', ');
p('Other data to include:');
ul([
"the IDE you are using and it's version${ideText.isEmpty ? '' : ' ($ideText)'}",
'the Dart SDK version (<code>${escape(_sdkVersion)}</code>)',
'your operating system (<code>${escape(Platform.operatingSystem)}</code>)',
], (line) => buf.writeln(line));
p('Thanks!');
}
}
class LspCapabilitiesPage extends DiagnosticPageWithNav {
@override
LspAnalysisServer server;
LspCapabilitiesPage(DiagnosticsSite site, this.server)
: super(site, 'lsp_capabilities', 'LSP Capabilities',
description: 'Client and Server LSP Capabilities.');
@override
Future generateContent(Map<String, String> params) async {
buf.writeln('<div class="columns">');
buf.writeln('<div class="column one-half">');
h3('Client Capabilities');
if (server.clientCapabilities == null) {
p('Client capabilities have not yet been received.');
} else {
prettyJson(server.clientCapabilities.toJson());
}
buf.writeln('</div>');
buf.writeln('<div class="column one-half">');
h3('Server Capabilities');
if (server.capabilities == null) {
p('Server capabilities have not yet been computed.');
} else {
prettyJson(server.capabilities.toJson());
}
buf.writeln('</div>'); // half for server capabilities
buf.writeln('</div>'); // columns
h3('Current registrations');
p('Showing the LSP method name and the registration params sent to the '
'client.');
prettyJson(server.capabilitiesComputer.currentRegistrations);
}
}
// class InstrumentationPage extends DiagnosticPageWithNav {
// InstrumentationPage(DiagnosticsSite site)
// : super(site, 'instrumentation', 'Instrumentation',
// description:
// 'Verbose instrumentation data from the analysis server.');
//
// @override
// Future generateContent(Map<String, String> params) async {
// p(
// 'Instrumentation can be enabled by starting the analysis server with the '
// '<code>--instrumentation-log-file=path/to/file</code> flag.',
// raw: true);
//
// if (!AnalysisEngine.instance.instrumentationService.isActive) {
// blankslate('Instrumentation not active.');
// return;
// }
//
// h3('Instrumentation');
//
// p('Instrumentation active.');
//
// InstrumentationServer instrumentation =
// AnalysisEngine.instance.instrumentationService.instrumentationServer;
// String description = instrumentation.describe;
// HtmlEscape htmlEscape = new HtmlEscape(HtmlEscapeMode.element);
// description = htmlEscape.convert(description);
// // Convert http(s): references to hyperlinks.
// final RegExp urlRegExp = new RegExp(r'[http|https]+:\/*(\S+)');
// description = description.replaceAllMapped(urlRegExp, (Match match) {
// return '<a href="${match.group(0)}">${match.group(1)}</a>';
// });
// p(description.replaceAll('\n', '<br>'), raw: true);
// }
// }
class LspCompletionPage extends AbstractCompletionPage {
@override
LspAnalysisServer server;
LspCompletionPage(DiagnosticsSite site, this.server) : super(site);
@override
path.Context get pathContext => server.resourceProvider.pathContext;
@override
List<CompletionPerformance> get performanceItems =>
server.performanceStats.completion.items.toList();
}
class MemoryAndCpuPage extends DiagnosticPageWithNav {
final ProcessProfiler profiler;
MemoryAndCpuPage(DiagnosticsSite site, this.profiler)
: super(site, 'memory', 'Memory and CPU Usage',
description: 'Memory and CPU usage for the analysis server.');
@override
Future generateContent(Map<String, String> params) async {
var usage = await profiler.getProcessUsage(pid);
var serviceProtocolInfo = await developer.Service.getInfo();
if (usage != null) {
buf.writeln(
writeOption('CPU', printPercentage(usage.cpuPercentage / 100.0)));
buf.writeln(
writeOption('Memory', '${printInteger(usage.memoryMB.round())} MB'));
h3('VM');
if (serviceProtocolInfo.serverUri == null) {
p('Service protocol not enabled.');
} else {
buf.writeln(writeOption('Service protocol connection available at',
'${serviceProtocolInfo.serverUri}'));
buf.writeln('<br>');
p(
'To get detailed performance data on the analysis server, we '
'recommend using Dart DevTools. For instructions on installing and '
'using DevTools, see '
'<a href="https://dart.dev/tools/dart-devtools">dart.dev/tools/dart-devtools</a>.',
raw: true,
);
}
} else {
p('Error retrieving the memory and cpu usage information.');
}
}
}
class NotFoundPage extends DiagnosticPage {
@override
final String path;
NotFoundPage(DiagnosticsSite site, this.path)
: super(site, '', '404 Not found', description: "'$path' not found.");
@override
Future generateContent(Map<String, String> params) async {}
}
class PluginsPage extends DiagnosticPageWithNav {
@override
AbstractAnalysisServer server;
PluginsPage(DiagnosticsSite site, this.server)
: super(site, 'plugins', 'Plugins', description: 'Plugins in use.');
@override
Future generateContent(Map<String, String> params) async {
h3('Analysis plugins');
var analysisPlugins = server.pluginManager.plugins;
if (analysisPlugins.isEmpty) {
blankslate('No known analysis plugins.');
} else {
analysisPlugins
.sort((first, second) => first.pluginId.compareTo(second.pluginId));
for (var plugin in analysisPlugins) {
var id = plugin.pluginId;
var data = plugin.data;
var responseTimes = PluginManager.pluginResponseTimes[plugin];
var components = path.split(id);
var length = components.length;
String name;
if (length == 0) {
name = 'unknown plugin';
} else if (length > 2) {
name = components[length - 3];
} else {
name = components[length - 1];
}
h4(name);
p('bootstrap package path: $id');
if (plugin is DiscoveredPluginInfo) {
p('execution path: ${plugin.executionPath}');
p('packages file path: ${plugin.packagesPath}');
}
if (data.name == null) {
if (plugin.exception != null) {
p('not running');
pre(() {
buf.write(plugin.exception);
});
} else {
p('not running for unknown reason');
}
} else {
p('name: ${data.name}');
p('version: ${data.version}');
p('Associated contexts:');
var contexts = plugin.contextRoots;
if (contexts.isEmpty) {
blankslate('none');
} else {
ul(contexts.toList(), (ContextRoot root) {
buf.writeln(root.root);
});
}
p('Performance:');
var requestNames = responseTimes.keys.toList();
requestNames.sort();
for (var requestName in requestNames) {
var data = responseTimes[requestName];
// TODO(brianwilkerson) Consider displaying these times as a graph,
// similar to the one in AbstractCompletionPage.generateContent.
var buffer = StringBuffer();
buffer.write(requestName);
buffer.write(' ');
buffer.write(data);
p(buffer.toString());
}
}
}
}
}
}
class StatusPage extends DiagnosticPageWithNav {
StatusPage(DiagnosticsSite site)
: super(site, 'status', 'Status',
description:
'General status and diagnostics for the analysis server.');
@override
Future generateContent(Map<String, String> params) async {
buf.writeln('<div class="columns">');
buf.writeln('<div class="column one-half">');
h3('Status');
buf.writeln(writeOption('Server type', server.runtimeType));
// buf.writeln(writeOption('Instrumentation enabled',
// AnalysisEngine.instance.instrumentationService.isActive));
buf.writeln(writeOption('Server process ID', pid));
buf.writeln('</div>');
buf.writeln('<div class="column one-half">');
h3('Versions');
buf.writeln(writeOption('Analysis server version', PROTOCOL_VERSION));
buf.writeln(writeOption('Dart SDK', Platform.version));
buf.writeln('</div>');
buf.writeln('</div>');
// SDK configuration overrides.
var sdkConfig = server.options.configurationOverrides;
if (sdkConfig?.hasAnyOverrides == true) {
buf.writeln('<div class="columns">');
buf.writeln('<div class="column one-half">');
h3('Configuration Overrides');
buf.writeln('<pre><code>${sdkConfig.displayString}</code></pre><br>');
buf.writeln('</div>');
buf.writeln('</div>');
}
var lines = site.lastPrintedLines;
if (lines.isNotEmpty) {
h3('Debug output');
p(lines.join('\n'), style: 'white-space: pre');
}
}
}
class SubscriptionsPage extends DiagnosticPageWithNav {
@override
AnalysisServer server;
SubscriptionsPage(DiagnosticsSite site, this.server)
: super(site, 'subscriptions', 'Subscriptions',
description: 'Registered subscriptions to analysis server events.');
@override
Future generateContent(Map<String, String> params) async {
// server domain
h3('Server domain subscriptions');
ul(ServerService.VALUES, (item) {
if (server.serverServices.contains(item)) {
buf.write('$item (has subscriptions)');
} else {
buf.write('$item (no subscriptions)');
}
});
// analysis domain
h3('Analysis domain subscriptions');
for (var service in AnalysisService.VALUES) {
buf.writeln('${service.name}<br>');
ul(server.analysisServices[service] ?? [], (item) {
buf.write('$item');
});
}
// completion domain
CompletionDomainHandler handler = server.handlers.firstWhere(
(handler) => handler is CompletionDomainHandler,
orElse: () => null);
h3('Completion domain subscriptions');
ul(CompletionService.VALUES, (service) {
if (handler.subscriptions.contains(service)) {
buf.write('$service (has subscriptions)');
} else {
buf.write('$service (no subscriptions)');
}
});
}
}