All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jivesoftware.openfire.user.UserManager Maven / Gradle / Ivy

The newest version!
/*
 * 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.user;

import gnu.inet.encoding.Stringprep;
import gnu.inet.encoding.StringprepException;

import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;

import org.dom4j.Element;
import org.jivesoftware.openfire.XMPPServer;
import org.jivesoftware.openfire.event.UserEventDispatcher;
import org.jivesoftware.openfire.event.UserEventListener;
import org.jivesoftware.openfire.user.property.DefaultUserPropertyProvider;
import org.jivesoftware.openfire.user.property.UserPropertyProvider;
import org.jivesoftware.util.ClassUtils;
import org.jivesoftware.util.JiveGlobals;
import org.jivesoftware.util.PropertyEventDispatcher;
import org.jivesoftware.util.PropertyEventListener;
import org.jivesoftware.util.StringUtils;
import org.jivesoftware.util.cache.Cache;
import org.jivesoftware.util.cache.CacheFactory;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.xmpp.component.IQResultListener;
import org.xmpp.packet.IQ;
import org.xmpp.packet.JID;

/**
 * Manages users, including loading, creating and deleting.
 *
 * @author Matt Tucker
 * @see User
 */
public class UserManager implements IQResultListener {

    private static final Logger Log = LoggerFactory.getLogger(UserManager.class);

    // Wrap this guy up so we can mock out the UserManager class.
    private static class UserManagerContainer {
        private static UserManager instance = new UserManager();
    }

    /**
     * Returns the currently-installed UserProvider. Warning: in virtually all
     * cases the user provider should not be used directly. Instead, the appropriate
     * methods in UserManager should be called. Direct access to the user provider is
     * only provided for special-case logic.
     *
     * @return the current UserProvider.
     */
    public static UserProvider getUserProvider() {
        return UserManagerContainer.instance.provider;
    }

    /**
     * Returns the currently-installed UserPropertyProvider.
     *
     * Warning: in virtually all cases the user property provider should not be used directly. Instead, use the
     * Map returned by {@link User#getProperties()} to create, read, update or delete user properties. Failure to do so
     * is likely to result in inconsistent data behavior and race conditions. Direct access to the user property
     * provider is only provided for special-case logic.
     *
     * @return the current UserPropertyProvider.
     * @see User#getProperties
     */
    public static UserPropertyProvider getUserPropertyProvider() {
        return UserManagerContainer.instance.propertyProvider;
    }

    /**
     * Returns a singleton UserManager instance.
     *
     * @return a UserManager instance.
     */
    public static UserManager getInstance() {
        return UserManagerContainer.instance;
    }

    /** Cache of local users. */
    private Cache userCache;
    /** Cache if a local or remote user exists. */
    private Cache remoteUsersCache;
    private UserProvider provider;
    private UserPropertyProvider propertyProvider;

    private UserManager() {
        // Initialize caches.
        userCache = CacheFactory.createCache("User");
        remoteUsersCache = CacheFactory.createCache("Remote Users Existence");

        // Load a user & property provider.
        initProvider();
        initPropertyProvider();

        // Detect when a new auth provider class is set
        PropertyEventListener propListener = new PropertyEventListener() {
            @Override
            public void propertySet(String property, Map params) {
                if ("provider.user.className".equals(property)) {
                    initProvider();
                }
                if ("provider.userproperty.className".equals(property)) {
                    initPropertyProvider();
                }
            }

            @Override
            public void propertyDeleted(String property, Map params) {
                if ("provider.user.className".equals(property)) {
                    initProvider();
                }
                if ("provider.userproperty.className".equals(property)) {
                    initPropertyProvider();
                }
            }

            @Override
            public void xmlPropertySet(String property, Map params) {
                //Ignore
            }

            @Override
            public void xmlPropertyDeleted(String property, Map params) {
                //Ignore
            }
        };
        PropertyEventDispatcher.addListener(propListener);

        UserEventListener userListener = new UserEventListener() {
            @Override
            public void userCreated(User user, Map params) {
                // Since the user could be created by the provider, add it possible again
                userCache.put(user.getUsername(), user);
            }

            @Override
            public void userDeleting(User user, Map params) {
                // Since the user could be deleted by the provider, remove it possible again
                userCache.remove(user.getUsername());
            }

            @Override
            public void userModified(User user, Map params) {
                // Set object again in cache. This is done so that other cluster nodes
                // get refreshed with latest version of the user
                userCache.put(user.getUsername(), user);
            }
        };
        UserEventDispatcher.addListener(userListener);
    }

