| // Copyright (c) 2024, 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:io'; |
| |
| import 'package:dtd/dtd.dart' show RpcErrorCodes; |
| import 'package:json_rpc_2/json_rpc_2.dart'; |
| import 'package:test/test.dart'; |
| import 'package:web_socket_channel/web_socket_channel.dart'; |
| import 'package:dtd_impl/dart_tooling_daemon.dart'; |
| |
| void main() { |
| late Peer client; |
| late DartToolingDaemon? dtd; |
| late String uri; |
| |
| tearDown(() async { |
| await dtd?.close(); |
| }); |
| |
| test( |
| 'forbids connections where the uri token is not the first element in the path', |
| () async { |
| dtd = await DartToolingDaemon.startService([]); |
| |
| expect( |
| () async => await WebSocket.connect( |
| dtd!.uri!.replace(path: 'someInvalidToken').toString(), // invalid token |
| ), |
| throwsA( |
| predicate( |
| (p0) => |
| p0 is WebSocketException && |
| RegExp("^Connection to '.*' was not upgraded to websocket\$") |
| .hasMatch(p0.message), |
| ), |
| ), |
| ); |
| |
| expect( |
| () async => await WebSocket.connect( |
| dtd!.uri!.replace(path: '').toString(), // no token |
| ), |
| throwsA( |
| predicate( |
| (p0) => |
| p0 is WebSocketException && |
| RegExp("^Connection to '.*' was not upgraded to websocket\$") |
| .hasMatch(p0.message), |
| ), |
| ), |
| ); |
| }); |
| |
| group('dtd', () { |
| setUp(() async { |
| dtd = await DartToolingDaemon.startService([]); |
| |
| // Wait for server to start and print to the port to stdout. |
| uri = dtd!.uri!.toString(); |
| |
| client = _createClient(uri); |
| }); |
| |
| tearDown(() async { |
| await client.close(); |
| }); |
| |
| group('streams', () { |
| final streamId = 'testStream'; |
| final eventKind = 'test'; |
| final eventData = {'the': 'data'}; |
| |
| test('basics', () async { |
| var completer = Completer(); |
| client.registerMethod('streamNotify', (Parameters parameters) { |
| completer.complete(parameters.asMap); |
| }); |
| final listenResult = await client.sendRequest('streamListen', { |
| "streamId": streamId, |
| }); |
| |
| expect(listenResult, {"type": "Success"}); |
| |
| final postResult = await client.sendRequest( |
| 'postEvent', |
| { |
| 'streamId': streamId, |
| 'eventKind': eventKind, |
| 'eventData': eventData, |
| }, |
| ); |
| expect(postResult, {"type": "Success"}); |
| |
| final dataFromTheStream = await completer.future; |
| expect(dataFromTheStream, { |
| "streamId": streamId, |
| "eventKind": eventKind, |
| "eventData": eventData, |
| "timestamp": anything, |
| }); |
| |
| // Now cancel the stream |
| completer = Completer(); // Reset the completer |
| final cancelResult = await client.sendRequest( |
| 'streamCancel', |
| { |
| 'streamId': streamId, |
| }, |
| ); |
| expect(cancelResult, {"type": "Success"}); |
| final postResult2 = await client.sendRequest( |
| 'postEvent', |
| { |
| 'streamId': streamId, |
| 'eventKind': eventKind, |
| 'eventData': eventData, |
| }, |
| ); |
| expect(postResult2, {"type": "Success"}); |
| expect( |
| completer.future.timeout( |
| const Duration(seconds: 1), |
| onTimeout: () => throw TimeoutException('Timed out'), |
| ), |
| throwsA(predicate((p0) => p0 is TimeoutException)), |
| ); |
| }); |
| |
| test('streamListen the same stream', () async { |
| final listenResult = await client.sendRequest('streamListen', { |
| "streamId": streamId, |
| }); |
| |
| expect(listenResult, {"type": "Success"}); |
| |
| expect( |
| () => client.sendRequest('streamListen', { |
| "streamId": streamId, |
| }), |
| throwsA( |
| predicate( |
| (e) => |
| e is RpcException && |
| e.code == RpcErrorCodes.kStreamAlreadySubscribed, |
| ), |
| ), |
| ); |
| }); |
| |
| test('stop listening to a stream that is not being listened to', () { |
| expect( |
| () => client.sendRequest('streamCancel', { |
| "streamId": streamId, |
| }), |
| throwsA( |
| predicate( |
| (e) => |
| e is RpcException && |
| e.code == RpcErrorCodes.kStreamNotSubscribed, |
| ), |
| ), |
| ); |
| }); |
| |
| test('postEvent when there are no listeners', () async { |
| final postResult = await client.sendRequest( |
| 'postEvent', |
| { |
| 'streamId': streamId, |
| 'eventKind': eventKind, |
| 'eventData': eventData, |
| }, |
| ); |
| expect(postResult, {"type": "Success"}); |
| }); |
| }); |
| |
| group('service methods', () { |
| final service1 = 'foo1'; |
| final method1 = 'bar1'; |
| final method2 = 'bar2'; |
| final data1 = {"data": 1}; |
| final response1 = {"response": 1}; |
| |
| test('basics', () async { |
| client.registerMethod('$service1.$method1', (Parameters parameters) { |
| return response1; |
| }); |
| final registerResult = await client.sendRequest('registerService', { |
| "service": service1, |
| "method": method1, |
| }); |
| |
| expect(registerResult, {"type": "Success"}); |
| |
| final register2Result = await client.sendRequest('registerService', { |
| "service": service1, |
| "method": method2, |
| }); |
| expect(register2Result, {"type": "Success"}); |
| |
| final methodResponse = await client.sendRequest( |
| '$service1.$method1', |
| data1, |
| ); |
| expect(methodResponse, response1); |
| }); |
| |
| test('registering a service method that already exists', () async { |
| final registerResult = await client.sendRequest('registerService', { |
| "service": service1, |
| "method": method1, |
| }); |
| |
| expect(registerResult, {"type": "Success"}); |
| expect( |
| () => client.sendRequest('registerService', { |
| "service": service1, |
| "method": method1, |
| }), |
| throwsA( |
| predicate( |
| (p0) => |
| p0 is RpcException && |
| p0.code == RpcErrorCodes.kServiceMethodAlreadyRegistered, |
| ), |
| ), |
| ); |
| }); |
| |
| test('calling a method that does not exist', () { |
| expect( |
| () => client.sendRequest('zoo.abc', {}), |
| throwsA( |
| predicate( |
| (p0) => |
| p0 is RpcException && |
| p0.code == RpcException.methodNotFound('zoo.abc').code, |
| ), |
| ), |
| ); |
| }); |
| |
| test('different clients cannot register the same service', () async { |
| final client2 = _createClient(uri); |
| final registerResult = await client.sendRequest('registerService', { |
| "service": service1, |
| "method": method1, |
| }); |
| expect(registerResult, {"type": "Success"}); |
| |
| expect( |
| () => client2.sendRequest('registerService', { |
| "service": service1, |
| "method": method2, |
| }), |
| throwsA( |
| predicate( |
| (p0) => |
| p0 is RpcException && |
| p0.code == RpcErrorCodes.kServiceAlreadyRegistered, |
| ), |
| ), |
| ); |
| }); |
| |
| test('releases service methods on disconnect', () async { |
| final client2 = _createClient(uri); |
| final registerResult = await client.sendRequest('registerService', { |
| "service": service1, |
| "method": method1, |
| }); |
| expect(registerResult, {"type": "Success"}); |
| |
| await client.close(); |
| |
| // TODO: replace this polling when notification streams are implemented. |
| |
| dynamic client2RegisterResult; |
| for (var i = 0; i < 10; i++) { |
| try { |
| // The service method registration should succeed once the other |
| // finishes closing. |
| client2RegisterResult = |
| await client2.sendRequest('registerService', { |
| "service": service1, |
| "method": method1, |
| }); |
| break; |
| } catch (_) {} |
| await Future.delayed(Duration(seconds: 1)); |
| } |
| expect(client2RegisterResult, {"type": "Success"}); |
| }); |
| }); |
| }); |
| |
| group('dtd arguments', () { |
| test('allow explicit port', () async { |
| const testPort = 8123; |
| dtd = await DartToolingDaemon.startService(['--port=$testPort']); |
| uri = dtd!.uri!.toString(); |
| expect(Uri.parse(uri).port, testPort); |
| }); |
| }); |
| } |
| |
| Peer _createClient(String uri) { |
| final channel = WebSocketChannel.connect( |
| Uri.parse(uri), |
| ); |
| |
| final client = Peer(channel.cast()); |
| unawaited(client.listen()); |
| return client; |
| } |