From 5bca23615f37bc1f93ee43e323dd9b0525003a4f Mon Sep 17 00:00:00 2001 From: Alexandre Jousset Date: Thu, 18 Apr 2024 17:41:55 +0300 Subject: [PATCH] XEP-0215: management of the field `expires` [2] Fix following comment and recommandations: - Use DateTime? type instead of int64? - check the computed delay value against sane values - use a HashMap to keep track of timers and cancel them on connection closed - add equals and hash funcs to XmppStream to use with the HashMap - rename the callback to reflect its meaning - test the TURN server conf before STUN default server --- plugins/ice/src/plugin.vala | 67 +++++++++++++++---- xmpp-vala/src/core/xmpp_stream.vala | 8 +++ .../xep/0215_external_service_discovery.vala | 4 +- 3 files changed, 64 insertions(+), 15 deletions(-) diff --git a/plugins/ice/src/plugin.vala b/plugins/ice/src/plugin.vala index 74295ce5e..aa4b6d418 100644 --- a/plugins/ice/src/plugin.vala +++ b/plugins/ice/src/plugin.vala @@ -6,6 +6,21 @@ using Xmpp.Xep; private extern const size_t NICE_ADDRESS_STRING_LEN; public class Dino.Plugins.Ice.Plugin : RootInterface, Object { + private const int64 delay_min = 300; // 10mn + private const int64 delay_max = (int64) uint.MAX; + + private class TimerPayload { + public Account account { get; set; } + public uint timeout_handle_id; + + public TimerPayload(Account account, uint timeout_handle_id) { + this.account = account; + this.timeout_handle_id = timeout_handle_id; + } + } + + private HashMap timeouts = new HashMap(XmppStream.hash_func, XmppStream.equals_func); + public Dino.Application app; public void registered(Dino.Application app) { @@ -22,10 +37,11 @@ public class Dino.Plugins.Ice.Plugin : RootInterface, Object { stream.get_module(JingleRawUdp.Module.IDENTITY).set_local_ip_address_handler(get_local_ip_addresses); } }); - app.stream_interactor.stream_negotiated.connect(on_stream_negotiated); + app.stream_interactor.stream_negotiated.connect(external_discovery_refresh_services); + app.stream_interactor.connection_manager.connection_state_changed.connect(on_connection_state_changed); } - private async void on_stream_negotiated(Account account, XmppStream stream) { + private async void external_discovery_refresh_services(Account account, XmppStream stream) { Module? ice_udp_module = stream.get_module(JingleIceUdp.Module.IDENTITY) as Module; if (ice_udp_module == null) return; Gee.List services = yield ExternalServiceDiscovery.request_services(stream); @@ -45,6 +61,28 @@ public class Dino.Plugins.Ice.Plugin : RootInterface, Object { } } } + + if (ice_udp_module.turn_service != null) { + DateTime? expires = ice_udp_module.turn_service.expires; + if (expires != null) { + int64 delay = (expires.to_unix() - new DateTime.now_utc().to_unix()) / 2; + + if (delay >= delay_min && delay <= delay_max) { + debug("Next server external service discovery in %lds (because of TURN credentials' expiry time)", (long) delay); + + uint timeout_handle_id = Timeout.add_seconds((uint) delay, () => { + on_timeout(stream); + return false; + }); + timeouts[stream] = new TimerPayload(account, timeout_handle_id); + timeouts[stream].account = account; + timeouts[stream].timeout_handle_id = timeout_handle_id; + } else { + warning("Bogus TURN credentials' expiry time (delay value = %ld), *not* planning next service discovery", (long) delay); + } + } + } + if (ice_udp_module.stun_ip == null) { InetAddress ip = yield lookup_ipv4_addess("stun.dino.im"); if (ip == null) return; @@ -54,20 +92,23 @@ public class Dino.Plugins.Ice.Plugin : RootInterface, Object { ice_udp_module.stun_ip = ip.to_string(); ice_udp_module.stun_port = 7886; } + } - if (ice_udp_module.turn_service != null) { - int64? expires = ice_udp_module.turn_service.expires; - if (expires != null) { - uint delay = (uint) (expires - new DateTime.now_utc().to_unix()) / 2; + public void on_timeout(XmppStream stream) { + if (!timeouts.has_key(stream)) return; + TimerPayload pl = timeouts[stream]; + timeouts.unset(stream); + external_discovery_refresh_services.begin(pl.account, stream); + } - debug("Next server external service discovery in %us (because of TURN credentials' expiring time)", delay); + public void on_connection_state_changed(Account account, ConnectionManager.ConnectionState state) { + if (state == ConnectionManager.ConnectionState.DISCONNECTED) { + XmppStream? stream = app.stream_interactor.connection_manager.get_stream(account); + if (stream == null) return; + if (!timeouts.has_key(stream)) return; - Timeout.add_seconds(delay, () => { - if (app.stream_interactor.connection_manager.get_state(account) != ConnectionManager.ConnectionState.CONNECTED) return false; - on_stream_negotiated.begin(account, stream); - return false; - }); - } + Source.remove(timeouts[stream].timeout_handle_id); + timeouts.unset(stream); } } diff --git a/xmpp-vala/src/core/xmpp_stream.vala b/xmpp-vala/src/core/xmpp_stream.vala index 54433b674..0b40a193e 100644 --- a/xmpp-vala/src/core/xmpp_stream.vala +++ b/xmpp-vala/src/core/xmpp_stream.vala @@ -191,4 +191,12 @@ public abstract class Xmpp.XmppStream : Object { } } } + + public static bool equals_func(XmppStream str1, XmppStream str2) { + return str1.remote_name.to_string() == str2.remote_name.to_string(); + } + + public static uint hash_func(XmppStream str) { + return str.remote_name.to_string().hash(); + } } \ No newline at end of file diff --git a/xmpp-vala/src/module/xep/0215_external_service_discovery.vala b/xmpp-vala/src/module/xep/0215_external_service_discovery.vala index 3c80ed9f3..6d41a2888 100644 --- a/xmpp-vala/src/module/xep/0215_external_service_discovery.vala +++ b/xmpp-vala/src/module/xep/0215_external_service_discovery.vala @@ -26,7 +26,7 @@ namespace Xmpp.Xep.ExternalServiceDiscovery { service.username = service_node.get_attribute("username", NS_URI); service.password = service_node.get_attribute("password", NS_URI); string? expires_str = service_node.get_attribute("expires", NS_URI); - if (expires_str != null) service.expires = DateTimeProfiles.parse_string(expires_str).to_unix(); + if (expires_str != null) service.expires = DateTimeProfiles.parse_string(expires_str); service.transport = service_node.get_attribute("transport", NS_URI); service.name = service_node.get_attribute("name", NS_URI); @@ -44,7 +44,7 @@ namespace Xmpp.Xep.ExternalServiceDiscovery { public string username { get; set; } public string password { get; set; } - public int64? expires { get; set; } + public DateTime? expires { get; set; } public string transport { get; set; } public string name { get; set; }