| Index: tool/input_sdk/lib/io/io_resource_info.dart
|
| diff --git a/tool/input_sdk/lib/io/io_resource_info.dart b/tool/input_sdk/lib/io/io_resource_info.dart
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..7cea2312493e5bd29eb6d4798d0cd5838717d79d
|
| --- /dev/null
|
| +++ b/tool/input_sdk/lib/io/io_resource_info.dart
|
| @@ -0,0 +1,283 @@
|
| +// Copyright (c) 2015, 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.
|
| +
|
| +part of dart.io;
|
| +
|
| +abstract class _IOResourceInfo {
|
| + final String type;
|
| + final int id;
|
| + String get name;
|
| + static int _count = 0;
|
| +
|
| + static final Stopwatch _sw = new Stopwatch()..start();
|
| + static final _startTime = new DateTime.now().millisecondsSinceEpoch;
|
| +
|
| + static double get timestamp => _startTime + _sw.elapsedMicroseconds/1000;
|
| +
|
| + _IOResourceInfo(this.type) : id = _IOResourceInfo.getNextID();
|
| +
|
| + /// Get the full set of values for a specific implementation. This is normally
|
| + /// looked up based on an id from a referenceValueMap.
|
| + Map<String, String> get fullValueMap;
|
| +
|
| + /// The reference map, used to return a list of values, e.g., getting
|
| + /// all open sockets. The structure of this is shared among all subclasses.
|
| + Map<String, String> get referenceValueMap =>
|
| + {
|
| + // The type for a reference object is prefixed with @ in observatory.
|
| + 'type': '@$type',
|
| + 'id': id,
|
| + 'name': name,
|
| + };
|
| +
|
| + static int getNextID() => _count++;
|
| +}
|
| +
|
| +abstract class _ReadWriteResourceInfo extends _IOResourceInfo {
|
| + int totalRead;
|
| + int totalWritten;
|
| + int readCount;
|
| + int writeCount;
|
| + double lastRead;
|
| + double lastWrite;
|
| +
|
| + // Not all call sites use this. In some cases, e.g., a socket, a read does
|
| + // not always mean that we actually read some bytes (we may do a read to see
|
| + // if there are some bytes available).
|
| + void addRead(int bytes) {
|
| + totalRead += bytes;
|
| + readCount++;
|
| + lastRead = _IOResourceInfo.timestamp;
|
| + }
|
| +
|
| + // In cases where we read but did not neccesarily get any bytes, use this to
|
| + // update the readCount and timestamp. Manually update totalRead if any bytes
|
| + // where acutally read.
|
| + void didRead() { addRead(0); }
|
| +
|
| + void addWrite(int bytes) {
|
| + totalWritten += bytes;
|
| + writeCount++;
|
| + lastWrite = _IOResourceInfo.timestamp;
|
| + }
|
| +
|
| + _ReadWriteResourceInfo(String type) :
|
| + totalRead = 0,
|
| + totalWritten = 0,
|
| + readCount = 0,
|
| + writeCount = 0,
|
| + lastRead = 0.0,
|
| + lastWrite = 0.0,
|
| + super(type);
|
| +
|
| + Map<String, String> get fullValueMap =>
|
| + {
|
| + 'type': type,
|
| + 'id': id,
|
| + 'name': name,
|
| + 'totalRead': totalRead,
|
| + 'totalWritten': totalWritten,
|
| + 'readCount': readCount,
|
| + 'writeCount': writeCount,
|
| + 'lastRead': lastRead,
|
| + 'lastWrite': lastWrite
|
| + };
|
| +}
|
| +
|
| +class _FileResourceInfo extends _ReadWriteResourceInfo {
|
| + static const String TYPE = '_file';
|
| +
|
| + final file;
|
| +
|
| + static Map<int, _FileResourceInfo> openFiles =
|
| + new Map<int, _FileResourceInfo>();
|
| +
|
| + _FileResourceInfo(this.file) : super(TYPE) {
|
| + FileOpened(this);
|
| + }
|
| +
|
| + static FileOpened(_FileResourceInfo info) {
|
| + assert(!openFiles.containsKey(info.id));
|
| + openFiles[info.id] = info;
|
| + }
|
| +
|
| + static FileClosed(_FileResourceInfo info) {
|
| + assert(openFiles.containsKey(info.id));
|
| + openFiles.remove(info.id);
|
| + }
|
| +
|
| + static Iterable<Map<String, String>> getOpenFilesList() {
|
| + return new List.from(openFiles.values.map((e) => e.referenceValueMap));
|
| + }
|
| +
|
| + static Future<ServiceExtensionResponse> getOpenFiles(function, params) {
|
| + assert(function == 'ext.dart.io.getOpenFiles');
|
| + var data = {'type': '_openfiles', 'data': getOpenFilesList()};
|
| + var json = JSON.encode(data);
|
| + return new Future.value(new ServiceExtensionResponse.result(json));
|
| + }
|
| +
|
| + Map<String, String> getFileInfoMap() {
|
| + var result = fullValueMap;
|
| + return result;
|
| + }
|
| +
|
| + static Future<ServiceExtensionResponse> getFileInfoMapByID(function, params) {
|
| + assert(params.containsKey('id'));
|
| + var id = int.parse(params['id']);
|
| + var result =
|
| + openFiles.containsKey(id) ? openFiles[id].getFileInfoMap() : {};
|
| + var json = JSON.encode(result);
|
| + return new Future.value(new ServiceExtensionResponse.result(json));
|
| + }
|
| +
|
| + String get name {
|
| + return '${file.path}';
|
| + }
|
| +}
|
| +
|
| +class _ProcessResourceInfo extends _IOResourceInfo{
|
| + static const String TYPE = '_process';
|
| + final process;
|
| + final double startedAt;
|
| +
|
| + static Map<int, _ProcessResourceInfo> startedProcesses =
|
| + new Map<int, _ProcessResourceInfo>();
|
| +
|
| + _ProcessResourceInfo(this.process) :
|
| + startedAt = _IOResourceInfo.timestamp,
|
| + super(TYPE) {
|
| + ProcessStarted(this);
|
| + }
|
| +
|
| + String get name => process._path;
|
| +
|
| + void stopped() { ProcessStopped(this); }
|
| +
|
| + Map<String, String> get fullValueMap =>
|
| + {
|
| + 'type': type,
|
| + 'id': id,
|
| + 'name': name,
|
| + 'pid': process.pid,
|
| + 'startedAt': startedAt,
|
| + 'arguments': process._arguments,
|
| + 'workingDirectory':
|
| + process._workingDirectory == null ? '.' : process._workingDirectory,
|
| + };
|
| +
|
| + static ProcessStarted(_ProcessResourceInfo info) {
|
| + assert(!startedProcesses.containsKey(info.id));
|
| + startedProcesses[info.id] = info;
|
| + }
|
| +
|
| + static ProcessStopped(_ProcessResourceInfo info) {
|
| + assert(startedProcesses.containsKey(info.id));
|
| + startedProcesses.remove(info.id);
|
| + }
|
| +
|
| + static Iterable<Map<String, String>> getStartedProcessesList() =>
|
| + new List.from(startedProcesses.values.map((e) => e.referenceValueMap));
|
| +
|
| + static Future<ServiceExtensionResponse> getStartedProcesses(
|
| + String function, Map<String, String> params) {
|
| + assert(function == 'ext.dart.io.getProcesses');
|
| + var data = {'type': '_startedprocesses', 'data': getStartedProcessesList()};
|
| + var json = JSON.encode(data);
|
| + return new Future.value(new ServiceExtensionResponse.result(json));
|
| + }
|
| +
|
| + static Future<ServiceExtensionResponse> getProcessInfoMapById(
|
| + String function, Map<String, String> params) {
|
| + var id = int.parse(params['id']);
|
| + var result = startedProcesses.containsKey(id)
|
| + ? startedProcesses[id].fullValueMap
|
| + : {};
|
| + var json = JSON.encode(result);
|
| + return new Future.value(new ServiceExtensionResponse.result(json));
|
| + }
|
| +}
|
| +
|
| +class _SocketResourceInfo extends _ReadWriteResourceInfo {
|
| + static const String TCP_STRING = 'TCP';
|
| + static const String UDP_STRING = 'UDP';
|
| + static const String TYPE = '_socket';
|
| +
|
| + final socket;
|
| +
|
| + static Map<int, _SocketResourceInfo> openSockets =
|
| + new Map<int, _SocketResourceInfo>();
|
| +
|
| + _SocketResourceInfo(this.socket) : super(TYPE) {
|
| + SocketOpened(this);
|
| + }
|
| +
|
| + String get name {
|
| + if (socket.isListening) {
|
| + return 'listening:${socket.address.host}:${socket.port}';
|
| + }
|
| + var remote = '';
|
| + try {
|
| + var remoteHost = socket.remoteAddress.host;
|
| + var remotePort = socket.remotePort;
|
| + remote = ' -> $remoteHost:$remotePort';
|
| + } catch (e) { } // ignored if we can't get the information
|
| + return '${socket.address.host}:${socket.port}$remote';
|
| + }
|
| +
|
| + static Iterable<Map<String, String>> getOpenSocketsList() {
|
| + return new List.from(openSockets.values.map((e) => e.referenceValueMap));
|
| + }
|
| +
|
| + Map<String, String> getSocketInfoMap() {
|
| + var result = fullValueMap;
|
| + result['socketType'] = socket.isTcp ? TCP_STRING : UDP_STRING;
|
| + result['listening'] = socket.isListening;
|
| + result['host'] = socket.address.host;
|
| + result['port'] = socket.port;
|
| + if (!socket.isListening) {
|
| + try {
|
| + result['remoteHost'] = socket.remoteAddress.host;
|
| + result['remotePort'] = socket.remotePort;
|
| + } catch (e) {
|
| + // UDP.
|
| + result['remotePort'] = 'NA';
|
| + result['remoteHost'] = 'NA';
|
| + }
|
| + } else {
|
| + result['remotePort'] = 'NA';
|
| + result['remoteHost'] = 'NA';
|
| + }
|
| + result['addressType'] = socket.address.type.name;
|
| + return result;
|
| + }
|
| +
|
| + static Future<ServiceExtensionResponse> getSocketInfoMapByID(
|
| + String function, Map<String, String> params) {
|
| + assert(params.containsKey('id'));
|
| + var id = int.parse(params['id']);
|
| + var result =
|
| + openSockets.containsKey(id) ? openSockets[id].getSocketInfoMap() : {};
|
| + var json = JSON.encode(result);
|
| + return new Future.value(new ServiceExtensionResponse.result(json));
|
| + }
|
| +
|
| + static Future<ServiceExtensionResponse> getOpenSockets(function, params) {
|
| + assert(function == 'ext.dart.io.getOpenSockets');
|
| + var data = {'type': '_opensockets', 'data': getOpenSocketsList()};
|
| + var json = JSON.encode(data);
|
| + return new Future.value(new ServiceExtensionResponse.result(json));
|
| + }
|
| +
|
| + static SocketOpened(_SocketResourceInfo info) {
|
| + assert(!openSockets.containsKey(info.id));
|
| + openSockets[info.id] = info;
|
| + }
|
| +
|
| + static SocketClosed(_SocketResourceInfo info) {
|
| + assert(openSockets.containsKey(info.id));
|
| + openSockets.remove(info.id);
|
| + }
|
| +
|
| +}
|
|
|