blob: 23bb09b9091acefec62cebbfcbdf2e1e807acf39 [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.
/// A helper library to connect to an existing VM and trigger a hot-reload via
/// its service protocol.
///
/// Usage:
///
/// ```
/// var remoteVm = new RemoteVm();
/// await remoteVm.reload(uriToEntryScript);
/// ...
/// await remoteVm.disconnect();
/// ```
library front_end.src.vm.reload;
import 'dart:async';
import 'package:json_rpc_2/json_rpc_2.dart' as json_rpc;
import 'package:web_socket_channel/io.dart';
/// APIs to communicate with a remote VM via the VM's service protocol.
///
/// Only supports APIs to resume the program execution (when isolates are paused
/// at startup) and to trigger hot reloads.
class RemoteVm {
/// Port used to connect to the vm service protocol, typically 8181.
final int port;
/// An peer point used to send service protocol messages. The service
/// protocol uses JSON rpc on top of web-sockets.
json_rpc.Peer get rpc => _rpc ??= _createPeer();
json_rpc.Peer? _rpc;
/// The main isolate ID of the running VM. Needed to indicate to the VM which
/// isolate to reload.
Future<String> get mainId async => _mainId ??= await _computeMainId();
String? _mainId;
/// The incoming message stream from the VM.
final Map<String, StreamController> _eventStreams = {};
Stream getEventStream(String streamId) {
final existing = _eventStreams[streamId];
if (existing != null) return existing.stream;
late final StreamController controller;
controller = StreamController.broadcast(onListen: () {
rpc.sendRequest('streamListen', {'streamId': streamId}).catchError(
(error, stack) {
controller.addError(error, stack);
});
}, onCancel: () {
rpc.sendRequest('streamCancel', {'streamId': streamId}).catchError(
(error, stack) {
controller.addError(error, stack);
});
});
return (_eventStreams[streamId] = controller).stream;
}
RemoteVm([this.port = 8181]);
/// Establishes the JSON rpc connection.
json_rpc.Peer _createPeer() {
var socket = new IOWebSocketChannel.connect('ws://127.0.0.1:$port/ws');
var peer = new json_rpc.Peer(socket.cast<String>());
peer.listen().then((_) {
if (VERBOSE_DEBUG) print('connection to vm-service closed');
return disconnect();
}).catchError((e) {
if (VERBOSE_DEBUG) print('error connecting to the vm-service');
return disconnect();
});
peer.registerMethod('streamNotify', (arg) {
final response = (arg as json_rpc.Parameters).asMap;
final streamId = response['streamId'];
final controller = _eventStreams[streamId];
if (controller != null) {
controller.add(response['event']);
}
});
return peer;
}
/// Retrieves the ID of the main isolate using the service protocol.
Future<String> _computeMainId() async {
final isolateStartEventFuture =
getEventStream('Isolate').firstWhere((event) {
return event['kind'] == 'IsolateStart';
});
var vm = await rpc.sendRequest('getVM', {});
var isolates = vm['isolates'];
for (var isolate in isolates) {
if (isolate['name'].contains(r'$main')) {
return isolate['id'];
}
}
for (var isolate in isolates) {
return isolate['id'];
}
final isolateStartEvent = await isolateStartEventFuture;
return isolateStartEvent['isolate']['id'];
}
/// Send a request to the VM to reload sources from [entryUri].
///
/// This will establish a connection with the VM assuming it is running on the
/// local machine and listening on [port] for service protocol requests.
///
/// The result is the JSON map received from the reload request.
Future<Map> reload(Uri entryUri) async {
var id = await mainId;
var result = await rpc.sendRequest('reloadSources', {
'isolateId': id,
'rootLibUri': entryUri.toFilePath(),
});
return result;
}
Future resume() async {
var id = await mainId;
await rpc.sendRequest('resume', {'isolateId': id});
}
/// Close any connections used to communicate with the VM.
Future disconnect() async {
if (_rpc == null) return null;
this._mainId = null;
if (!_rpc!.isClosed) {
var future = _rpc!.close();
_rpc = null;
return future;
}
return null;
}
}
const VERBOSE_DEBUG = false;
/// This library can be used as a script as well. It connects to an existing
/// VM's service protocol and issues a hot-reload request. The VM must have been
/// launched with `--observe` to enable the service protocol.
///
// TODO(sigmund): provide flags to configure the vm-service port.
Future<void> main(List<String> args) async {
if (args.length == 0) {
print('usage: reload <entry-uri>');
return;
}
var remoteVm = new RemoteVm();
await remoteVm.reload(Uri.base.resolve(args.first));
await remoteVm.disconnect();
}