mirror of
https://github.com/vanitasvitae/Smack.git
synced 2024-12-24 11:38:00 +01:00
[extensions] Improved Support for Direct MUC Invitations (XEP-0249)
[flow: rebase of paul's initial submission which required adjustments] Co-authored-by: Florian Schmaus <flo@geekplace.eu>
This commit is contained in:
parent
726dbc0d27
commit
664a141190
7 changed files with 263 additions and 45 deletions
|
@ -0,0 +1,25 @@
|
|||
/**
|
||||
*
|
||||
* Copyright 2020 Paul Schaub.
|
||||
*
|
||||
* 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 org.jivesoftware.smack.packet.Stanza;
|
||||
import org.jivesoftware.smackx.muc.packet.GroupChatInvitation;
|
||||
|
||||
public interface DirectMucInvitationListener {
|
||||
|
||||
void invitationReceived(GroupChatInvitation invitation, Stanza stanza);
|
||||
}
|
|
@ -0,0 +1,111 @@
|
|||
/**
|
||||
*
|
||||
* Copyright 2020 Paul Schaub.
|
||||
*
|
||||
* 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 java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.WeakHashMap;
|
||||
|
||||
import org.jivesoftware.smack.Manager;
|
||||
import org.jivesoftware.smack.SmackException;
|
||||
import org.jivesoftware.smack.XMPPConnection;
|
||||
import org.jivesoftware.smack.XMPPConnectionRegistry;
|
||||
import org.jivesoftware.smack.XMPPException;
|
||||
import org.jivesoftware.smack.filter.StanzaExtensionFilter;
|
||||
import org.jivesoftware.smack.packet.Message;
|
||||
import org.jivesoftware.smack.packet.MessageBuilder;
|
||||
import org.jivesoftware.smackx.disco.ServiceDiscoveryManager;
|
||||
import org.jivesoftware.smackx.muc.packet.GroupChatInvitation;
|
||||
|
||||
import org.jxmpp.jid.EntityBareJid;
|
||||
|
||||
/**
|
||||
* Smacks API for XEP-0249: Direct MUC Invitations.
|
||||
* Use this instead of {@link org.jivesoftware.smackx.muc.packet.MUCUser.Invite}.
|
||||
*
|
||||
* To invite a user to a group chat, use {@link #inviteToMuc(MultiUserChat, EntityBareJid)}.
|
||||
*
|
||||
* In order to listen for incoming invitations, register a {@link DirectMucInvitationListener} using
|
||||
* {@link #addInvitationListener(DirectMucInvitationListener)}.
|
||||
*
|
||||
* @see <a href="https://xmpp.org/extensions/xep-0249.html">Direct MUC Invitations</a>
|
||||
*/
|
||||
public final class DirectMucInvitationManager extends Manager {
|
||||
|
||||
private static final Map<XMPPConnection, DirectMucInvitationManager> INSTANCES = new WeakHashMap<>();
|
||||
private final List<DirectMucInvitationListener> directMucInvitationListeners = new ArrayList<>();
|
||||
private final ServiceDiscoveryManager serviceDiscoveryManager;
|
||||
|
||||
static {
|
||||
XMPPConnectionRegistry.addConnectionCreationListener(DirectMucInvitationManager::getInstanceFor);
|
||||
}
|
||||
|
||||
public static synchronized DirectMucInvitationManager getInstanceFor(XMPPConnection connection) {
|
||||
DirectMucInvitationManager manager = INSTANCES.get(connection);
|
||||
if (manager == null) {
|
||||
manager = new DirectMucInvitationManager(connection);
|
||||
INSTANCES.put(connection, manager);
|
||||
}
|
||||
return manager;
|
||||
}
|
||||
|
||||
private DirectMucInvitationManager(XMPPConnection connection) {
|
||||
super(connection);
|
||||
serviceDiscoveryManager = ServiceDiscoveryManager.getInstanceFor(connection);
|
||||
|
||||
connection().addAsyncStanzaListener(stanza -> {
|
||||
GroupChatInvitation invitation = stanza.getExtension(GroupChatInvitation.class);
|
||||
for (DirectMucInvitationListener listener : directMucInvitationListeners) {
|
||||
listener.invitationReceived(invitation, stanza);
|
||||
}
|
||||
}, new StanzaExtensionFilter(GroupChatInvitation.ELEMENT, GroupChatInvitation.NAMESPACE));
|
||||
serviceDiscoveryManager.addFeature(GroupChatInvitation.NAMESPACE);
|
||||
}
|
||||
|
||||
public void inviteToMuc(MultiUserChat muc, EntityBareJid user)
|
||||
throws SmackException.NotConnectedException, InterruptedException {
|
||||
inviteToMuc(muc, user, null, null, false, null);
|
||||
}
|
||||
|
||||
public void inviteToMuc(MultiUserChat muc, EntityBareJid user, String password, String reason, boolean continueAsOneToOneChat, String thread)
|
||||
throws SmackException.NotConnectedException, InterruptedException {
|
||||
inviteToMuc(user, new GroupChatInvitation(muc.getRoom(), reason, password, continueAsOneToOneChat, thread));
|
||||
}
|
||||
|
||||
public void inviteToMuc(EntityBareJid jid, GroupChatInvitation invitation) throws SmackException.NotConnectedException, InterruptedException {
|
||||
Message invitationMessage = MessageBuilder.buildMessage()
|
||||
.to(jid)
|
||||
.addExtension(invitation)
|
||||
.build();
|
||||
connection().sendStanza(invitationMessage);
|
||||
}
|
||||
|
||||
public boolean userSupportsInvitations(EntityBareJid jid)
|
||||
throws XMPPException.XMPPErrorException, SmackException.NotConnectedException, InterruptedException,
|
||||
SmackException.NoResponseException {
|
||||
return serviceDiscoveryManager.supportsFeature(jid, GroupChatInvitation.NAMESPACE);
|
||||
}
|
||||
|
||||
public synchronized void addInvitationListener(DirectMucInvitationListener listener) {
|
||||
directMucInvitationListeners.add(listener);
|
||||
}
|
||||
|
||||
public synchronized void removeInvitationListener(DirectMucInvitationListener listener) {
|
||||
directMucInvitationListeners.remove(listener);
|
||||
}
|
||||
}
|
|
@ -83,6 +83,10 @@ import org.jxmpp.util.cache.ExpirationCache;
|
|||
* further attempts will be made for the other rooms.
|
||||
* </p>
|
||||
*
|
||||
* Note:
|
||||
* For inviting other users to a group chat or listening for such invitations, take a look at the
|
||||
* {@link DirectMucInvitationManager} which provides an implementation of XEP-0249: Direct MUC Invitations.
|
||||
*
|
||||
* @see <a href="http://xmpp.org/extensions/xep-0045.html">XEP-0045: Multi-User Chat</a>
|
||||
*/
|
||||
public final class MultiUserChatManager extends Manager {
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
/**
|
||||
*
|
||||
* Copyright 2003-2007 Jive Software.
|
||||
* Copyright 2003-2007 Jive Software, 2020 Paul Schaub.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
|
@ -20,6 +20,8 @@ import javax.xml.namespace.QName;
|
|||
|
||||
import org.jivesoftware.smack.packet.ExtensionElement;
|
||||
import org.jivesoftware.smack.packet.Stanza;
|
||||
import org.jivesoftware.smack.util.EqualsUtil;
|
||||
import org.jivesoftware.smack.util.HashCode;
|
||||
import org.jivesoftware.smack.util.Objects;
|
||||
import org.jivesoftware.smack.util.XmlStringBuilder;
|
||||
|
||||
|
@ -27,32 +29,13 @@ import org.jxmpp.jid.EntityBareJid;
|
|||
|
||||
/**
|
||||
* A group chat invitation stanza extension, which is used to invite other
|
||||
* users to a group chat room. To invite a user to a group chat room, address
|
||||
* a new message to the user and set the room name appropriately, as in the
|
||||
* following code example:
|
||||
* users to a group chat room.
|
||||
*
|
||||
* <pre>
|
||||
* Message message = new Message("user@chat.example.com");
|
||||
* message.setBody("Join me for a group chat!");
|
||||
* message.addExtension(new GroupChatInvitation("room@chat.example.com"););
|
||||
* con.sendStanza(message);
|
||||
* </pre>
|
||||
*
|
||||
* To listen for group chat invitations, use a StanzaExtensionFilter for the
|
||||
* <code>x</code> element name and <code>jabber:x:conference</code> namespace, as in the
|
||||
* following code example:
|
||||
*
|
||||
* <pre>
|
||||
* PacketFilter filter = new StanzaExtensionFilter("x", "jabber:x:conference");
|
||||
* // Create a stanza collector or stanza listeners using the filter...
|
||||
* </pre>
|
||||
*
|
||||
* <b>Note</b>: this protocol is outdated now that the Multi-User Chat (MUC) XEP is available
|
||||
* (<a href="http://www.xmpp.org/extensions/jep-0045.html">XEP-45</a>). However, most
|
||||
* existing clients still use this older protocol. Once MUC support becomes more
|
||||
* widespread, this API may be deprecated.
|
||||
* This implementation now conforms to XEP-0249: Direct MUC Invitations,
|
||||
* while staying backwards compatible to legacy MUC invitations.
|
||||
*
|
||||
* @author Matt Tucker
|
||||
* @author Paul Schaub
|
||||
*/
|
||||
public class GroupChatInvitation implements ExtensionElement {
|
||||
|
||||
|
@ -68,6 +51,12 @@ public class GroupChatInvitation implements ExtensionElement {
|
|||
|
||||
public static final QName QNAME = new QName(NAMESPACE, ELEMENT);
|
||||
|
||||
public static final String ATTR_CONTINUE = "continue";
|
||||
public static final String ATTR_JID = "jid";
|
||||
public static final String ATTR_PASSWORD = "password";
|
||||
public static final String ATTR_REASON = "reason";
|
||||
public static final String ATTR_THREAD = "thread";
|
||||
|
||||
private final EntityBareJid roomAddress;
|
||||
private final String reason;
|
||||
private final String password;
|
||||
|
@ -170,18 +159,37 @@ public class GroupChatInvitation implements ExtensionElement {
|
|||
@Override
|
||||
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.jidAttribute(getRoomAddress());
|
||||
xml.optAttribute(ATTR_REASON, getReason());
|
||||
xml.optAttribute(ATTR_PASSWORD, getPassword());
|
||||
xml.optAttribute(ATTR_THREAD, getThread());
|
||||
xml.optBooleanAttribute(ATTR_CONTINUE, continueAsOneToOneChat());
|
||||
|
||||
xml.closeEmptyElement();
|
||||
return xml;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
return EqualsUtil.equals(this, obj, (equalsBuilder, other) -> equalsBuilder
|
||||
.append(getRoomAddress(), other.getRoomAddress())
|
||||
.append(getPassword(), other.getPassword())
|
||||
.append(getReason(), other.getReason())
|
||||
.append(continueAsOneToOneChat(), other.continueAsOneToOneChat())
|
||||
.append(getThread(), other.getThread()));
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return HashCode.builder()
|
||||
.append(getRoomAddress())
|
||||
.append(getPassword())
|
||||
.append(getReason())
|
||||
.append(continueAsOneToOneChat())
|
||||
.append(getThread())
|
||||
.build();
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the group chat invitation from the given stanza.
|
||||
* @param packet TODO javadoc me please
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
/**
|
||||
*
|
||||
* Copyright 2003-2007 Jive Software, 2022 Florian Schmaus.
|
||||
* Copyright 2003-2007 Jive Software, 2020 Paul Schaub, 2022-2023 Florian Schmaus.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
|
@ -16,8 +16,12 @@
|
|||
*/
|
||||
package org.jivesoftware.smackx.muc.provider;
|
||||
|
||||
import static org.jivesoftware.smackx.muc.packet.GroupChatInvitation.ATTR_CONTINUE;
|
||||
import static org.jivesoftware.smackx.muc.packet.GroupChatInvitation.ATTR_PASSWORD;
|
||||
import static org.jivesoftware.smackx.muc.packet.GroupChatInvitation.ATTR_REASON;
|
||||
import static org.jivesoftware.smackx.muc.packet.GroupChatInvitation.ATTR_THREAD;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.text.ParseException;
|
||||
|
||||
import org.jivesoftware.smack.packet.XmlEnvironment;
|
||||
import org.jivesoftware.smack.parsing.SmackParsingException;
|
||||
|
@ -33,11 +37,14 @@ public class GroupChatInvitationProvider extends ExtensionElementProvider<GroupC
|
|||
|
||||
@Override
|
||||
public GroupChatInvitation parse(XmlPullParser parser, int initialDepth, XmlEnvironment xmlEnvironment)
|
||||
throws XmlPullParserException, IOException, SmackParsingException, ParseException {
|
||||
EntityBareJid roomAddress = ParserUtils.getBareJidAttribute(parser);
|
||||
// Advance to end of extension.
|
||||
parser.next();
|
||||
return new GroupChatInvitation(roomAddress);
|
||||
}
|
||||
throws XmlPullParserException, IOException, SmackParsingException {
|
||||
|
||||
EntityBareJid roomJid = ParserUtils.getBareJidAttribute(parser);
|
||||
String password = parser.getAttributeValue(ATTR_PASSWORD);
|
||||
String reason = parser.getAttributeValue(ATTR_REASON);
|
||||
boolean isContinue = ParserUtils.getBooleanAttribute(parser, ATTR_CONTINUE, false);
|
||||
String thread = parser.getAttributeValue(ATTR_THREAD);
|
||||
|
||||
return new GroupChatInvitation(roomJid, reason, password, isContinue, thread);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,69 @@
|
|||
/**
|
||||
*
|
||||
* Copyright 2020 Paul Schaub.
|
||||
*
|
||||
* 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.packet;
|
||||
|
||||
import static org.jivesoftware.smack.test.util.XmlAssertUtil.assertXmlSimilar;
|
||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
import org.jivesoftware.smack.parsing.SmackParsingException;
|
||||
import org.jivesoftware.smack.test.util.TestUtils;
|
||||
import org.jivesoftware.smack.xml.XmlPullParserException;
|
||||
import org.jivesoftware.smackx.muc.provider.GroupChatInvitationProvider;
|
||||
|
||||
import org.junit.jupiter.api.Test;
|
||||
import org.jxmpp.jid.EntityBareJid;
|
||||
import org.jxmpp.jid.impl.JidCreate;
|
||||
|
||||
public class GroupChatInvitationElementTest {
|
||||
private static final GroupChatInvitationProvider TEST_PROVIDER = new GroupChatInvitationProvider();
|
||||
|
||||
private static final EntityBareJid mucJid = JidCreate.entityBareFromOrThrowUnchecked("darkcave@macbeth.shakespeare.lit");
|
||||
|
||||
@Test
|
||||
public void serializeFullElement() throws XmlPullParserException, IOException, SmackParsingException {
|
||||
final String expectedXml = "" +
|
||||
"<x xmlns='jabber:x:conference'\n" +
|
||||
" continue='true'\n" +
|
||||
" jid='darkcave@macbeth.shakespeare.lit'\n" +
|
||||
" password='cauldronburn'\n" +
|
||||
" reason='Hey Hecate, this is the place for all good witches!'\n" +
|
||||
" thread='e0ffe42b28561960c6b12b944a092794b9683a38'/>";
|
||||
|
||||
GroupChatInvitation invitation = new GroupChatInvitation(mucJid,
|
||||
"Hey Hecate, this is the place for all good witches!",
|
||||
"cauldronburn",
|
||||
true,
|
||||
"e0ffe42b28561960c6b12b944a092794b9683a38");
|
||||
assertXmlSimilar(expectedXml, invitation.toXML());
|
||||
|
||||
GroupChatInvitation parsed = TEST_PROVIDER.parse(TestUtils.getParser(expectedXml));
|
||||
assertEquals(invitation, parsed);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void serializeMinimalElementTest() throws XmlPullParserException, IOException, SmackParsingException {
|
||||
final String expectedXml = "<x xmlns='jabber:x:conference' jid='darkcave@macbeth.shakespeare.lit'/>";
|
||||
|
||||
GroupChatInvitation invitation = new GroupChatInvitation(mucJid);
|
||||
assertXmlSimilar(expectedXml, invitation.toXML());
|
||||
|
||||
GroupChatInvitation parsed = TEST_PROVIDER.parse(TestUtils.getParser(expectedXml));
|
||||
assertEquals(invitation, parsed);
|
||||
}
|
||||
}
|
|
@ -584,12 +584,6 @@
|
|||
* <td>Multi-User Chats for mobile XMPP applications and specific environment.</td>
|
||||
* </tr>
|
||||
* <tr>
|
||||
* <td>Group Chat Invitations</td>
|
||||
* <td></td>
|
||||
* <td></td>
|
||||
* <td>Send invitations to other users to join a group chat room.</td>
|
||||
* </tr>
|
||||
* <tr>
|
||||
* <td><a href="properties.md">Jive Properties</a></td>
|
||||
* <td></td>
|
||||
* <td></td>
|
||||
|
|
Loading…
Reference in a new issue