diff --git a/node/src/WebRtcServer.ts b/node/src/WebRtcServer.ts index 625addf21a..5ca855f7a4 100644 --- a/node/src/WebRtcServer.ts +++ b/node/src/WebRtcServer.ts @@ -25,7 +25,7 @@ export interface WebRtcServerListenInfo /** * Listening port. */ - port: number; + port?: number; } export type WebRtcServerOptions = diff --git a/node/tests/test-WebRtcServer.js b/node/tests/test-WebRtcServer.js index a66ced3b79..e24c850a29 100644 --- a/node/tests/test-WebRtcServer.js +++ b/node/tests/test-WebRtcServer.js @@ -91,6 +91,81 @@ test('worker.createWebRtcServer() succeeds', async () => expect(worker.webRtcServersForTesting.size).toBe(0); }, 2000); +test('worker.createWebRtcServer() use portrange succeeds', async () => +{ + worker = await createWorker(); + + const onObserverNewWebRtcServer = jest.fn(); + + worker.observer.once('newwebrtcserver', onObserverNewWebRtcServer); + + const webRtcServer = await worker.createWebRtcServer( + { + listenInfos : + [ + { + protocol : 'udp', + ip : '127.0.0.1' + }, + { + protocol : 'tcp', + ip : '127.0.0.1', + announcedIp : '1.2.3.4' + } + ], + appData : { foo: 123 } + }); + + expect(onObserverNewWebRtcServer).toHaveBeenCalledTimes(1); + expect(onObserverNewWebRtcServer).toHaveBeenCalledWith(webRtcServer); + expect(webRtcServer.id).toBeType('string'); + expect(webRtcServer.closed).toBe(false); + expect(webRtcServer.appData).toEqual({ foo: 123 }); + + await expect(worker.dump()) + .resolves + .toEqual( + { + pid : worker.pid, + webRtcServerIds : [ webRtcServer.id ], + routerIds : [], + channelMessageHandlers : + { + channelRequestHandlers : [ webRtcServer.id ], + payloadChannelRequestHandlers : [], + payloadChannelNotificationHandlers : [] + } + }); + + await expect(webRtcServer.dump()) + .resolves + .toMatchObject( + { + id : webRtcServer.id, + udpSockets : + [ + { ip: '127.0.0.1', port: expect.any(Number) } + ], + tcpServers : + [ + { ip: '127.0.0.1', port: expect.any(Number) } + ], + webRtcTransportIds : [], + localIceUsernameFragments : [], + tupleHashes : [] + }); + + // Private API. + expect(worker.webRtcServersForTesting.size).toBe(1); + + worker.close(); + + expect(webRtcServer.closed).toBe(true); + + // Private API. + expect(worker.webRtcServersForTesting.size).toBe(0); +}, 2000); + test('worker.createWebRtcServer() with wrong arguments rejects with TypeError', async () => { worker = await createWorker(); diff --git a/rust/src/router/webrtc_transport/tests.rs b/rust/src/router/webrtc_transport/tests.rs index 9bfd1e2181..9ddf7f66f8 100644 --- a/rust/src/router/webrtc_transport/tests.rs +++ b/rust/src/router/webrtc_transport/tests.rs @@ -58,7 +58,7 @@ fn create_with_webrtc_server_succeeds() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port: port1, + port: Some(port1), }); let listen_infos = listen_infos.insert(WebRtcServerListenInfo { protocol: Protocol::Tcp, @@ -66,7 +66,7 @@ fn create_with_webrtc_server_succeeds() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port: port2, + port: Some(port2), }); WebRtcServerOptions::new(listen_infos) }) @@ -263,7 +263,7 @@ fn webrtc_server_close_event() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port: port1, + port: Some(port1), }); let listen_infos = listen_infos.insert(WebRtcServerListenInfo { protocol: Protocol::Tcp, @@ -271,7 +271,7 @@ fn webrtc_server_close_event() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port: port2, + port: Some(port2), }); WebRtcServerOptions::new(listen_infos) }) diff --git a/rust/src/webrtc_server.rs b/rust/src/webrtc_server.rs index 41d9293821..7d70430e0d 100644 --- a/rust/src/webrtc_server.rs +++ b/rust/src/webrtc_server.rs @@ -83,7 +83,8 @@ pub struct WebRtcServerListenInfo { #[serde(flatten)] pub listen_ip: ListenIp, /// Listening port. - pub port: u16, + #[serde(skip_serializing_if = "Option::is_none")] + pub port: Option, } /// Struct that protects an invariant of having non-empty list of listen infos. diff --git a/rust/src/webrtc_server/tests.rs b/rust/src/webrtc_server/tests.rs index e452bcd20c..a87f17d653 100644 --- a/rust/src/webrtc_server/tests.rs +++ b/rust/src/webrtc_server/tests.rs @@ -39,7 +39,7 @@ fn worker_close_event() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port, + port: Some(port), }, ))) .await diff --git a/rust/tests/integration/webrtc_server.rs b/rust/tests/integration/webrtc_server.rs index 58c6c01957..fec1c39992 100644 --- a/rust/tests/integration/webrtc_server.rs +++ b/rust/tests/integration/webrtc_server.rs @@ -67,7 +67,7 @@ fn create_webrtc_server_succeeds() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port: port1, + port: Some(port1), }); let listen_infos = listen_infos.insert(WebRtcServerListenInfo { protocol: Protocol::Tcp, @@ -75,7 +75,7 @@ fn create_webrtc_server_succeeds() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: Some(IpAddr::V4(Ipv4Addr::new(1, 2, 3, 4))), }, - port: port2, + port: Some(port2), }); let mut webrtc_server_options = WebRtcServerOptions::new(listen_infos); @@ -132,6 +132,83 @@ fn create_webrtc_server_succeeds() { }); } +#[test] +fn create_webrtc_server_use_portrange_succeeds() { + future::block_on(async move { + let (worker1, _worker2) = init().await; + + let new_webrtc_server_count = Arc::new(AtomicUsize::new(0)); + + worker1 + .on_new_webrtc_server({ + let new_webrtc_server_count = Arc::clone(&new_webrtc_server_count); + + move |_webrtc_server| { + new_webrtc_server_count.fetch_add(1, Ordering::SeqCst); + } + }) + .detach(); + + #[derive(Debug, PartialEq)] + struct CustomAppData { + foo: u32, + } + + let webrtc_server = worker1 + .create_webrtc_server({ + let listen_infos = WebRtcServerListenInfos::new(WebRtcServerListenInfo { + protocol: Protocol::Udp, + listen_ip: ListenIp { + ip: IpAddr::V4(Ipv4Addr::LOCALHOST), + announced_ip: None, + }, + port: None, + }); + let listen_infos = listen_infos.insert(WebRtcServerListenInfo { + protocol: Protocol::Tcp, + listen_ip: ListenIp { + ip: IpAddr::V4(Ipv4Addr::LOCALHOST), + announced_ip: Some(IpAddr::V4(Ipv4Addr::new(1, 2, 3, 4))), + }, + port: None, + }); + let mut webrtc_server_options = WebRtcServerOptions::new(listen_infos); + + webrtc_server_options.app_data = AppData::new(CustomAppData { foo: 123 }); + + webrtc_server_options + }) + .await + .expect("Failed to create router"); + + assert_eq!(new_webrtc_server_count.load(Ordering::SeqCst), 1); + assert!(!webrtc_server.closed()); + assert_eq!( + webrtc_server.app_data().downcast_ref::(), + Some(&CustomAppData { foo: 123 }), + ); + + let worker_dump = worker1.dump().await.expect("Failed to dump worker"); + + assert_eq!(worker_dump.router_ids, vec![]); + assert_eq!(worker_dump.webrtc_server_ids, vec![webrtc_server.id()]); + + let dump = webrtc_server + .dump() + .await + .expect("Failed to dump WebRTC server"); + + assert_eq!(dump.id, webrtc_server.id()); + + assert_eq!(dump.udp_sockets[0].ip, IpAddr::V4(Ipv4Addr::LOCALHOST)); + assert_eq!(dump.tcp_servers[0].ip, IpAddr::V4(Ipv4Addr::LOCALHOST)); + + assert_eq!(dump.webrtc_transport_ids, HashedSet::default()); + assert_eq!(dump.local_ice_username_fragments, vec![]); + assert_eq!(dump.tuple_hashes, vec![]); + }); +} + #[test] fn unavailable_infos_fails() { future::block_on(async move { @@ -166,7 +243,7 @@ fn unavailable_infos_fails() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port: port1, + port: Some(port1), }); let listen_infos = listen_infos.insert(WebRtcServerListenInfo { protocol: Protocol::Udp, @@ -174,7 +251,7 @@ fn unavailable_infos_fails() { ip: IpAddr::V4(Ipv4Addr::new(1, 2, 3, 4)), announced_ip: None, }, - port: port2, + port: Some(port2), }); WebRtcServerOptions::new(listen_infos) @@ -197,7 +274,7 @@ fn unavailable_infos_fails() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port: port1, + port: Some(port1), }); let listen_infos = listen_infos.insert(WebRtcServerListenInfo { protocol: Protocol::Udp, @@ -205,7 +282,7 @@ fn unavailable_infos_fails() { ip: IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), announced_ip: Some(IpAddr::V4(Ipv4Addr::new(1, 2, 3, 4))), }, - port: port1, + port: Some(port1), }); WebRtcServerOptions::new(listen_infos) @@ -228,7 +305,7 @@ fn unavailable_infos_fails() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port: port1, + port: Some(port1), }, ))) .await @@ -242,7 +319,7 @@ fn unavailable_infos_fails() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port: port1, + port: Some(port1), }, ))) .await; @@ -270,7 +347,7 @@ fn close_event() { ip: IpAddr::V4(Ipv4Addr::LOCALHOST), announced_ip: None, }, - port, + port: Some(port), }, ))) .await diff --git a/worker/src/RTC/WebRtcServer.cpp b/worker/src/RTC/WebRtcServer.cpp index d2bb3ab4f8..91ba00c8e8 100644 --- a/worker/src/RTC/WebRtcServer.cpp +++ b/worker/src/RTC/WebRtcServer.cpp @@ -94,14 +94,18 @@ namespace RTC listenInfo.announcedIp.assign(jsonAnnouncedIpIt->get()); } + uint16_t port{ 0 }; auto jsonPortIt = jsonListenInfo.find("port"); - if (jsonPortIt == jsonListenInfo.end()) - MS_THROW_TYPE_ERROR("missing listenInfo.port"); - else if (!(jsonPortIt->is_number() && Utils::Json::IsPositiveInteger(*jsonPortIt))) - MS_THROW_TYPE_ERROR("wrong listenInfo.port (not a positive number)"); + if (jsonPortIt != jsonListenInfo.end()) + { + if (!(jsonPortIt->is_number() && Utils::Json::IsPositiveInteger(*jsonPortIt))) + MS_THROW_TYPE_ERROR("wrong port (not a positive number)"); + + port = jsonPortIt->get(); + } - listenInfo.port = jsonPortIt->get(); + listenInfo.port = port; } try @@ -111,14 +115,24 @@ namespace RTC if (listenInfo.protocol == RTC::TransportTuple::Protocol::UDP) { // This may throw. - auto* udpSocket = new RTC::UdpSocket(this, listenInfo.ip, listenInfo.port); + RTC::UdpSocket* udpSocket; + + if (listenInfo.port != 0) + udpSocket = new RTC::UdpSocket(this, listenInfo.ip, listenInfo.port); + else + udpSocket = new RTC::UdpSocket(this, listenInfo.ip); this->udpSocketOrTcpServers.emplace_back(udpSocket, nullptr, listenInfo.announcedIp); } else if (listenInfo.protocol == RTC::TransportTuple::Protocol::TCP) { // This may throw. - auto* tcpServer = new RTC::TcpServer(this, this, listenInfo.ip, listenInfo.port); + RTC::TcpServer* tcpServer; + + if (listenInfo.port != 0) + tcpServer = new RTC::TcpServer(this, this, listenInfo.ip, listenInfo.port); + else + tcpServer = new RTC::TcpServer(this, this, listenInfo.ip); this->udpSocketOrTcpServers.emplace_back(nullptr, tcpServer, listenInfo.announcedIp); }