    /**
     * Creates a new User. Required values are username and password. The email address
     * and name can optionally be null, unless the UserProvider deems that
     * either of them are required.
     *
     * @param username the new and unique username for the account.
     * @param password the password for the account (plain text).
     * @param name the name of the user, which can be null unless the UserProvider
     *      deems that it's required.
     * @param email the email address to associate with the new account, which can
     *      be null, unless the UserProvider deems that it's required.
     * @return a new User.
     * @throws UserAlreadyExistsException if the username already exists in the system.
     * @throws UnsupportedOperationException if the provider does not support the
     *      operation.
     */
    public User createUser(String username, String password, String name, String email)
            throws UserAlreadyExistsException
    {
        if (provider.isReadOnly()) {
            throw new UnsupportedOperationException("User provider is read-only.");
        }
        if (username == null || username.isEmpty()) {
            throw new IllegalArgumentException("Null or empty username.");
        }
        if (password == null || password.isEmpty()) {
            throw new IllegalArgumentException("Null or empty password.");
        }
        // Make sure that the username is valid.
        try {
            username = Stringprep.nodeprep(username);
        }
        catch (StringprepException se) {
            throw new IllegalArgumentException("Invalid username: " + username,  se);
        }
        if (provider.isNameRequired() && (name == null || name.matches("\\s*"))) {
            throw new IllegalArgumentException("Invalid or empty name specified with provider that requires name. User: "
                                                + username + " Name: " + name);
        }
        if (provider.isEmailRequired() && !StringUtils.isValidEmailAddress(email)) {
            throw new IllegalArgumentException("Invalid or empty email address specified with provider that requires email address. User: "
                                                + username + " Email: " + email);
        }
        User user = provider.createUser(username, password, name, email);
        userCache.put(username, user);

        // Fire event.
        Map params = Collections.emptyMap();
        UserEventDispatcher.dispatchEvent(user, UserEventDispatcher.EventType.user_created, params);

        return user;
    }

    /**
     * Deletes a user (optional operation).
     *
     * @param user the user to delete.
     */
    public void deleteUser(User user) {
        if (provider.isReadOnly()) {
            throw new UnsupportedOperationException("User provider is read-only.");
        }

        String username = user.getUsername();
        // Make sure that the username is valid.
        try {
            /*username =*/ Stringprep.nodeprep(username);
        }
        catch (StringprepException se) {
            throw new IllegalArgumentException("Invalid username: " + username,  se);
        }

        // Fire event.
        Map params = Collections.emptyMap();
        UserEventDispatcher.dispatchEvent(user, UserEventDispatcher.EventType.user_deleting, params);

        provider.deleteUser(user.getUsername());
        // Remove the user from cache.
        userCache.remove(user.getUsername());
    }

    /**
     * Returns the User specified by username.
     *
     * @param username the username of the user.
     * @return the User that matches username.
     * @throws UserNotFoundException if the user does not exist.
     */
    public User getUser(String username) throws UserNotFoundException {
        if (username == null) {
            throw new UserNotFoundException("Username cannot be null");
        }
        // Make sure that the username is valid.
        username = username.trim().toLowerCase();
        User user = userCache.get(username);
        if (user == null) {
            synchronized (username.intern()) {
                user = userCache.get(username);
                if (user == null) {
                    user = provider.loadUser(username);
                    userCache.put(username, user);
                }
            }
        }
        return user;
    }

