/** * * Copyright 2013-2014 Georg Lukas, 2017-2020 Florian Schmaus, 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.carbons; import java.util.Map; import java.util.Set; import java.util.WeakHashMap; import java.util.concurrent.CopyOnWriteArraySet; import java.util.logging.Level; import java.util.logging.Logger; import org.jivesoftware.smack.AsyncButOrdered; import org.jivesoftware.smack.ConnectionCreationListener; import org.jivesoftware.smack.ConnectionListener; import org.jivesoftware.smack.Manager; import org.jivesoftware.smack.SmackException; import org.jivesoftware.smack.SmackException.NoResponseException; import org.jivesoftware.smack.SmackException.NotConnectedException; import org.jivesoftware.smack.SmackFuture; import org.jivesoftware.smack.StanzaListener; import org.jivesoftware.smack.XMPPConnection; import org.jivesoftware.smack.XMPPConnectionRegistry; import org.jivesoftware.smack.XMPPException; import org.jivesoftware.smack.XMPPException.XMPPErrorException; import org.jivesoftware.smack.filter.AndFilter; import org.jivesoftware.smack.filter.FromMatchesFilter; import org.jivesoftware.smack.filter.OrFilter; import org.jivesoftware.smack.filter.StanzaExtensionFilter; import org.jivesoftware.smack.filter.StanzaFilter; import org.jivesoftware.smack.filter.StanzaTypeFilter; import org.jivesoftware.smack.packet.IQ; import org.jivesoftware.smack.packet.Message; import org.jivesoftware.smack.packet.Stanza; import org.jivesoftware.smack.util.ExceptionCallback; import org.jivesoftware.smack.util.SuccessCallback; import org.jivesoftware.smackx.carbons.packet.Carbon; import org.jivesoftware.smackx.carbons.packet.CarbonExtension; import org.jivesoftware.smackx.carbons.packet.CarbonExtension.Direction; import org.jivesoftware.smackx.carbons.packet.CarbonExtension.Private; import org.jivesoftware.smackx.disco.ServiceDiscoveryManager; import org.jivesoftware.smackx.forward.packet.Forwarded; import org.jxmpp.jid.BareJid; import org.jxmpp.jid.EntityFullJid; /** * Manager for XEP-0280: Message Carbons. This class implements the manager for registering {@link CarbonExtension} * support, enabling and disabling message carbons, and for {@link CarbonCopyReceivedListener}. *

* Note that it is important to match the 'from' attribute of the message wrapping a carbon copy, as otherwise it would * may be possible for others to impersonate users. Smack's CarbonManager takes care of that in * {@link CarbonCopyReceivedListener}s which were registered with * {@link #addCarbonCopyReceivedListener(CarbonCopyReceivedListener)}. *

*

* You should call enableCarbons() before sending your first undirected presence (aka. the "initial presence"). *

* * @author Georg Lukas * @author Florian Schmaus * @author Paul Schaub */ public final class CarbonManager extends Manager { private static final Logger LOGGER = Logger.getLogger(CarbonManager.class.getName()); private static Map INSTANCES = new WeakHashMap<>(); private static boolean ENABLED_BY_DEFAULT = false; static { XMPPConnectionRegistry.addConnectionCreationListener(new ConnectionCreationListener() { @Override public void connectionCreated(XMPPConnection connection) { getInstanceFor(connection); } }); } private static final StanzaFilter CARBON_EXTENSION_FILTER = // @formatter:off new AndFilter( new OrFilter( new StanzaExtensionFilter(CarbonExtension.Direction.sent.name(), CarbonExtension.NAMESPACE), new StanzaExtensionFilter(CarbonExtension.Direction.received.name(), CarbonExtension.NAMESPACE) ), StanzaTypeFilter.MESSAGE ); // @formatter:on private final Set listeners = new CopyOnWriteArraySet<>(); private volatile boolean enabled_state = false; private volatile boolean enabledByDefault = ENABLED_BY_DEFAULT; private final StanzaListener carbonsListener; private final AsyncButOrdered carbonsListenerAsyncButOrdered = new AsyncButOrdered<>(); /** * Should Carbons be automatically be enabled once the connection is authenticated? * Default: false * * @param enabledByDefault new default value */ public static void setEnabledByDefault(boolean enabledByDefault) { ENABLED_BY_DEFAULT = enabledByDefault; } private CarbonManager(XMPPConnection connection) { super(connection); ServiceDiscoveryManager sdm = ServiceDiscoveryManager.getInstanceFor(connection); sdm.addFeature(CarbonExtension.NAMESPACE); carbonsListener = new StanzaListener() { @Override public void processStanza(final Stanza stanza) { final Message wrappingMessage = (Message) stanza; final CarbonExtension carbonExtension = CarbonExtension.from(wrappingMessage); final Direction direction = carbonExtension.getDirection(); final Forwarded forwarded = carbonExtension.getForwarded(); final Message carbonCopy = forwarded.getForwardedStanza(); final BareJid from = carbonCopy.getFrom().asBareJid(); carbonsListenerAsyncButOrdered.performAsyncButOrdered(from, new Runnable() { @Override public void run() { for (CarbonCopyReceivedListener listener : listeners) { listener.onCarbonCopyReceived(direction, carbonCopy, wrappingMessage); } } }); } }; connection.addConnectionListener(new ConnectionListener() { @Override public void connectionClosed() { // Reset the state if the connection was cleanly closed. Note that this is not strictly necessary, // because we also reset in authenticated() if the stream got not resumed, but for maximum correctness, // also reset here. enabled_state = false; connection().removeSyncStanzaListener(carbonsListener); } @Override public void authenticated(XMPPConnection connection, boolean resumed) { if (!resumed) { // Non-resumed XMPP sessions always start with disabled carbons enabled_state = false; try { if (shouldCarbonsBeEnabled() && isSupportedByServer()) { setCarbonsEnabled(true); } } catch (InterruptedException | XMPPErrorException | NotConnectedException | NoResponseException e) { LOGGER.log(Level.WARNING, "Cannot check for Carbon support and / or enable carbons.", e); } } addCarbonsListener(connection); } }); addCarbonsListener(connection); } private void addCarbonsListener(XMPPConnection connection) { EntityFullJid localAddress = connection.getUser(); if (localAddress == null) { // We where not connected yet and thus we don't know our XMPP address at the moment, which we need to match incoming // carbons securely. Abort here. The ConnectionListener above will eventually setup the carbons listener. return; } // XEP-0280 ยง 11. Security Considerations "Any forwarded copies received by a Carbons-enabled client MUST be // from that user's bare JID; any copies that do not meet this requirement MUST be ignored." Otherwise, if // those copies do not get ignored, malicious users may be able to impersonate other users. That is why the // 'from' matcher is important here. connection.addSyncStanzaListener(carbonsListener, new AndFilter(CARBON_EXTENSION_FILTER, FromMatchesFilter.createBare(localAddress))); } /** * Obtain the CarbonManager responsible for a connection. * * @param connection the connection object. * * @return a CarbonManager instance */ public static synchronized CarbonManager getInstanceFor(XMPPConnection connection) { CarbonManager carbonManager = INSTANCES.get(connection); if (carbonManager == null) { carbonManager = new CarbonManager(connection); INSTANCES.put(connection, carbonManager); } return carbonManager; } private static IQ carbonsEnabledIQ(final boolean new_state) { IQ request; if (new_state) { request = new Carbon.Enable(); } else { request = new Carbon.Disable(); } return request; } /** * Add a carbon copy received listener. * * @param listener the listener to register. * @return true if the filter was not already registered. * @since 4.2 */ public boolean addCarbonCopyReceivedListener(CarbonCopyReceivedListener listener) { return listeners.add(listener); } /** * Remove a carbon copy received listener. * * @param listener the listener to register. * @return true if the filter was registered. * @since 4.2 */ public boolean removeCarbonCopyReceivedListener(CarbonCopyReceivedListener listener) { return listeners.remove(listener); } /** * Returns true if XMPP Carbons are supported by the server. * * @return true if supported * @throws NotConnectedException if the XMPP connection is not connected. * @throws XMPPErrorException if there was an XMPP error returned. * @throws NoResponseException if there was no response from the remote entity. * @throws InterruptedException if the calling thread was interrupted. */ public boolean isSupportedByServer() throws NoResponseException, XMPPErrorException, NotConnectedException, InterruptedException { return ServiceDiscoveryManager.getInstanceFor(connection()).serverSupportsFeature(CarbonExtension.NAMESPACE); } /** * Notify server to change the carbons state. This method returns * immediately and changes the variable when the reply arrives. * * You should first check for support using isSupportedByServer(). * * @param new_state whether carbons should be enabled or disabled * @deprecated use {@link #enableCarbonsAsync(ExceptionCallback)} or {@link #disableCarbonsAsync(ExceptionCallback)} instead. */ @Deprecated public void sendCarbonsEnabled(final boolean new_state) { sendUseCarbons(new_state, null); } /** * Enable carbons asynchronously. If an error occurs as result of the attempt to enable carbons, the optional * exceptionCallback will be invoked. *

* Note that although this method is asynchronous, it may block if the outgoing stream element queue is full (e.g. * because of a slow network connection). Thus, if the thread performing this operation is interrupted while the * queue is full, an {@link InterruptedException} is thrown. *

* * @param exceptionCallback the optional exception callback. * @since 4.2 */ public void enableCarbonsAsync(ExceptionCallback exceptionCallback) { sendUseCarbons(true, exceptionCallback); } /** * Disable carbons asynchronously. If an error occurs as result of the attempt to disable carbons, the optional * exceptionCallback will be invoked. *

* Note that although this method is asynchronous, it may block if the outgoing stream element queue is full (e.g. * because of a slow network connection). Thus, if the thread performing this operation is interrupted while the * queue is full, an {@link InterruptedException} is thrown. *

* * @param exceptionCallback the optional exception callback. * @since 4.2 */ public void disableCarbonsAsync(ExceptionCallback exceptionCallback) { sendUseCarbons(false, exceptionCallback); } private void sendUseCarbons(final boolean use, ExceptionCallback exceptionCallback) { enabledByDefault = use; IQ setIQ = carbonsEnabledIQ(use); SmackFuture future = connection().sendIqRequestAsync(setIQ); future.onSuccess(new SuccessCallback() { @Override public void onSuccess(IQ result) { enabled_state = use; } }).onError(exceptionCallback); } /** * Notify server to change the carbons state. This method blocks * some time until the server replies to the IQ and returns true on * success. * * You should first check for support using isSupportedByServer(). * * @param new_state whether carbons should be enabled or disabled * @throws XMPPErrorException if there was an XMPP error returned. * @throws NoResponseException if there was no response from the remote entity. * @throws NotConnectedException if the XMPP connection is not connected. * @throws InterruptedException if the calling thread was interrupted. * */ public synchronized void setCarbonsEnabled(final boolean new_state) throws NoResponseException, XMPPErrorException, NotConnectedException, InterruptedException { enabledByDefault = new_state; if (enabled_state == new_state) return; IQ setIQ = carbonsEnabledIQ(new_state); connection().createStanzaCollectorAndSend(setIQ).nextResultOrThrow(); enabled_state = new_state; } /** * Helper method to enable carbons. * * @throws XMPPException if an XMPP protocol error was received. * @throws SmackException if there was no response from the server. * @throws InterruptedException if the calling thread was interrupted. */ public void enableCarbons() throws XMPPException, SmackException, InterruptedException { setCarbonsEnabled(true); } /** * Helper method to disable carbons. * * @throws XMPPException if an XMPP protocol error was received. * @throws SmackException if there was no response from the server. * @throws InterruptedException if the calling thread was interrupted. */ public void disableCarbons() throws XMPPException, SmackException, InterruptedException { setCarbonsEnabled(false); } /** * Check if carbons are enabled on this connection. * * @return true if carbons are enabled, else false. */ public boolean getCarbonsEnabled() { return this.enabled_state; } private boolean shouldCarbonsBeEnabled() { return enabledByDefault; } /** * Mark a message as "private", so it will not be carbon-copied. * * @param msg Message object to mark private * @deprecated use {@link Private#addTo(Message)} */ @Deprecated public static void disableCarbons(Message msg) { msg.addExtension(Private.INSTANCE); } }