blob: 89c35f74818eb69da4325e2da701465ca3439188 [file] [log] [blame]
// Copyright (c) 2021, 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:convert';
import 'dart:io';
import 'dart:typed_data';
import 'package:devtools_shared/devtools_deeplink_io.dart';
import 'package:devtools_shared/devtools_extensions.dart';
import 'package:devtools_shared/devtools_extensions_io.dart';
import 'package:devtools_shared/devtools_server.dart' hide Handler;
import 'package:devtools_shared/devtools_shared.dart';
import 'package:mime/mime.dart';
import 'package:path/path.dart' as path;
import 'package:shelf/shelf.dart';
import 'package:shelf_static/shelf_static.dart';
import 'package:sse/server/sse_handler.dart';
import '../constants.dart';
import '../dds_impl.dart';
import 'client.dart';
import 'utils.dart';
/// Returns a [Handler] which handles serving DevTools and the DevTools server
/// API.
///
/// [buildDir] is the path to the pre-compiled DevTools instance to be served.
///
/// [notFoundHandler] is a [Handler] to which requests that could not be handled
/// by the DevTools handler are forwarded (e.g., a proxy to the VM
/// service).
///
/// If [dds] is null, DevTools is not being served by a DDS instance and is
/// served by a standalone server (see `package:dds/devtools_server.dart`).
///
/// If [dtd] or [dtd.uri] is null, the Dart Tooling Daemon is not available for
/// this DevTools server connection.
///
/// If [dtd.uri] is non-null, but [dtd.secret] is null, then DTD was started by a
/// client that is not the DevTools server (e.g. an IDE).
FutureOr<Handler> defaultHandler({
DartDevelopmentServiceImpl? dds,
String? buildDir,
ClientManager? clientManager,
Handler? notFoundHandler,
DtdInfo? dtd,
required ExtensionsManager devtoolsExtensionsManager,
}) {
// When served through DDS, the app root is /devtools.
// This variable is used in base href and must start and end with `/`
var appRoot = dds != null ? '/devtools/' : '/';
if (dds?.authCodesEnabled ?? false) {
appRoot = '/${dds!.authCode}$appRoot';
}
/// A wrapper around [devtoolsStaticAssetHandler] that handles serving
/// index.html up for / and non-file requests like /memory, /inspector, etc.
/// with the correct base href for the DevTools root.
FutureOr<Response> devtoolsAssetHandler(Request request) {
// To avoid hard-coding a set of page names here (or needing access to one
// from DevTools, assume any single-segment path with no extension is a
// DevTools page that needs to serve up index.html).
final pathSegments = request.url.pathSegments;
final isExtensionRequest = pathSegments.safeGet(0) == extensionRequestPath;
if (isExtensionRequest) {
// This identifier should be the extension name appended with its version.
final extensionIdentifier = pathSegments.safeGet(1);
if (extensionIdentifier != null) {
final extensionAssetsLocation =
devtoolsExtensionsManager.lookupLocationFor(extensionIdentifier);
if (extensionAssetsLocation != null) {
// Remove the first two elements (devtools_extensions/foo_1.0.0) to
// get the relative path to the extension asset.
final relativePathToExtensionAsset =
path.joinAll(pathSegments.sublist(2));
final assetPath = path.normalize(
path.join(extensionAssetsLocation, relativePathToExtensionAsset),
);
// Ensure the normalized path is still within the expected
// [extensionAssetsLocation] to protect against directory traversal.
if (path.isWithin(extensionAssetsLocation, assetPath)) {
final contentType = lookupMimeType(assetPath) ?? 'text/html';
final baseHref =
'$appRoot$extensionRequestPath/$extensionIdentifier/';
return _serveStaticFile(
request,
File(assetPath),
contentType,
baseHref: baseHref,
);
}
}
}
}
if (buildDir == null) {
return Response.notFound('No build directory was specified.');
}
const defaultDocument = 'index.html';
final indexFile = File(path.join(buildDir, defaultDocument));
final isValidRootPage = pathSegments.isEmpty ||
(pathSegments.length == 1 && !pathSegments[0].contains('.'));
if (isValidRootPage) {
return _serveStaticFile(
request,
indexFile,
'text/html',
baseHref: appRoot,
);
}
// Serves the static web assets for DevTools.
final devtoolsStaticAssetHandler = createStaticHandler(
buildDir,
defaultDocument: defaultDocument,
);
return devtoolsStaticAssetHandler(request);
}
// Support DevTools client-server interface via SSE.
// Note: the handler path needs to match the full *original* path, not the
// current request URL (we remove '/devtools' in the initial router but we
// need to include it here).
final devToolsSseHandlerPath = '${appRoot}api/sse';
final devToolsApiHandler = SseHandler(
Uri.parse(devToolsSseHandlerPath),
keepAlive: sseKeepAlive,
);
clientManager ??= ClientManager(requestNotificationPermissions: false);
devToolsApiHandler.connections.rest.listen(
(sseConnection) => clientManager!.acceptClient(
sseConnection,
enableLogging: dds?.shouldLogRequests ?? false,
),
);
FutureOr<Response> devtoolsHandler(Request request) {
// If the request isn't of the form api/<method> assume it's a request for
// DevTools assets.
final pathSegments = request.url.pathSegments;
if (pathSegments.length < 2 || pathSegments.first != 'api') {
return devtoolsAssetHandler(request);
}
final method = request.url.pathSegments[1];
if (method == 'ping') {
// Note: we have an 'OK' body response, otherwise the response has an
// incorrect status code (204 instead of 200).
return Response.ok('OK');
}
if (method == 'sse') {
return devToolsApiHandler.handler(request);
}
if (!ServerApi.canHandle(request)) {
return Response.notFound('$method is not a valid API');
}
return ServerApi.handle(
request,
extensionsManager: devtoolsExtensionsManager,
deeplinkManager: DeeplinkManager(),
dtd: dtd,
);
}
return (Request request) {
if (notFoundHandler != null) {
final pathSegments = request.url.pathSegments;
if (pathSegments.isEmpty || pathSegments.first != 'devtools') {
return notFoundHandler(request);
}
// Forward all requests to /devtools/* to the DevTools handler.
request = request.change(path: 'devtools');
}
return devtoolsHandler(request);
};
}
/// Serves [file] for all requests.
///
/// For files with [contentType] 'text/html' and a provided [baseHref] value,
/// any existing `<base href="">` tag will be rewritten with the provided path.
Future<Response> _serveStaticFile(
Request request,
File file,
String contentType, {
String? baseHref,
}) async {
final headers = {HttpHeaders.contentTypeHeader: contentType};
if (contentType != 'text/html') {
late final Uint8List fileBytes;
try {
fileBytes = file.readAsBytesSync();
} on PathNotFoundException catch (_) {
// Wait a short delay, and then retry in case we have hit a race condition
// between a static file being served and accessed. See
// https://github.com/flutter/devtools/issues/6365.
await Future.delayed(Duration(milliseconds: 500));
try {
fileBytes = file.readAsBytesSync();
} catch (e) {
return Response.notFound(
'could not read file as bytes: ${request.url.path}',
);
}
}
return Response.ok(fileBytes, headers: headers);
}
late String contents;
try {
contents = file.readAsStringSync();
} catch (e) {
return Response.notFound(
'could not read file as String: ${request.url.path}',
);
}
if (baseHref != null) {
assert(baseHref.startsWith('/'));
assert(baseHref.endsWith('/'));
// Always use a relative base href to support going through proxies that
// rewrite paths. For example if the server thinks we are hosting at
// `http://localhost/devtools/` but a frontend proxy means the client app
// is at `http://localhost/proxy/1234/devtools/` then we need the base href
// to effectively be `/proxy/1234/devtools/`, however we have no knowledge
// of `/proxy/1234` because it was stripped by the proxy on its way to us.
baseHref = computeRelativeBaseHref(baseHref, request.requestedUri);
// Replace the base href to match where the app is being served from.
final baseHrefPattern = RegExp(r'<base href="\/"\s?\/?>');
contents = contents.replaceFirst(
baseHrefPattern,
'<base href="${htmlEscape.convert(baseHref)}">',
);
}
return Response.ok(contents, headers: headers);
}
/// Computes a relative "base href" that can be used in place of
/// [absoluteBaseHref] for a request served at [requestUri].
String computeRelativeBaseHref(String absoluteBaseHref, Uri requestUri) {
// path.relative will always treat `from` as if it's a directory, but for
// URIs that's not correct. If the request is /foo/bar then `.` is `/foo` and
// not `/foo/bar`. To handle this, trim the last segment if the request does
// not end with a slash.
final requestFolderPath = requestUri.path.endsWith('/')
? requestUri.path
: path.posix.dirname(requestUri.path);
return path.posix.relative(absoluteBaseHref, from: requestFolderPath);
}