2023-12-29 23:28:57 +00:00
// Copyright (c) 2023 xmpp-rs contributors.
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
2023-12-31 06:17:29 +00:00
use tokio_xmpp ::connect ::ServerConnector ;
2023-12-29 23:28:57 +00:00
use tokio_xmpp ::{
2024-07-24 18:52:10 +00:00
jid ::BareJid ,
2023-12-29 23:28:57 +00:00
parsers ::{
muc ::Muc ,
presence ::{ Presence , Type as PresenceType } ,
} ,
} ;
2024-12-16 17:43:05 +00:00
use crate ::Agent ;
2023-12-29 23:28:57 +00:00
2023-12-31 06:17:29 +00:00
pub async fn join_room < C : ServerConnector > (
agent : & mut Agent < C > ,
2023-12-29 23:28:57 +00:00
room : BareJid ,
nick : Option < String > ,
password : Option < String > ,
lang : & str ,
status : & str ,
) {
2024-12-16 17:43:05 +00:00
if agent . rooms_joining . contains_key ( & room ) {
// We are already joining
warn! ( " Requesting to join again room {room} which is already joining... " ) ;
return ;
}
if ! agent . rooms_joined . contains_key ( & room ) {
// We are already joined, cannot join
warn! ( " Requesting to join room {room} which is already joined... " ) ;
return ;
}
2023-12-29 23:28:57 +00:00
let mut muc = Muc ::new ( ) ;
if let Some ( password ) = password {
muc = muc . with_password ( password ) ;
}
2024-08-06 14:04:04 +00:00
let nick = if let Some ( nick ) = nick {
nick
} else {
agent . default_nick . read ( ) . await . clone ( )
} ;
2023-12-29 23:28:57 +00:00
let room_jid = room . with_resource_str ( & nick ) . unwrap ( ) ;
let mut presence = Presence ::new ( PresenceType ::None ) . with_to ( room_jid ) ;
presence . add_payload ( muc ) ;
presence . set_status ( String ::from ( lang ) , String ::from ( status ) ) ;
let _ = agent . client . send_stanza ( presence . into ( ) ) . await ;
2024-12-16 17:43:05 +00:00
agent . rooms_joining . insert ( room , nick ) ;
2023-12-29 23:28:57 +00:00
}
/// Send a "leave room" request to the server (specifically, an "unavailable" presence stanza).
///
/// The returned future will resolve when the request has been sent,
/// not when the room has actually been left.
///
2024-01-27 17:18:58 +00:00
/// If successful, a `RoomLeft` event should be received later as a confirmation. See [XEP-0045](https://xmpp.org/extensions/xep-0045.html#exit).
2023-12-29 23:28:57 +00:00
///
/// Note that this method does NOT remove the room from the auto-join list; the latter
/// is more a list of bookmarks that the account knows about and that have a flag set
/// to indicate that they should be joined automatically after connecting (see the JoinRoom event).
///
2024-01-27 17:18:58 +00:00
/// Regarding the latter, see the these [ModernXMPP minutes about auto-join behavior](https://docs.modernxmpp.org/meetings/2019-01-brussels/#bookmarks).
2023-12-29 23:28:57 +00:00
///
/// # Arguments
///
/// * `room_jid`: The JID of the room to leave.
/// * `lang`: The language of the status message.
/// * `status`: The status message to send.
2023-12-31 06:17:29 +00:00
pub async fn leave_room < C : ServerConnector > (
agent : & mut Agent < C > ,
2024-12-16 17:43:05 +00:00
room : BareJid ,
2023-12-29 23:28:57 +00:00
lang : impl Into < String > ,
status : impl Into < String > ,
) {
2024-12-16 17:43:05 +00:00
if agent . rooms_leaving . contains_key ( & room ) {
// We are already leaving
warn! ( " Requesting to leave again room {room} which is already leaving... " ) ;
return ;
}
if ! agent . rooms_joined . contains_key ( & room ) {
// We are not joined, cannot leave
warn! ( " Requesting to leave room {room} which is not joined... " ) ;
return ;
}
// Get currently-used nickname
let nickname = agent . rooms_joined . get ( & room ) . unwrap ( ) ;
2023-12-29 23:28:57 +00:00
// XEP-0045 specifies that, to leave a room, the client must send a presence stanza
// with type="unavailable".
let mut presence = Presence ::new ( PresenceType ::Unavailable ) . with_to (
2024-12-16 17:43:05 +00:00
room . with_resource_str ( nickname )
2023-12-29 23:28:57 +00:00
. expect ( " Invalid room JID after adding resource part. " ) ,
) ;
// Optionally, the client may include a status message in the presence stanza.
// TODO: Should this be optional? The XEP says "MAY", but the method signature requires the arguments.
// XEP-0045: "The occupant MAY include normal <status/> information in the unavailable presence stanzas"
presence . set_status ( lang , status ) ;
// Send the presence stanza.
if let Err ( e ) = agent . client . send_stanza ( presence . into ( ) ) . await {
// Report any errors to the log.
error! ( " Failed to send leave room presence: {} " , e ) ;
}
2024-12-16 17:43:05 +00:00
agent . rooms_leaving . insert ( room , nickname . to_string ( ) ) ;
2023-12-29 23:28:57 +00:00
}