Add partial support for XEP-0249 Direct MUC Invitations.

Exposes a method for a MUC to invite a user to the room, and adds a listener to the MultiUserChat listener to inform users of direct invitations they have received.

Fixes SMACK-932.
This commit is contained in:
Martin Fidczuk 2022-08-01 17:06:14 +01:00 committed by Martin Fidczuk
parent f6c85d9fb3
commit 9e9c233468
No known key found for this signature in database
GPG Key ID: 8B533F18DF43D922
5 changed files with 481 additions and 0 deletions

View File

@ -74,6 +74,7 @@ import org.jivesoftware.smackx.muc.MultiUserChatException.MucNotJoinedException;
import org.jivesoftware.smackx.muc.MultiUserChatException.NotAMucServiceException;
import org.jivesoftware.smackx.muc.filter.MUCUserStatusCodeFilter;
import org.jivesoftware.smackx.muc.packet.Destroy;
import org.jivesoftware.smackx.muc.packet.GroupChatInvitation;
import org.jivesoftware.smackx.muc.packet.MUCAdmin;
import org.jivesoftware.smackx.muc.packet.MUCInitialPresence;
import org.jivesoftware.smackx.muc.packet.MUCItem;
@ -1063,6 +1064,51 @@ public class MultiUserChat {
connection.sendStanza(message);
}
/**
* Invites another user to the room in which one is an occupant. In contrast
* to the method "invite", the invitation is sent directly to the user rather
* than via the chat room. This is useful when the user being invited is
* offline, as otherwise the invitation would be dropped.
*
* @param address the user to send the invitation to
* @throws NotConnectedException if the XMPP connection is not connected.
* @throws InterruptedException if the calling thread was interrupted.
*/
public void inviteDirectly(EntityBareJid address) throws NotConnectedException, InterruptedException {
inviteDirectly(address, null, null, false, null);
}
/**
* Invites another user to the room in which one is an occupant. In contrast
* to the method "invite", the invitation is sent directly to the user rather
* than via the chat room. This is useful when the user being invited is
* offline, as otherwise the invitation would be dropped.
*
* @param address the user to send the invitation to
* @param reason the purpose for the invitation
* @param password specifies a password needed for entry
* @param continueAsOneToOneChat specifies if the groupchat room continues a one-to-one chat having the designated thread
* @param thread the thread to continue
* @throws NotConnectedException if the XMPP connection is not connected.
* @throws InterruptedException if the calling thread was interrupted.
*/
public void inviteDirectly(EntityBareJid address, String reason, String password, boolean continueAsOneToOneChat, String thread)
throws NotConnectedException, InterruptedException {
// Add the extension for direct invitation
GroupChatInvitation invitationExt = new GroupChatInvitation(room,
reason,
password,
continueAsOneToOneChat,
thread);
Message message = connection.getStanzaFactory().buildMessageStanza()
.to(address)
.addExtension(invitationExt)
.build();
connection.sendStanza(message);
}
/**
* Adds a listener to invitation rejections notifications. The listener will be fired anytime
* an invitation is declined.

View File

@ -40,6 +40,7 @@ import org.jivesoftware.smack.XMPPConnection;
import org.jivesoftware.smack.XMPPConnectionRegistry;
import org.jivesoftware.smack.XMPPException.XMPPErrorException;
import org.jivesoftware.smack.filter.AndFilter;
import org.jivesoftware.smack.filter.ExtensionElementFilter;
import org.jivesoftware.smack.filter.MessageTypeFilter;
import org.jivesoftware.smack.filter.NotFilter;
import org.jivesoftware.smack.filter.StanzaExtensionFilter;
@ -57,6 +58,7 @@ import org.jivesoftware.smackx.disco.packet.DiscoverInfo;
import org.jivesoftware.smackx.disco.packet.DiscoverItems;
import org.jivesoftware.smackx.muc.MultiUserChatException.MucNotJoinedException;
import org.jivesoftware.smackx.muc.MultiUserChatException.NotAMucServiceException;
import org.jivesoftware.smackx.muc.packet.GroupChatInvitation;
import org.jivesoftware.smackx.muc.packet.MUCInitialPresence;
import org.jivesoftware.smackx.muc.packet.MUCUser;
@ -139,6 +141,11 @@ public final class MultiUserChatManager extends Manager {
private static final StanzaFilter INVITATION_FILTER = new AndFilter(StanzaTypeFilter.MESSAGE, new StanzaExtensionFilter(new MUCUser()),
new NotFilter(MessageTypeFilter.ERROR));
private static final StanzaFilter DIRECT_INVITATION_FILTER =
new AndFilter(StanzaTypeFilter.MESSAGE,
new ExtensionElementFilter<GroupChatInvitation>(GroupChatInvitation.class),
new NotFilter(MessageTypeFilter.ERROR));
private static final ExpirationCache<DomainBareJid, DiscoverInfo> KNOWN_MUC_SERVICES = new ExpirationCache<>(
100, 1000 * 60 * 60 * 24);
@ -199,6 +206,33 @@ public final class MultiUserChatManager extends Manager {
};
connection.addAsyncStanzaListener(invitationPacketListener, INVITATION_FILTER);
// Listens for all messages that include an XEP-0249 GroupChatInvitation extension and fire the invitation
// listeners
StanzaListener directInvitationStanzaListener = new StanzaListener() {
@Override
public void processStanza(Stanza stanza) {
final Message message = (Message) stanza;
GroupChatInvitation invite =
stanza.getExtension(GroupChatInvitation.class);
// Fire event for invitation listeners
final MultiUserChat muc = getMultiUserChat(invite.getRoomAddress());
final XMPPConnection connection = connection();
final EntityJid from = message.getFrom().asEntityJidIfPossible();
if (from == null) {
LOGGER.warning("Group Chat Invitation from non entity JID in '" + message + "'");
return;
}
final String reason = invite.getReason();
final String password = invite.getPassword();
final MUCUser.Invite mucInvite = new MUCUser.Invite(reason, from, connection.getUser().asEntityBareJid());
for (final InvitationListener listener : invitationsListeners) {
listener.invitationReceived(connection, muc, from, reason, password, message, mucInvite);
}
}
};
connection.addAsyncStanzaListener(directInvitationStanzaListener, DIRECT_INVITATION_FILTER);
connection.addConnectionListener(new ConnectionListener() {
@Override
public void authenticated(XMPPConnection connection, boolean resumed) {

View File

@ -69,6 +69,10 @@ public class GroupChatInvitation implements ExtensionElement {
public static final QName QNAME = new QName(NAMESPACE, ELEMENT);
private final EntityBareJid roomAddress;
private final String reason;
private final String password;
private final String thread;
private final boolean continueAsOneToOneChat;
/**
* Creates a new group chat invitation to the specified room address.
@ -79,7 +83,67 @@ public class GroupChatInvitation implements ExtensionElement {
* @param roomAddress the address of the group chat room.
*/
public GroupChatInvitation(EntityBareJid roomAddress) {
this(roomAddress, null, null, false, null);
}
/**
* Creates a new group chat invitation to the specified room address.
* GroupChat room addresses are in the form <code>room@service</code>,
* where <code>service</code> is the name of group chat server, such as
* <code>chat.example.com</code>.
*
* @param roomAddress the address of the group chat room.
* @param reason the purpose for the invitation
* @param password specifies a password needed for entry
* @param continueAsOneToOneChat specifies if the groupchat room continues a one-to-one chat having the designated thread
* @param thread the thread to continue
*/
public GroupChatInvitation(EntityBareJid roomAddress,
String reason,
String password,
boolean continueAsOneToOneChat,
String thread) {
this.roomAddress = Objects.requireNonNull(roomAddress);
this.reason = reason;
this.password = password;
this.continueAsOneToOneChat = continueAsOneToOneChat;
this.thread = thread;
}
/**
* Returns the purpose for the invitation.
*
* @return the address of the group chat room.
*/
public String getReason() {
return reason;
}
/**
* Returns the password needed for entry.
*
* @return the password needed for entry
*/
public String getPassword() {
return password;
}
/**
* Returns the thread to continue.
*
* @return the thread to continue.
*/
public String getThread() {
return thread;
}
/**
* Returns whether the groupchat room continues a one-to-one chat.
*
* @return whether the groupchat room continues a one-to-one chat.
*/
public boolean continueAsOneToOneChat() {
return continueAsOneToOneChat;
}
/**
@ -107,6 +171,13 @@ public class GroupChatInvitation implements ExtensionElement {
public XmlStringBuilder toXML(org.jivesoftware.smack.packet.XmlEnvironment enclosingNamespace) {
XmlStringBuilder xml = new XmlStringBuilder(this);
xml.attribute("jid", getRoomAddress());
xml.optAttribute("reason", getReason());
xml.optAttribute("password", getPassword());
xml.optAttribute("thread", getThread());
if (continueAsOneToOneChat())
xml.optBooleanAttribute("continue", true);
xml.closeEmptyElement();
return xml;
}

View File

@ -0,0 +1,324 @@
/**
*
* Copyright 2021-2022 Microsoft Corporation.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jivesoftware.smackx.muc;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertNull;
import static org.junit.jupiter.api.Assertions.assertSame;
import static org.junit.jupiter.api.Assertions.assertTrue;
import java.util.concurrent.atomic.AtomicBoolean;
import org.jivesoftware.smack.DummyConnection;
import org.jivesoftware.smack.XMPPConnection;
import org.jivesoftware.smack.packet.Message;
import org.jivesoftware.smack.packet.Stanza;
import org.jivesoftware.smack.test.util.SmackTestSuite;
import org.jivesoftware.smack.test.util.WaitForPacketListener;
import org.jivesoftware.smackx.muc.packet.GroupChatInvitation;
import org.jivesoftware.smackx.muc.packet.MUCUser;
import org.jivesoftware.smackx.muc.packet.MUCUser.Invite;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.jxmpp.jid.EntityBareJid;
import org.jxmpp.jid.EntityFullJid;
import org.jxmpp.jid.EntityJid;
import org.jxmpp.jid.impl.JidCreate;
import org.jxmpp.stringprep.XmppStringprepException;
/**
* A test for for following features:
* <li>Adds support for Direct MUC invitations (see <a href="https://xmpp.org/extensions/xep-0249.html">XEP-0249</a>), which allows offline users to be invited to group chats.</li>
* </ul>
*/
public class MultiUserChatTest extends SmackTestSuite {
private static final int RESPONSE_TIMEOUT_IN_MILLIS = 10000;
private DummyConnection connection;
private MultiUserChatManager multiUserChatManager;
@BeforeEach
public void setUp() throws Exception {
connection = new DummyConnection();
connection.connect();
connection.login();
multiUserChatManager = MultiUserChatManager.getInstanceFor(connection);
}
@AfterEach
public void tearDown() {
if (connection != null) {
connection.disconnect();
connection = null;
}
}
@Test
public void testInviteDirectly() throws Throwable {
EntityBareJid roomJid = JidCreate.entityBareFrom("room@example.com");
EntityBareJid userJid = JidCreate.entityBareFrom("user@example.com");
AtomicBoolean updateRequestSent = new AtomicBoolean();
InvokeDirectlyResponder serverSimulator = new InvokeDirectlyResponder() {
@Override
void verifyRequest(Message updateRequest) {
assertEquals(userJid, updateRequest.getTo(), "The provided JID doesn't match the request!");
GroupChatInvitation groupChatInvitation = (GroupChatInvitation) updateRequest.getExtension(GroupChatInvitation.NAMESPACE);
assertNotNull(groupChatInvitation, "Missing GroupChatInvitation extension");
assertEquals(roomJid, groupChatInvitation.getRoomAddress());
assertNull(groupChatInvitation.getReason());
assertNull(groupChatInvitation.getPassword());
assertFalse(groupChatInvitation.continueAsOneToOneChat());
assertNull(groupChatInvitation.getThread());
updateRequestSent.set(true);
}
};
serverSimulator.start();
// Create multi user chat
MultiUserChat multiUserChat = multiUserChatManager.getMultiUserChat(roomJid);
// Call tested method
multiUserChat.inviteDirectly(userJid);
// Wait for processing requests
serverSimulator.join(RESPONSE_TIMEOUT_IN_MILLIS);
// Check if an error occurred within the simulator
final Throwable exception = serverSimulator.getException();
if (exception != null) {
throw exception;
}
assertTrue(updateRequestSent.get(), "Invite directly request not sent");
}
@Test
public void testInviteDirectlyWithAllOptionalAttributes() throws Throwable {
EntityBareJid roomJid = JidCreate.entityBareFrom("room@example.com");
EntityBareJid userJid = JidCreate.entityBareFrom("user@example.com");
String reason = "reason";
String password = "password";
boolean continueAsOneToOneChat = true;
String thread = "e0ffe42b28561960c6b12b944a092794b9683a38";
AtomicBoolean updateRequestSent = new AtomicBoolean();
InvokeDirectlyResponder serverSimulator = new InvokeDirectlyResponder() {
@Override
void verifyRequest(Message updateRequest) {
assertEquals(userJid, updateRequest.getTo(), "The provided JID doesn't match the request!");
GroupChatInvitation groupChatInvitation = (GroupChatInvitation) updateRequest.getExtension(GroupChatInvitation.NAMESPACE);
assertNotNull(groupChatInvitation, "Missing GroupChatInvitation extension");
assertEquals(roomJid, groupChatInvitation.getRoomAddress());
assertSame(reason, groupChatInvitation.getReason());
assertSame(password, groupChatInvitation.getPassword());
assertSame(continueAsOneToOneChat, groupChatInvitation.continueAsOneToOneChat());
assertSame(thread, groupChatInvitation.getThread());
updateRequestSent.set(true);
}
};
serverSimulator.start();
// Create multi user chat
MultiUserChat multiUserChat = multiUserChatManager.getMultiUserChat(roomJid);
// Call tested method
multiUserChat.inviteDirectly(userJid, reason, password, continueAsOneToOneChat, thread);
// Wait for processing requests
serverSimulator.join(RESPONSE_TIMEOUT_IN_MILLIS);
// Check if an error occurred within the simulator
final Throwable exception = serverSimulator.getException();
if (exception != null) {
throw exception;
}
assertTrue(updateRequestSent.get(), "Invite directly request not sent");
}
@Test
public void shouldReceiveOfflineInvitation() throws XmppStringprepException, Throwable {
EntityBareJid roomJid = JidCreate.entityBareFrom("room@example.com");
EntityFullJid inviterJid = JidCreate.entityFullFrom("inviter@example.com/user1");
EntityBareJid inviteeJid = JidCreate.entityBareFrom("invitee@example.com");
Invite invite = new MUCUser.Invite(null, inviterJid);
GroupChatInvitation groupChatInvitation = new GroupChatInvitation(roomJid);
Message sentMessage = connection.getStanzaFactory().buildMessageStanza()
.from(inviterJid)
.to(inviteeJid)
.addExtension(groupChatInvitation)
.build();
// Prepare listener to receive a group invitation
GroupInvitationListener groupInvitationListener = new GroupInvitationListener() {
@Override
public void verifyInvitation(XMPPConnection conn, MultiUserChat room, EntityJid inviter,
String reason, String password, Message message, MUCUser.Invite invitation) {
try {
// Check all parameters' values.
assertSame(connection, conn);
assertSame(connection, room.getXmppConnection());
assertEquals(roomJid, room.getRoom());
assertEquals(inviterJid, inviter);
assertNull(reason);
assertNull(password);
assertSame(sentMessage, message);
assertEquals(invite.getReason(), invitation.getReason());
assertEquals(invite.getFrom(), invitation.getFrom());
} catch (final Throwable e) {
this.setError(e);
}
}
};
multiUserChatManager.addInvitationListener(groupInvitationListener);
// Simulate sending a message with a group invitation
connection.processStanza(sentMessage);
// Wait for the listener to be called or throw a timeout exception
groupInvitationListener.waitUntilInvocationOrTimeout();
if (groupInvitationListener.getError() != null) {
throw groupInvitationListener.getError();
}
}
@Test
public void shouldReceiveOfflineInvitationWithAllOptionalAttributes() throws XmppStringprepException, Throwable {
EntityBareJid roomJid = JidCreate.entityBareFrom("room@example.com");
String expectedReason = "reason";
String expectedPassword = "password";
boolean expectedContinueAsOneToOneChat = true;
String expectedThread = "e0ffe42b28561960c6b12b944a092794b9683a38";
EntityFullJid inviterJid = JidCreate.entityFullFrom("inviter@example.com/user1");
EntityBareJid inviteeJid = JidCreate.entityBareFrom("invitee@example.com");
Invite invite = new MUCUser.Invite(expectedReason, inviterJid);
GroupChatInvitation groupChatInvitation =
new GroupChatInvitation(roomJid, expectedReason, expectedPassword, expectedContinueAsOneToOneChat, expectedThread);
Message sentMessage = connection.getStanzaFactory().buildMessageStanza()
.from(inviterJid)
.to(inviteeJid)
.addExtension(groupChatInvitation)
.build();
// Prepare listener to receive a group invitation
GroupInvitationListener groupInvitationListener = new GroupInvitationListener() {
@Override
public void verifyInvitation(XMPPConnection conn, MultiUserChat room, EntityJid inviter,
String reason, String password, Message message, MUCUser.Invite invitation) {
try {
// Check all parameters' values.
assertSame(connection, conn);
assertSame(connection, room.getXmppConnection());
assertEquals(roomJid, room.getRoom());
assertEquals(inviterJid, inviter);
assertEquals(expectedReason, reason);
assertEquals(expectedPassword, password);
assertSame(sentMessage, message);
assertEquals(invite.getReason(), invitation.getReason());
assertEquals(invite.getFrom(), invitation.getFrom());
} catch (final Throwable e) {
this.setError(e);
}
}
};
multiUserChatManager.addInvitationListener(groupInvitationListener);
// Simulate sending a message with a group invitation
connection.processStanza(sentMessage);
// Wait for the listener to be called or throw a timeout exception
groupInvitationListener.waitUntilInvocationOrTimeout();
if (groupInvitationListener.getError() != null) {
throw groupInvitationListener.getError();
}
}
/**
* This class can be used to simulate the server response for invoke directly request.
*/
private abstract class InvokeDirectlyResponder extends Thread {
protected Throwable exception;
abstract void verifyRequest(Message updateRequest);
@Override
public void run() {
try {
while (true) {
final Stanza stanza = connection.getSentPacket();
if (stanza instanceof Message) {
Message message = (Message) stanza;
verifyRequest(message);
break;
}
}
}
catch (Throwable e) {
exception = e;
}
}
/**
* Returns the exception or error if something went wrong.
*
* @return the Throwable exception or error that occurred.
*/
Throwable getException() {
return exception;
}
}
/**
* This class can be used to simulate receiving an invitation.
*/
private abstract static class GroupInvitationListener extends WaitForPacketListener implements InvitationListener {
protected volatile Throwable exception;
public abstract void verifyInvitation(XMPPConnection conn, MultiUserChat room, EntityJid inviter,
String reason, String password, Message message, MUCUser.Invite invitation);
@Override
public void invitationReceived(XMPPConnection conn, MultiUserChat room, EntityJid inviter,
String reason, String password, Message message, MUCUser.Invite invitation) {
verifyInvitation(conn, room, inviter, reason, password, message, invitation);
reportInvoked();
}
public synchronized Throwable getError() {
return exception;
}
public synchronized void setError(Throwable e) {
exception = e;
}
}
}

View File

@ -352,6 +352,12 @@
* <td>Efficient roster synchronization.</td>
* </tr>
* <tr>
* <td>Direct MUC Invitations</td>
* <td><a href="https://xmpp.org/extensions/xep-0249.html">XEP-0249</a></td>
* <td></td>
* <td>Allows sending a MUC invitation directly from the user to the contact with mediation by the room.</td>
* </tr>
* <tr>
* <td>Message Carbons</td>
* <td><a href="https://xmpp.org/extensions/xep-0280.html">XEP-0280</a></td>
* <td>{@link org.jivesoftware.smackx.carbons}</td>