/** * $RCSfile: OfflineMessageStrategy.java,v $ * $Revision: 3114 $ * $Date: 2005-11-23 18:12:54 -0300 (Wed, 23 Nov 2005) $ * * Copyright (C) 2005-2008 Jive Software. All rights reserved. * * This software is published under the terms of the GNU Public License (GPL), * a copy of which is included in this distribution, or a commercial license * agreement with Jive. */ package org.jivesoftware.openfire; import org.jivesoftware.openfire.container.BasicModule; import org.jivesoftware.openfire.privacy.PrivacyList; import org.jivesoftware.openfire.privacy.PrivacyListManager; import org.jivesoftware.openfire.user.UserManager; import org.jivesoftware.util.JiveGlobals; import org.jivesoftware.util.Log; import org.xmpp.packet.JID; import org.xmpp.packet.Message; import org.xmpp.packet.PacketError; import java.util.List; import java.util.concurrent.CopyOnWriteArrayList; /** * Controls what is done with offline messages. * * @author Iain Shigeoka */ public class OfflineMessageStrategy extends BasicModule { private static int quota = 100*1024; // Default to 100 K. private static Type type = Type.store_and_bounce; private static List<OfflineMessageListener> listeners = new CopyOnWriteArrayList<OfflineMessageListener>(); private OfflineMessageStore messageStore; private JID serverAddress; private PacketRouter router; public OfflineMessageStrategy() { super("Offline Message Strategy"); } public int getQuota() { return quota; } public void setQuota(int quota) { OfflineMessageStrategy.quota = quota; JiveGlobals.setProperty("xmpp.offline.quota", Integer.toString(quota)); } public OfflineMessageStrategy.Type getType() { return type; } public void setType(OfflineMessageStrategy.Type type) { if (type == null) { throw new IllegalArgumentException(); } OfflineMessageStrategy.type = type; JiveGlobals.setProperty("xmpp.offline.type", type.toString()); } public void storeOffline(Message message) { if (message != null) { // Do nothing if the message was sent to the server itself, an anonymous user or a non-existent user JID recipientJID = message.getTo(); if (recipientJID == null || serverAddress.equals(recipientJID) || recipientJID.getNode() == null || !UserManager.getInstance().isRegisteredUser(recipientJID.getNode())) { return; } // Do not store messages of type groupchat, error or headline as specified in JEP-160 if (Message.Type.groupchat == message.getType() || Message.Type.error == message.getType() || Message.Type.headline == message.getType()) { return; } // Do not store messages if communication is blocked PrivacyList list = PrivacyListManager.getInstance().getDefaultPrivacyList(recipientJID.getNode()); if (list != null && list.shouldBlockPacket(message)) { return; } if (type == Type.bounce) { bounce(message); } else if (type == Type.store) { store(message); } else if (type == Type.store_and_bounce) { if (underQuota(message)) { store(message); } else { bounce(message); } } else if (type == Type.store_and_drop) { if (underQuota(message)) { store(message); } } } } /** * Registers a listener to receive events. * * @param listener the listener. */ public static void addListener(OfflineMessageListener listener) { if (listener == null) { throw new NullPointerException(); } listeners.add(listener); } /** * Unregisters a listener to receive events. * * @param listener the listener. */ public static void removeListener(OfflineMessageListener listener) { listeners.remove(listener); } private boolean underQuota(Message message) { return quota > messageStore.getSize(message.getTo().getNode()) + message.toXML().length(); } private void store(Message message) { messageStore.addMessage(message); // Inform listeners that an offline message was stored if (!listeners.isEmpty()) { for (OfflineMessageListener listener : listeners) { listener.messageStored(message); } } } private void bounce(Message message) { // Do nothing if the sender was the server itself if (message.getFrom() == null) { return; } try { // Generate a rejection response to the sender Message errorResponse = message.createCopy(); errorResponse.setError(new PacketError(PacketError.Condition.item_not_found, PacketError.Type.continue_processing)); errorResponse.setFrom(message.getTo()); errorResponse.setTo(message.getFrom()); // Send the response router.route(errorResponse); // Inform listeners that an offline message was bounced if (!listeners.isEmpty()) { for (OfflineMessageListener listener : listeners) { listener.messageBounced(message); } } } catch (Exception e) { Log.error(e); } } public void initialize(XMPPServer server) { super.initialize(server); messageStore = server.getOfflineMessageStore(); router = server.getPacketRouter(); serverAddress = new JID(server.getServerInfo().getXMPPDomain()); String quota = JiveGlobals.getProperty("xmpp.offline.quota"); if (quota != null && quota.length() > 0) { OfflineMessageStrategy.quota = Integer.parseInt(quota); } String type = JiveGlobals.getProperty("xmpp.offline.type"); if (type != null && type.length() > 0) { OfflineMessageStrategy.type = Type.valueOf(type); } } /** * Strategy types. */ public enum Type { /** * All messages are bounced to the sender. */ bounce, /** * All messages are silently dropped. */ drop, /** * All messages are stored. */ store, /** * Messages are stored up to the storage limit, and then bounced. */ store_and_bounce, /** * Messages are stored up to the storage limit, and then silently dropped. */ store_and_drop } }