/* * Copyright (C) 2004-2008 Jive Software. All rights reserved. * * 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.openfire.vcard; import java.util.Collections; import java.util.Iterator; import java.util.Map; import java.util.StringTokenizer; import org.dom4j.Element; import org.jivesoftware.openfire.XMPPServer; import org.jivesoftware.openfire.container.BasicModule; import org.jivesoftware.openfire.disco.ServerFeaturesProvider; import org.jivesoftware.openfire.event.UserEventAdapter; import org.jivesoftware.openfire.event.UserEventDispatcher; import org.jivesoftware.openfire.user.User; import org.jivesoftware.util.AlreadyExistsException; import org.jivesoftware.util.ClassUtils; import org.jivesoftware.util.JiveGlobals; import org.jivesoftware.util.NotFoundException; import org.jivesoftware.util.PropertyEventDispatcher; import org.jivesoftware.util.PropertyEventListener; import org.jivesoftware.util.cache.Cache; import org.jivesoftware.util.cache.CacheFactory; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * Manages VCard information for users. * * @author Matt Tucker */ public class VCardManager extends BasicModule implements ServerFeaturesProvider { private static final Logger Log = LoggerFactory.getLogger(VCardManager.class); private VCardProvider provider; private static VCardManager instance; private EventHandler eventHandler; private Cache<String, Element> vcardCache; public static VCardManager getInstance() { return instance; } /** * Returns the currently-installed VCardProvider. <b>Warning:</b> in virtually all * cases the vcard provider should not be used directly. Instead, the appropriate * methods in VCardManager should be called. Direct access to the vcard provider is * only provided for special-case logic. * * @return the current VCardProvider. */ public static VCardProvider getProvider() { return instance.provider; } public VCardManager() { super("VCard Manager"); String cacheName = "VCard"; vcardCache = CacheFactory.createCache(cacheName); this.eventHandler = new EventHandler(); // Keeps the cache updated in case the vCard action was not performed by VCardManager VCardEventDispatcher.addListener(new VCardListener() { @Override public void vCardCreated(String username, Element vCard) { // Since the vCard could be created by the provider, add it to the cache. vcardCache.put(username, vCard); } @Override public void vCardUpdated(String username, Element vCard) { // Since the vCard could be updated by the provider, update it to the cache. vcardCache.put(username, vCard); } @Override public void vCardDeleted(String username, Element vCard) { // Since the vCard could be delated by the provider, remove it to the cache. vcardCache.remove(username); } }); } /** * Returns the user's vCard information for a given vcard property name. If the property * has no defined text then an empty string will be returned. However, if the property * does not exist then a <tt>null</tt> value will be answered. Advanced user systems can * use vCard information to link to user directory information or store other relevant * user information. * <p> * Note that many elements in the vCard may have the same path so the returned value in that * case will be the first found element. For instance, "ADR:STREET" may be present in * many addresses of the user. Use {@link #getVCard(String)} to get the whole vCard of * the user.</p> * * @param username The username of the user to return his vCard property. * @param name The name of the vcard property to retrieve encoded with ':' to denote * the path. * @return The vCard value found */ public String getVCardProperty(String username, String name) { String answer = null; Element vCardElement = getOrLoadVCard(username); if (vCardElement != null) { // A vCard was found for this user so now look for the correct element Element subElement = null; StringTokenizer tokenizer = new StringTokenizer(name, ":"); while (tokenizer.hasMoreTokens()) { String tok = tokenizer.nextToken(); if (subElement == null) { subElement = vCardElement.element(tok); } else { subElement = subElement.element(tok); } if (subElement == null) { break; } } if (subElement != null) { answer = subElement.getTextTrim(); } } return answer; } /** * Sets the user's vCard information. The new vCard information will be persistent. Advanced * user systems can use vCard information to link to user directory information or store * other relevant user information. * * @param username The username of the user to set his new vCard. * @param vCardElement The DOM element sent by the user as his new vCard. * @throws Exception if an error occurred while storing the new vCard. */ public void setVCard(String username, Element vCardElement) throws Exception { boolean created = false; boolean updated = false; if (provider.isReadOnly()) { throw new UnsupportedOperationException("VCard provider is read-only."); } Element oldVCard = getOrLoadVCard(username); Element newvCard = null; // See if we need to update the vCard or insert a new one. if (oldVCard != null) { // Only update the vCard in the database if the vCard has changed. if (!oldVCard.equals(vCardElement)) { try { newvCard = provider.updateVCard(username, vCardElement); vcardCache.put(username, newvCard); updated = true; } catch (NotFoundException e) { Log.warn("Tried to update a vCard that does not exist", e); newvCard = provider.createVCard(username, vCardElement); vcardCache.put(username, newvCard); created = true; } } } else { try { newvCard = provider.createVCard(username, vCardElement); vcardCache.put(username, newvCard); created = true; } catch (AlreadyExistsException e) { Log.warn("Tried to create a vCard when one already exist", e); newvCard = provider.updateVCard(username, vCardElement); vcardCache.put(username, newvCard); updated = true; } } // Dispatch vCard events if (created) { // Alert listeners that a new vCard has been created VCardEventDispatcher.dispatchVCardCreated(username, newvCard); } else if (updated) { // Alert listeners that a vCard has been updated VCardEventDispatcher.dispatchVCardUpdated(username, newvCard); } } /** * Deletes the user's vCard from the user account. * * @param username The username of the user to delete his vCard. * @throws UnsupportedOperationException If the provider is read-only and the data * cannot be deleted, this exception is thrown */ public void deleteVCard(String username) { if (provider.isReadOnly()) { throw new UnsupportedOperationException("VCard provider is read-only."); } Element oldVCard = getOrLoadVCard(username); if (oldVCard != null) { vcardCache.remove(username); // Delete the property from the DB if it was present in memory provider.deleteVCard(username); // Alert listeners that a vCard has been deleted VCardEventDispatcher.dispatchVCardDeleted(username, oldVCard); } } /** * Returns the vCard of a given user or null if none was defined before. Changes to the * returned vCard will not be stored in the database. Use the returned vCard as a * read-only vCard. * * @param username Username (not full JID) whose vCard to retrieve. * @return the vCard of a given user. */ public Element getVCard(String username) { Element vCardElement = getOrLoadVCard(username); return vCardElement == null ? null : vCardElement.createCopy(); } private Element getOrLoadVCard(String username) { Element vCardElement = vcardCache.get(username); if (vCardElement == null) { vCardElement = provider.loadVCard(username); if (vCardElement != null) { vcardCache.put(username, vCardElement); } } return vCardElement; } @Override public void initialize(XMPPServer server) { instance = this; // Convert XML based provider setup to Database based JiveGlobals.migrateProperty("provider.vcard.className"); // Load a VCard provider. String className = JiveGlobals.getProperty("provider.vcard.className", DefaultVCardProvider.class.getName()); try { Class c = ClassUtils.forName(className); provider = (VCardProvider) c.newInstance(); } catch (Exception e) { Log.error("Error loading vcard provider: " + className, e); provider = new DefaultVCardProvider(); } } @Override public void start() { // Add this module as a user event listener so we can delete // all user properties when a user is deleted if (!provider.isReadOnly()) { UserEventDispatcher.addListener(eventHandler); } // Detect when a new vcard provider class is set PropertyEventListener propListener = new PropertyEventListener() { @Override public void propertySet(String property, Map params) { if ("provider.vcard.className".equals(property)) { initialize(XMPPServer.getInstance()); } } @Override public void propertyDeleted(String property, Map params) { //Ignore } @Override public void xmlPropertySet(String property, Map params) { //Ignore } @Override public void xmlPropertyDeleted(String property, Map params) { //Ignore } }; PropertyEventDispatcher.addListener(propListener); } @Override public void stop() { // Remove this module as a user event listener UserEventDispatcher.removeListener(eventHandler); } /** * Resets the manager state. The cache where loaded vCards are stored will be flushed. */ public void reset() { vcardCache.clear(); } @Override public Iterator<String> getFeatures() { return Collections.singleton("vcard-temp").iterator(); } private class EventHandler extends UserEventAdapter { @Override public void userDeleting(User user, Map params) { try { deleteVCard(user.getUsername()); } catch (UnsupportedOperationException ue) { /* Do Nothing */ } } } }