    /**
     * Returns the total number of users in the system.
     *
     * @return the total number of users.
     */
    public int getUserCount() {
        return provider.getUserCount();
    }

    /**
     * Returns an unmodifiable Collection of all users in the system.
     *
     * @return an unmodifiable Collection of all users.
     */
    public Collection getUsers() {
        return provider.getUsers();
    }

    /**
     * Returns an unmodifiable Collection of usernames of all users in the system.
     *
     * @return an unmodifiable Collection of all usernames in the system.
     */
    public Collection getUsernames() {
        return provider.getUsernames();
    }

    /**
     * Returns an unmodifiable Collection of all users starting at startIndex
     * with the given number of results. This is useful to support pagination in a GUI
     * where you may only want to display a certain number of results per page. It is
     * possible that the number of results returned will be less than that specified
     * by numResults if numResults is greater than the number of
     * records left to display.
     *
     * @param startIndex the beginning index to start the results at.
     * @param numResults the total number of results to return.
     * @return a Collection of users in the specified range.
     */
    public Collection getUsers(int startIndex, int numResults) {
        return provider.getUsers(startIndex, numResults);
    }

    /**
     * Returns the set of fields that can be used for searching for users. Each field
     * returned must support wild-card and keyword searching. For example, an
     * implementation might send back the set {"Username", "Name", "Email"}. Any of
     * those three fields can then be used in a search with the
     * {@link #findUsers(Set,String)} method.

* * This method should throw an UnsupportedOperationException if this * operation is not supported by the backend user store. * * @return the valid search fields. * @throws UnsupportedOperationException if the provider does not * support the operation (this is an optional operation). */ public Set getSearchFields() throws UnsupportedOperationException { return provider.getSearchFields(); } /** * Searches for users based on a set of fields and a query string. The fields must * be taken from the values returned by {@link #getSearchFields()}. The query can * include wildcards. For example, a search on the field "Name" with a query of "Ma*" * might return user's with the name "Matt", "Martha" and "Madeline".

* * This method throws an UnsupportedOperationException if this operation is * not supported by the user provider. * * @param fields the fields to search on. * @param query the query string. * @return a Collection of users that match the search. * @throws UnsupportedOperationException if the provider does not * support the operation (this is an optional operation). */ public Collection findUsers(Set fields, String query) throws UnsupportedOperationException { return provider.findUsers(fields, query); } /** * Searches for users based on a set of fields and a query string. The fields must * be taken from the values returned by {@link #getSearchFields()}. The query can * include wildcards. For example, a search on the field "Name" with a query of "Ma*" * might return user's with the name "Matt", "Martha" and "Madeline".

* * The startIndex and numResults parameters are used to page through search * results. For example, if the startIndex is 0 and numResults is 10, the first * 10 search results will be returned. Note that numResults is a request for the * number of results to return and that the actual number of results returned * may be fewer.

* * This method should throw an UnsupportedOperationException if this * operation is not supported by the backend user store. * * @param fields the fields to search on. * @param query the query string. * @param startIndex the starting index in the search result to return. * @param numResults the number of users to return in the search result. * @return a Collection of users that match the search. * @throws UnsupportedOperationException if the provider does not * support the operation (this is an optional operation). */ public Collection findUsers(Set fields, String query, int startIndex, int numResults) throws UnsupportedOperationException { return provider.findUsers(fields, query, startIndex, numResults); } /** * Returns true if the specified local username belongs to a registered local user. * * @param username to username of the user to check it it's a registered user. * @return true if the specified JID belongs to a local registered user. */ public boolean isRegisteredUser(String username) { if (username == null || "".equals(username)) { return false; } try { getUser(username); return true; } catch (UserNotFoundException e) { return false; } } /** * Returns true if the specified JID belongs to a local or remote registered user. For * remote users (i.e. domain does not match local domain) a disco#info request is going * to be sent to the bare JID of the user. * * @param user to JID of the user to check it it's a registered user. * @return true if the specified JID belongs to a local or remote registered user. */ public boolean isRegisteredUser(JID user) { XMPPServer server = XMPPServer.getInstance(); if (server.isLocal(user)) { try { getUser(user.getNode()); return true; } catch (UserNotFoundException e) { return false; } } else { // Look up in the cache using the full JID Boolean isRegistered = remoteUsersCache.get(user.toString()); if (isRegistered == null) { // Check if the bare JID of the user is cached isRegistered = remoteUsersCache.get(user.toBareJID()); if (isRegistered == null) { // No information is cached so check user identity and cache it // A disco#info is going to be sent to the bare JID of the user. This packet // is going to be handled by the remote server. IQ iq = new IQ(IQ.Type.get); iq.setFrom(server.getServerInfo().getXMPPDomain()); iq.setTo(user.toBareJID()); iq.setChildElement("query", "http://jabber.org/protocol/disco#info"); // Send the disco#info request to the remote server. The reply will be // processed by the IQResultListener (interface that this class implements) server.getIQRouter().addIQResultListener(iq.getID(), this); synchronized (user.toBareJID().intern()) { server.getIQRouter().route(iq); // Wait for the reply to be processed. Time out in 1 minute. try { user.toBareJID().intern().wait(60000); } catch (InterruptedException e) { // Do nothing } } // Get the discovered result isRegistered = remoteUsersCache.get(user.toBareJID()); if (isRegistered == null) { // Disco failed for some reason (i.e. we timed out before getting a result) // so assume that user is not anonymous and cache result isRegistered = Boolean.FALSE; remoteUsersCache.put(user.toString(), isRegistered); } } } return isRegistered; } } @Override public void receivedAnswer(IQ packet) { JID from = packet.getFrom(); // Assume that the user is not a registered user Boolean isRegistered = Boolean.FALSE; // Analyze the disco result packet if (IQ.Type.result == packet.getType()) { Element child = packet.getChildElement(); if (child != null) { for (Iterator it=child.elementIterator("identity"); it.hasNext();) { Element identity = (Element) it.next(); String accountType = identity.attributeValue("type"); if ("registered".equals(accountType) || "admin".equals(accountType)) { isRegistered = Boolean.TRUE; break; } } } } // Update cache of remote registered users remoteUsersCache.put(from.toBareJID(), isRegistered); // Wake up waiting thread synchronized (from.toBareJID().intern()) { from.toBareJID().intern().notifyAll(); } } @Override public void answerTimeout(String packetId) { Log.warn("An answer to a previously sent IQ stanza was never received. Packet id: " + packetId); } private void initProvider() { // Convert XML based provider setup to Database based JiveGlobals.migrateProperty("provider.user.className"); String className = JiveGlobals.getProperty("provider.user.className", "org.jivesoftware.openfire.user.DefaultUserProvider"); // Check if we need to reset the provider class if (provider == null || !className.equals(provider.getClass().getName())) { try { Class c = ClassUtils.forName(className); provider = (UserProvider) c.newInstance(); } catch (Exception e) { Log.error("Error loading user provider: " + className, e); provider = new DefaultUserProvider(); } } } private void initPropertyProvider() { // Convert XML based provider setup to Database based JiveGlobals.migrateProperty("provider.userproperty.className"); String className = JiveGlobals.getProperty("provider.userproperty.className", "org.jivesoftware.openfire.user.property.DefaultUserPropertyProvider"); // Check if we need to reset the provider class if (propertyProvider == null || !className.equals(propertyProvider.getClass().getName())) { try { Class c = ClassUtils.forName(className); propertyProvider = (UserPropertyProvider) c.newInstance(); } catch (Exception e) { Log.error("Error loading user property provider: " + className, e); propertyProvider = new DefaultUserPropertyProvider(); } } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy