/** * $RCSfile: RosterItem.java,v $ * $Revision: 3080 $ * $Date: 2005-11-15 01:28:23 -0300 (Tue, 15 Nov 2005) $ * * Copyright (C) 2004-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.roster; import org.jivesoftware.openfire.SharedGroupException; import org.jivesoftware.openfire.group.Group; import org.jivesoftware.openfire.group.GroupManager; import org.jivesoftware.openfire.group.GroupNotFoundException; import org.jivesoftware.openfire.user.UserNameManager; import org.jivesoftware.openfire.user.UserNotFoundException; import org.jivesoftware.util.IntEnum; import org.jivesoftware.util.cache.CacheSizes; import org.jivesoftware.util.cache.Cacheable; import org.jivesoftware.util.cache.ExternalizableUtil; import org.xmpp.packet.JID; import java.io.Externalizable; import java.io.IOException; import java.io.ObjectInput; import java.io.ObjectOutput; import java.util.*; /** * <p>Represents a single roster item for a User's Roster.</p> * <p>The server doesn't need to know anything about roster groups so they are * not stored with easy retrieval or manipulation in mind. The important data * elements of a roster item (beyond the jid adddress of the roster entry) includes:</p> * <p/> * <ul> * <li>nick - A nickname for the user when used in this roster</li> * <li>sub - A subscription type: to, from, none, both</li> * <li>ask - An optional subscription ask status: subscribe, unsubscribe</li> * <li>groups - A list of groups to organize roster entries under (e.g. friends, co-workers, etc)</li> * </ul> * * @author Gaston Dombiak */ public class RosterItem implements Cacheable, Externalizable { public static class SubType extends IntEnum { protected SubType(String name, int value) { super(name, value); register(this); } public static SubType getTypeFromInt(int value) { return (SubType)getEnumFromInt(SubType.class, value); } } public static class AskType extends IntEnum { protected AskType(String name, int value) { super(name, value); register(this); } public static AskType getTypeFromInt(int value) { return (AskType)getEnumFromInt(AskType.class, value); } } public static class RecvType extends IntEnum { protected RecvType(String name, int value) { super(name, value); register(this); } public static RecvType getTypeFromInt(int value) { return (RecvType)getEnumFromInt(RecvType.class, value); } } /** * <p>Indicates the roster item should be removed.</p> */ public static final SubType SUB_REMOVE = new SubType("remove", -1); /** * <p>No subscription is established.</p> */ public static final SubType SUB_NONE = new SubType("none", 0); /** * <p>The roster owner has a subscription to the roster item's presence.</p> */ public static final SubType SUB_TO = new SubType("to", 1); /** * <p>The roster item has a subscription to the roster owner's presence.</p> */ public static final SubType SUB_FROM = new SubType("from", 2); /** * <p>The roster item and owner have a mutual subscription.</p> */ public static final SubType SUB_BOTH = new SubType("both", 3); /** * <p>The roster item has no pending subscription requests.</p> */ public static final AskType ASK_NONE = new AskType("", -1); /** * <p>The roster item has been asked for permission to subscribe to their presence * but no response has been received.</p> */ public static final AskType ASK_SUBSCRIBE = new AskType("subscribe", 0); /** * <p>The roster owner has asked to the roster item to unsubscribe from it's * presence but has not received confirmation.</p> */ public static final AskType ASK_UNSUBSCRIBE = new AskType("unsubscribe", 1); /** * <p>There are no subscriptions that have been received but not presented to the user.</p> */ public static final RecvType RECV_NONE = new RecvType("", -1); /** * <p>The server has received a subscribe request, but has not forwarded it to the user.</p> */ public static final RecvType RECV_SUBSCRIBE = new RecvType("sub", 1); /** * <p>The server has received an unsubscribe request, but has not forwarded it to the user.</p> */ public static final RecvType RECV_UNSUBSCRIBE = new RecvType("unsub", 2); protected RecvType recvStatus; protected JID jid; protected String nickname; protected List<String> groups; protected Set<String> sharedGroups = new HashSet<String>(); protected Set<String> invisibleSharedGroups = new HashSet<String>(); protected SubType subStatus; protected AskType askStatus; /** * Holds the ID that uniquely identifies the roster in the backend store. A value of * zero means that the roster item is not persistent. */ private long rosterID; /** * Constructor added for Externalizable. Do not use this constructor. */ public RosterItem() { } public RosterItem(long id, JID jid, SubType subStatus, AskType askStatus, RecvType recvStatus, String nickname, List<String> groups) { this(jid, subStatus, askStatus, recvStatus, nickname, groups); this.rosterID = id; } public RosterItem(JID jid, SubType subStatus, AskType askStatus, RecvType recvStatus, String nickname, List<String> groups) { this.jid = jid; this.subStatus = subStatus; this.askStatus = askStatus; this.recvStatus = recvStatus; this.nickname = nickname; this.groups = new LinkedList<String>(); if (groups != null) { for (String group : groups) { this.groups.add(group); } } } /** * Create a roster item from the data in another one. * * @param item Item that contains the info of the roster item. */ public RosterItem(org.xmpp.packet.Roster.Item item) { this(item.getJID(), getSubType(item), getAskStatus(item), RosterItem.RECV_NONE, item.getName(), new LinkedList<String>(item.getGroups())); } private static RosterItem.AskType getAskStatus(org.xmpp.packet.Roster.Item item) { if (item.getAsk() == org.xmpp.packet.Roster.Ask.subscribe) { return RosterItem.ASK_SUBSCRIBE; } else if (item.getAsk() == org.xmpp.packet.Roster.Ask.unsubscribe) { return RosterItem.ASK_UNSUBSCRIBE; } else { return RosterItem.ASK_NONE; } } private static RosterItem.SubType getSubType(org.xmpp.packet.Roster.Item item) { if (item.getSubscription() == org.xmpp.packet.Roster.Subscription.to) { return RosterItem.SUB_TO; } else if (item.getSubscription() == org.xmpp.packet.Roster.Subscription.from) { return RosterItem.SUB_FROM; } else if (item.getSubscription() == org.xmpp.packet.Roster.Subscription.both) { return RosterItem.SUB_BOTH; } else if (item.getSubscription() == org.xmpp.packet.Roster.Subscription.remove) { return RosterItem.SUB_REMOVE; } else { return RosterItem.SUB_NONE; } } /** * <p>Obtain the current subscription status of the item.</p> * * @return The subscription status of the item */ public SubType getSubStatus() { return subStatus; } /** * <p>Set the current subscription status of the item.</p> * * @param subStatus The subscription status of the item */ public void setSubStatus(SubType subStatus) { // Optimization: Load user only if we need to set the nickname of the roster item if ("".equals(nickname) && (subStatus == SUB_BOTH || subStatus == SUB_TO)) { try { nickname = UserNameManager.getUserName(jid); } catch (UserNotFoundException e) { // Do nothing } } this.subStatus = subStatus; } /** * <p>Obtain the current ask status of the item.</p> * * @return The ask status of the item */ public AskType getAskStatus() { if (isShared()) { // Redefine the ask status since the item belongs to a shared group return ASK_NONE; } else { return askStatus; } } /** * <p>Set the current ask status of the item.</p> * * @param askStatus The ask status of the item */ public void setAskStatus(AskType askStatus) { this.askStatus = askStatus; } /** * <p>Obtain the current recv status of the item.</p> * * @return The recv status of the item */ public RecvType getRecvStatus() { return recvStatus; } /** * <p>Set the current recv status of the item.</p> * * @param recvStatus The recv status of the item */ public void setRecvStatus(RecvType recvStatus) { this.recvStatus = recvStatus; } /** * <p>Obtain the address of the item.</p> * * @return The address of the item */ public JID getJid() { return jid; } /** * <p>Obtain the current nickname for the item.</p> * * @return The subscription status of the item */ public String getNickname() { return nickname; } /** * <p>Set the current nickname for the item.</p> * * @param nickname The subscription status of the item */ public void setNickname(String nickname) { this.nickname = nickname; } /** * Returns the groups for the item. Shared groups won't be included in the answer. * * @return The groups for the item. */ public List<String> getGroups() { return groups; } /** * Set the current groups for the item. * * @param groups The new lists of groups the item belongs to. * @throws org.jivesoftware.openfire.SharedGroupException if trying to remove shared group. */ public void setGroups(List<String> groups) throws SharedGroupException { if (groups == null) { this.groups = new LinkedList<String>(); } else { // Raise an error if the user is trying to remove the item from a shared group for (Group group: getSharedGroups()) { // Get the display name of the group String groupName = group.getProperties().get("sharedRoster.displayName"); // Check if the group has been removed from the new groups list if (!groups.contains(groupName)) { throw new SharedGroupException("Cannot remove item from shared group"); } } // Remove shared groups from the param Collection<Group> existingGroups = GroupManager.getInstance().getSharedGroups(); for (Iterator<String> it=groups.iterator(); it.hasNext();) { String groupName = it.next(); try { // Optimistic approach for performance reasons. Assume first that the shared // group name is the same as the display name for the shared roster // Check if exists a shared group with this name Group group = GroupManager.getInstance().getGroup(groupName); // Get the display name of the group String displayName = group.getProperties().get("sharedRoster.displayName"); if (displayName != null && displayName.equals(groupName)) { // Remove the shared group from the list (since it exists) try { it.remove(); } catch (IllegalStateException e) { // Do nothing } } } catch (GroupNotFoundException e) { // Check now if there is a group whose display name matches the requested group for (Group group : existingGroups) { // Get the display name of the group String displayName = group.getProperties().get("sharedRoster.displayName"); if (displayName != null && displayName.equals(groupName)) { // Remove the shared group from the list (since it exists) try { it.remove(); } catch (IllegalStateException ise) { // Do nothing } } } } } this.groups = groups; } } /** * Returns the shared groups for the item. * * @return The shared groups this item belongs to. */ public Collection<Group> getSharedGroups() { Collection<Group> groups = new ArrayList<Group>(sharedGroups.size()); for (String groupName : sharedGroups) { try { groups.add(GroupManager.getInstance().getGroup(groupName)); } catch (GroupNotFoundException e) { // Do nothing } } return groups; } /** * Returns the invisible shared groups for the item. These groups are for internal use * and help track the reason why a roster item has a presence subscription of type FROM * when using shared groups. * * @return The shared groups this item belongs to. */ public Collection<Group> getInvisibleSharedGroups() { Collection<Group> groups = new ArrayList<Group>(invisibleSharedGroups.size()); for (String groupName : invisibleSharedGroups) { try { groups.add(GroupManager.getInstance().getGroup(groupName)); } catch (GroupNotFoundException e) { // Do nothing } } return groups; } Set<String> getInvisibleSharedGroupsNames() { return invisibleSharedGroups; } void setInvisibleSharedGroupsNames(Set<String> groupsNames) { invisibleSharedGroups = groupsNames; } /** * Adds a new group to the shared groups list. * * @param sharedGroup The shared group to add to the list of shared groups. */ public void addSharedGroup(Group sharedGroup) { sharedGroups.add(sharedGroup.getName()); invisibleSharedGroups.remove(sharedGroup.getName()); } /** * Adds a new group to the list shared groups that won't be sent to the user. These groups * are for internal use and help track the reason why a roster item has a presence * subscription of type FROM when using shared groups. * * @param sharedGroup The shared group to add to the list of shared groups. */ public void addInvisibleSharedGroup(Group sharedGroup) { invisibleSharedGroups.add(sharedGroup.getName()); } /** * Removes a group from the shared groups list. * * @param sharedGroup The shared group to remove from the list of shared groups. */ public void removeSharedGroup(Group sharedGroup) { sharedGroups.remove(sharedGroup.getName()); invisibleSharedGroups.remove(sharedGroup.getName()); } /** * Returns true if this item belongs to a shared group. Return true even if the item belongs * to a personal group and a shared group. * * @return true if this item belongs to a shared group. */ public boolean isShared() { return !sharedGroups.isEmpty() || !invisibleSharedGroups.isEmpty(); } /** * Returns true if this item belongs ONLY to shared groups. This means that the the item is * considered to be "only shared" if it doesn't belong to a personal group but only to shared * groups. * * @return true if this item belongs ONLY to shared groups. */ public boolean isOnlyShared() { return isShared() && groups.isEmpty(); } /** * Returns the roster ID associated with this particular roster item. A value of zero * means that the roster item is not being persisted in the backend store.<p> * * Databases can use the roster ID as the key in locating roster items. * * @return The roster ID */ public long getID() { return rosterID; } /** * Sets the roster ID associated with this particular roster item. A value of zero * means that the roster item is not being persisted in the backend store.<p> * * Databases can use the roster ID as the key in locating roster items. * * @param rosterID The roster ID. */ public void setID(long rosterID) { this.rosterID = rosterID; } /** * <p>Update the cached item as a copy of the given item.</p> * <p/> * <p>A convenience for getting the item and setting each attribute.</p> * * @param item The item who's settings will be copied into the cached copy * @throws org.jivesoftware.openfire.SharedGroupException if trying to remove shared group. */ public void setAsCopyOf(org.xmpp.packet.Roster.Item item) throws SharedGroupException { setGroups(new LinkedList<String>(item.getGroups())); setNickname(item.getName()); } public int getCachedSize() { int size = jid.toBareJID().length(); size += CacheSizes.sizeOfString(nickname); size += CacheSizes.sizeOfCollection(groups); size += CacheSizes.sizeOfInt(); // subStatus size += CacheSizes.sizeOfInt(); // askStatus size += CacheSizes.sizeOfLong(); // id return size; } public void writeExternal(ObjectOutput out) throws IOException { ExternalizableUtil.getInstance().writeSafeUTF(out, jid.toString()); ExternalizableUtil.getInstance().writeBoolean(out, nickname != null); if (nickname != null) { ExternalizableUtil.getInstance().writeSafeUTF(out, nickname); } ExternalizableUtil.getInstance().writeStrings(out, groups); ExternalizableUtil.getInstance().writeStrings(out, sharedGroups); ExternalizableUtil.getInstance().writeStrings(out, invisibleSharedGroups); ExternalizableUtil.getInstance().writeInt(out, recvStatus.getValue()); ExternalizableUtil.getInstance().writeInt(out, subStatus.getValue()); ExternalizableUtil.getInstance().writeInt(out, askStatus.getValue()); ExternalizableUtil.getInstance().writeLong(out, rosterID); } public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException { jid = new JID(ExternalizableUtil.getInstance().readSafeUTF(in)); if (ExternalizableUtil.getInstance().readBoolean(in)) { nickname = ExternalizableUtil.getInstance().readSafeUTF(in); } this.groups = new LinkedList<String>(); ExternalizableUtil.getInstance().readStrings(in, groups); ExternalizableUtil.getInstance().readStrings(in, sharedGroups); ExternalizableUtil.getInstance().readStrings(in, invisibleSharedGroups); recvStatus = RecvType.getTypeFromInt(ExternalizableUtil.getInstance().readInt(in)); subStatus = SubType.getTypeFromInt(ExternalizableUtil.getInstance().readInt(in)); askStatus = AskType.getTypeFromInt(ExternalizableUtil.getInstance().readInt(in)); rosterID = ExternalizableUtil.getInstance().readLong(in); } }