Commit 503ac87f authored by Derek DeMoro's avatar Derek DeMoro Committed by derek

Need to ensure backwards compatibilty with Sasl Mechs and AuthorizationPolicies.

git-svn-id: http://svn.igniterealtime.org/svn/repos/openfire/branches@8265 b35dd754-fafc-0310-a699-88a17e54d16e
parent c2f43c2c
/**
* $RCSfile$
* $Revision: $
* $Date: 2006-04-07 09:28:54 -0500 (Fri, 07 Apr 2006) $
*
* Copyright (C) 2004 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.
*/
package org.jivesoftware.openfire.ldap;
import org.jivesoftware.util.JiveGlobals;
import org.jivesoftware.openfire.sasl.AbstractAuthorizationProvider;
import org.jivesoftware.openfire.sasl.AuthorizationPolicy;
import org.xmpp.packet.JID;
import javax.naming.directory.Attribute;
import javax.naming.directory.Attributes;
import javax.naming.directory.DirContext;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Enumeration;
/**
* Provider for authorization using LDAP. Checks if the authenticated
* principal is in the user's LDAP object using the authorizeField
* from the <tt>openfire.xml</tt> file. An entry in that file would
* look like the following:
*
* <pre>
* &lt;ldap&gt;
* &lt;authorizeField&gt; k5login &lt;/authorizeField&gt;
* &lt;/ldap&gt;</pre>
*
* This implementation requires that LDAP be configured, obviously.
*
* @author Jay Kline
*/
public class LdapAuthorizationProvider extends AbstractAuthorizationProvider implements AuthorizationPolicy {
private LdapManager manager;
private String usernameField;
private String authorizeField;
public LdapAuthorizationProvider() {
manager = LdapManager.getInstance();
usernameField = manager.getUsernameField();
authorizeField = JiveGlobals.getXMLProperty("ldap.authorizeField", "k5login");
}
/**
* Returns if the principal is explicity authorized to the JID, throws
* an UnauthorizedException otherwise
*
* @param username The username requested.import org.jivesoftware.openfire.ldap.*;
* @param principal The principal requesting the username.
*
*/
public boolean authorize(String username, String principal) {
return getAuthorized(username).contains(principal);
}
/**
* Returns a String Collection of principals that are authorized to use
* the named user.
*
* @param username the username.
* @return A String Collection of principals that are authorized.
*/
public Collection<String> getAuthorized(String username) {
// Un-escape Node
username = JID.unescapeNode(username);
Collection<String> authorized = new ArrayList<String>();
DirContext ctx = null;
try {
String userDN = manager.findUserDN(username);
// Load record.
String[] attributes = new String[]{
usernameField,
authorizeField
};
ctx = manager.getContext();
Attributes attrs = ctx.getAttributes(userDN, attributes);
Attribute authorizeField_a = attrs.get(manager.getNameField());
if (authorizeField_a != null) {
for(Enumeration e = authorizeField_a.getAll(); e.hasMoreElements();) {
authorized.add((String)e.nextElement());
}
}
return authorized;
}
catch (Exception e) {
// Ignore.
}
finally {
try {
if (ctx != null) {
ctx.close();
}
}
catch (Exception ignored) {
// Ignore.
}
}
return authorized;
}
/**
* Returns false, this implementation is not writeable.
*
* @return False.
*/
public boolean isWritable() {
return false;
}
/**
* Always throws UnsupportedOperationException.
*
* @param username The username.
* @param principal The principal authorized to use the named user.
* @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated.
*/
public void addAuthorized(String username, String principal) throws UnsupportedOperationException {
throw new UnsupportedOperationException();
}
/**
* Always throws UnsupportedOperationException.
*
* @param username The username.
* @param principals The Collection of principals authorized to use the named user.
*/
public void addAuthorized(String username, Collection<String> principals) throws UnsupportedOperationException {
throw new UnsupportedOperationException();
}
/**
* Always throws UnsupportedOperationException.
*
* @param username The username.
* @param principals The Collection of principals authorized to use the named user.
* @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated.
*/
public void setAuthorized(String username, Collection<String> principals) throws UnsupportedOperationException {
throw new UnsupportedOperationException();
}
/**
* Returns the short name of the Policy
*
* @return The short name of the Policy
*/
public String name() {
return "LDAP Authorization Provider";
}
/**
* Returns a description of the Policy
*
* @return The description of the Policy.
*/
public String description() {
return "Provider for authorization using LDAP. Checks if the authenticated principal is in the user's LDAP object using the authorizeField property.";
}
}
\ No newline at end of file
...@@ -15,14 +15,14 @@ import java.util.Collection; ...@@ -15,14 +15,14 @@ import java.util.Collection;
/** /**
* Provider for authorization. Unlike the AbstractAuthorizationPolicy * Provider for authorization. Unlike the AbstractAuthorizationPolicy
* class, this is intended for classes that need a more "heavyweight" * class, this is intended for classes that need a more "heavyweight"
* solution, often that requires consulting some storage or external * solution, often that requires consulting some storage or external
* entity about each specific case. This class allows individual mappings * entity about each specific case. This class allows individual mappings
* between authenticated principals and usernames, and if the storage * between authenticated principals and usernames, and if the storage
* mechanism allows it, management of those mappings. * mechanism allows it, management of those mappings.
* *
* Users that wish to integrate with their own authorization * Users that wish to integrate with their own authorization
* system must extend this class and implement the * system must extend this class and implement the
* AuthorizationProvider interface then register the class * AuthorizationProvider interface then register the class
* with Openfire in the <tt>openfire.xml</tt> file. An entry * with Openfire in the <tt>openfire.xml</tt> file. An entry
* in that file would look like the following: * in that file would look like the following:
...@@ -46,7 +46,7 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli ...@@ -46,7 +46,7 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli
* @return true is the user is authorized to be principal * @return true is the user is authorized to be principal
*/ */
public abstract boolean authorize(String username, String principal); public abstract boolean authorize(String username, String principal);
/** /**
* Returns a String Collection of principals that are authorized to use * Returns a String Collection of principals that are authorized to use
* the named user. * the named user.
...@@ -55,16 +55,16 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli ...@@ -55,16 +55,16 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli
* @return A String Collection of principals that are authorized. * @return A String Collection of principals that are authorized.
*/ */
public abstract Collection<String> getAuthorized(String username); public abstract Collection<String> getAuthorized(String username);
/** /**
* Returns true if this AuthorizationProvider supports changing the * Returns true if this AuthorizationProvider supports changing the
* list of authorized principals for users. * list of authorized principals for users.
* *
* @return true if updating the list of authorized principals is * @return true if updating the list of authorized principals is
* supported by this AuthorizationProvider. * supported by this AuthorizationProvider.
*/ */
public abstract boolean isWritable(); public abstract boolean isWritable();
/** /**
* Add a single authorized principal to use the named user. * Add a single authorized principal to use the named user.
* *
...@@ -73,7 +73,7 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli ...@@ -73,7 +73,7 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli
* @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated. * @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated.
*/ */
public abstract void addAuthorized(String username, String principal) throws UnsupportedOperationException; public abstract void addAuthorized(String username, String principal) throws UnsupportedOperationException;
/** /**
* Add a Collection of users authorized to use the named user. * Add a Collection of users authorized to use the named user.
* *
...@@ -82,7 +82,7 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli ...@@ -82,7 +82,7 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli
* @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated. * @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated.
*/ */
public abstract void addAuthorized(String username, Collection<String> principals) throws UnsupportedOperationException; public abstract void addAuthorized(String username, Collection<String> principals) throws UnsupportedOperationException;
/** /**
* Set the users authorized to use the named user. All existing principals listed * Set the users authorized to use the named user. All existing principals listed
* will be removed. * will be removed.
...@@ -99,11 +99,11 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli ...@@ -99,11 +99,11 @@ public abstract class AbstractAuthorizationProvider implements AuthorizationPoli
* @return The short name of the Policy * @return The short name of the Policy
*/ */
public abstract String name(); public abstract String name();
/** /**
* Returns a description of the Policy * Returns a description of the Policy
* *
* @return The description of the Policy. * @return The description of the Policy.
*/ */
public abstract String description(); public abstract String description();
} }
\ No newline at end of file
...@@ -11,14 +11,13 @@ ...@@ -11,14 +11,13 @@
package org.jivesoftware.openfire.sasl; package org.jivesoftware.openfire.sasl;
import org.jivesoftware.openfire.user.UserAlreadyExistsException;
import org.jivesoftware.openfire.user.UserManager;
import org.jivesoftware.openfire.user.UserNotFoundException;
import org.jivesoftware.openfire.user.UserProvider;
import org.jivesoftware.util.ClassUtils; import org.jivesoftware.util.ClassUtils;
import org.jivesoftware.util.JiveGlobals; import org.jivesoftware.util.JiveGlobals;
import org.jivesoftware.util.Log; import org.jivesoftware.util.Log;
import org.jivesoftware.util.StringUtils;
import java.util.ArrayList;
import java.util.Collection;
import java.util.StringTokenizer;
/** /**
* Manages the AuthorizationProvider objects. * Manages the AuthorizationProvider objects.
...@@ -43,28 +42,48 @@ import org.jivesoftware.util.StringUtils; ...@@ -43,28 +42,48 @@ import org.jivesoftware.util.StringUtils;
*/ */
public class AuthorizationManager { public class AuthorizationManager {
private static ArrayList<AuthorizationPolicy> authorizationPolicies = new ArrayList<AuthorizationPolicy>();
private static AuthorizationManager instance = new AuthorizationManager(); private static AuthorizationManager instance = new AuthorizationManager();
private static AuthorizationPolicy authorizationProvider;
static { static {
String className = JiveGlobals.getXMLProperty("provider.authorization.className"); String classList = JiveGlobals.getXMLProperty("provider.authorization.classList");
if (className != null) { if (classList != null) {
try { StringTokenizer st = new StringTokenizer(classList, " ,\t\n\r\f");
Class c_provider = ClassUtils.forName(className); while (st.hasMoreTokens()) {
AuthorizationPolicy provider = (AuthorizationPolicy)(c_provider.newInstance()); String s_provider = st.nextToken();
Log.debug("AuthorizationManager: Loaded " + provider); try {
authorizationProvider = provider; Class c_provider = ClassUtils.forName(s_provider);
} AuthorizationPolicy provider =
catch (Exception e) { (AuthorizationPolicy)(c_provider.newInstance());
Log.error("Error loading AuthorizationProvider: " + className + "\n" + e); Log.debug("AuthorizationManager: Loaded " + s_provider);
authorizationPolicies.add(provider);
}
catch (Exception e) {
Log.error("Error loading AuthorizationProvider: " + s_provider + "\n" + e);
}
} }
} }
if (authorizationPolicies.isEmpty()) {
if (authorizationProvider == null) {
Log.debug("No AuthorizationProvider's found. Loading DefaultAuthorizationPolicy"); Log.debug("No AuthorizationProvider's found. Loading DefaultAuthorizationPolicy");
// authorizationPolicies.add(new GSSAPIAuthorizationProvider());
} }
} }
private AuthorizationManager() {
}
/**
* Returns the currently-installed AuthorizationProvider. Warning: You
* should not be calling the AuthorizationProvider directly to perform
* authorizations, it will not take into account the policy selected in
* the <tt>openfire.xml</tt>. Use @see{authorize} in this class, instead.
*
* @return the current AuthorizationProvider.
*/
public static Collection<AuthorizationPolicy> getAuthorizationPolicies() {
return authorizationPolicies;
}
/** /**
* Returns a singleton AuthorizationManager instance. * Returns a singleton AuthorizationManager instance.
...@@ -75,53 +94,19 @@ public class AuthorizationManager { ...@@ -75,53 +94,19 @@ public class AuthorizationManager {
return instance; return instance;
} }
private AuthorizationManager() {
}
/** /**
* Authorize the authenticated used to the requested username. This uses the * Authorize the authenticated used to the requested username. This uses the
* GSSAPIAuthProvider if it is the specified provider. * selected the selected AuthenticationProviders.
* *
* @param username the username.
* @param principal the principal.
* @return true if the user is authorized. * @return true if the user is authorized.
*/ */
public static boolean authorize(String username, String principal) {
boolean authorized = false;
if (authorizationProvider != null) {
authorized = authorizationProvider.authorize(username, principal);
if (!authorized) {
return false;
}
}
// If the user is authorized, we want to check if the user is listed as a public static boolean authorize(String authorId, String authenId) {
// member of the <code>UserProvider</code>, and if not, add the user (if writable). for (AuthorizationPolicy ap : authorizationPolicies) {
// See if the user exists in the database. If not, automatically create them. if (ap.authorize(authorId, authenId)) {
UserManager userManager = UserManager.getInstance();
try {
userManager.getUser(username);
}
catch (UserNotFoundException userNotFound) {
try {
Log.debug("Automatically creating new user account for " + username);
// Create user; use a random password for better safety in the future.
// Note that we have to go to the user provider directly -- because the
// provider is read-only, UserManager will usually deny access to createUser.
UserProvider userProvider = UserManager.getUserProvider();
if (userProvider.isReadOnly()) {
Log.error("Error: Unable to add the user. The UserProvider is not writable.");
}
UserManager.getUserProvider().createUser(username, StringUtils.randomString(8),
null, null);
return true; return true;
} }
catch (UserAlreadyExistsException uaee) {
// Ignore.
}
} }
return true; return false;
} }
} }
/**
* $RCSfile$
* $Revision: $
* $Date: 2006-04-20 10:46:24 -0500 (Thu, 20 Apr 2006) $
*
* Copyright (C) 2004 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.
*/
package org.jivesoftware.openfire.sasl;
import org.jivesoftware.openfire.auth.UnauthorizedException;
/**
* Provider interface for authorization policy. Users that wish to integrate with
* their own authorization system must implement this class and then register
* the implementation with Openfire in the <tt>openfire.xml</tt>
* file. An entry in that file would look like the following:
*
* <pre>
* &lt;provider&gt;
* &lt;authorizationpolicy&gt;
* &lt;className&gt;com.foo.auth.CustomPolicyProvider&lt;/className&gt;
* &lt;/authorizationpolicy&gt;
* &lt;/provider&gt;</pre>
*
* @author Jay Kline
*/
public interface AuthorizationPolicyProvider {
/**
* Returns if the principal is explicity authorized to the JID, throws
* an UnauthorizedException otherwise
*
* @param username The username requested.
* @param principal The principal requesting the username.
* @throws UnauthorizedException
*/
public void authorize(String username, String principal) throws UnauthorizedException;
}
\ No newline at end of file
/**
* $RCSfile$
* $Revision: $
* $Date: 2006-04-07 09:28:54 -0500 (Fri, 07 Apr 2006) $
*
* Copyright (C) 2004 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.
*/
package org.jivesoftware.openfire.sasl;
import org.jivesoftware.database.DbConnectionManager;
import org.jivesoftware.util.LocaleUtils;
import org.jivesoftware.util.Log;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.util.ArrayList;
import java.util.Collection;
/**
* Provider for authorization using the default storage database. Checks
* if the authenticated principal is in the user's list of authorized
* principals.
*
* @author Jay Kline
*/
public class DefaultAuthorizationProvider extends AbstractAuthorizationProvider
implements AuthorizationPolicy {
private static final String MATCH_AUTHORIZED =
"SELECT username FROM jiveSASLAuthorized WHERE username=? AND authorized=?";
private static final String GET_AUTHORIZED =
"SELECT authorized FROM jiveSASLAuthorized WHERE username=?";
private static final String INSERT_AUTHORIZED =
"INSERT into jiveSASLAuthorized (username,authorized) VALUES (?,?)";
private static final String DELETE_AUTHORIZED =
"DELETE FROM jiveSASLAuthorized WHERE username=? AND authorized=?";
private static final String DELETE_USER = "DELETE FROM jiveSASLAuthorized WHERE username=?";
/**
* Returns true if the principal is explicity authorized to the JID
*
* @param username The username requested.
* @param principal The principal requesting the username.
* @return true is the user is authorized to be principal
*/
public boolean authorize(String username, String principal) {
Connection con = null;
PreparedStatement pstmt = null;
try {
con = DbConnectionManager.getConnection();
pstmt = con.prepareStatement(MATCH_AUTHORIZED);
pstmt.setString(1, username);
pstmt.setString(2, principal);
ResultSet rs = pstmt.executeQuery();
return rs.next();
}
catch (Exception e) {
return false;
}
finally {
try { if (pstmt != null) { pstmt.close(); } }
catch (Exception e) { Log.error(e); }
try { if (con != null) { con.close(); } }
catch (Exception e) { Log.error(e); }
}
// not reachable
//return false;
}
/**
* Returns a String Collection of principals that are authorized to use
* the named user.
*
* @param username The username.
* @return A String Collection of principals that are authorized.
*/
public Collection<String> getAuthorized(String username) {
Connection con = null;
PreparedStatement pstmt = null;
Collection<String> authorized = new ArrayList<String>();
try {
con = DbConnectionManager.getConnection();
pstmt = con.prepareStatement(GET_AUTHORIZED);
pstmt.setString(1, username);
ResultSet rs = pstmt.executeQuery();
while (rs.next()) {
authorized.add(rs.getString("authorized"));
}
return authorized;
} catch (Exception e) {
return new ArrayList<String>();
}
finally {
try { if (pstmt != null) { pstmt.close(); } }
catch (Exception e) { Log.error(e); }
try { if (con != null) { con.close(); } }
catch (Exception e) { Log.error(e); }
}
}
/**
* Returns true.
*
* @return true
*/
public boolean isWritable() {
return true;
}
/**
* Add a single authorized principal to use the named user.
*
* @param username The username.
* @param principal The principal authorized to use the named user.
* @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated.
*/
public void addAuthorized(String username, String principal)
throws UnsupportedOperationException {
if (authorize(username, principal)) {
// Already exists
return;
}
Connection con = null;
PreparedStatement pstmt = null;
try {
con = DbConnectionManager.getConnection();
pstmt = con.prepareStatement(INSERT_AUTHORIZED);
pstmt.setString(1, username);
pstmt.setString(2, principal);
pstmt.execute();
}
catch (Exception e) {
Log.error(LocaleUtils.getLocalizedString("admin.error"), e);
} finally {
try { if (pstmt != null) { pstmt.close(); } }
catch (Exception e) { Log.error(e); }
try { if (con != null) { con.close(); } }
catch (Exception e) { Log.error(e); }
}
}
/**
* Add a Collection of users authorized to use the named user.
*
* @param username The username.
* @param principals The Collection of principals authorized to use the named user.
* @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated.
*/
public void addAuthorized(String username, Collection<String> principals)
throws UnsupportedOperationException {
for (String principal : principals) {
addAuthorized(username, principal);
}
}
/**
* Set the users authorized to use the named user. All existing principals listed
* will be removed.
*
* @param username The username.
* @param principals The Collection of principals authorized to use the named user.
* @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated.
*/
public void setAuthorized(String username, Collection<String> principals)
throws UnsupportedOperationException {
Connection con = null;
PreparedStatement pstmt = null;
try {
con = DbConnectionManager.getConnection();
pstmt = con.prepareStatement(DELETE_USER);
pstmt.setString(1, username);
pstmt.execute();
addAuthorized(username, principals);
}
catch (Exception e) {
Log.error(LocaleUtils.getLocalizedString("admin.error"), e);
}
finally {
try { if (pstmt != null) { pstmt.close(); } }
catch (Exception e) { Log.error(e); }
try { if (con != null) { con.close(); } }
catch (Exception e) { Log.error(e); }
}
}
/**
* Returns the short name of the Policy
*
* @return The short name of the Policy
*/
public String name() {
return "Default Provider";
}
/**
* Returns a description of the Policy
*
* @return The description of the Policy.
*/
public String description() {
return "Provider for authorization using the default storage database.";
}
}
\ No newline at end of file
/**
* $RCSfile$
* $Revision: $
* $Date: 2006-04-20 10:46:24 -0500 (Thu, 20 Apr 2006) $
*
* Copyright (C) 2004 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.
*/
package org.jivesoftware.openfire.sasl;
/**
* This policy will authorize any principal who's username matches exactly
* the username of the JID. This means when cross realm authentication is
* allowed, user@REALM_A.COM and user@REALM_B.COM could both authorize as
* user@servername, so there is some risk here. But if usernames across the
*
* @author Jay Kline
*/
public class LooseAuthorizationPolicy implements AuthorizationPolicy {
/**
* Returns true if the principal is explicity authorized to the JID
*
* @param username The username requested.
* @param principal The principal requesting the username.
* @return true is the user is authorized to be principal
*/
public boolean authorize(String username, String principal) {
return (principal.startsWith(username + "@"));
}
/**
* Returns the short name of the Policy
*
* @return The short name of the Policy
*/
public String name() {
return "Loose Authorization Provider";
}
/**
* Returns a description of the Policy
*
* @return The description of the Policy.
*/
public String description() {
return "This policy will authorize any principal who's username matches exactly the username of the JID. This means when cross realm authentication is allowed, user@REALM_A.COM and user@REALM_B.COM could both authorize as user@servername, so there is some risk here. But if usernames across the realms are unique, this can be very helpful.";
}
}
\ No newline at end of file
/**
* $RCSfile$
* $Revision: $
* $Date: 2006-04-20 10:46:24 -0500 (Thu, 20 Apr 2006) $
*
* Copyright (C) 2004 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.
*/
package org.jivesoftware.openfire.sasl;
import org.jivesoftware.util.JiveGlobals;
/**
* This policy will authorize any principal who:
* <p/>
* <li> Username of principal matches exactly the username of the JID </li>
* <li> The user principal's realm matches exactly the realm of the server.</li>
* Note that the realm may not match the servername, and in fact for this
* policy to be useful it will not match the servername. RFC3920 Section
* 6.1, item 7 states that if the principal (authorization entity) is the
* same as the JID (initiating entity), its MUST NOT provide an authorization
* identity. In practice however, GSSAPI will provide both. (Note: Ive
* not done extensive testing on this)
*
* @author Jay Kline
*/
public class StrictAuthorizationPolicy implements AuthorizationPolicy {
/**
* Returns true if the principal is explicity authorized to the JID
*
* @param username The username requested.
* @param principal The principal requesting the username.
* @return true is the user is authorized to be principal
*/
public boolean authorize(String username, String principal) {
return (principal.equals(username + "@" + JiveGlobals.getXMLProperty("sasl.realm")));
}
/**
* Returns the short name of the Policy
*
* @return The short name of the Policy
*/
public String name() {
return "Strict Policy";
}
/**
* Returns a description of the Policy
*
* @return The description of the Policy.
*/
public String description() {
return "This policy will authorize any principal whos username matches exactly the username of the JID and whos realm matches exactly the realm of the server.";
}
}
\ No newline at end of file
...@@ -26,29 +26,29 @@ import java.util.Collection; ...@@ -26,29 +26,29 @@ import java.util.Collection;
* the user's .k5login file. A traditional Unix Kerberos methodology. The * the user's .k5login file. A traditional Unix Kerberos methodology. The
* location of this file can be configured in the <tt>openfire.xml</tt> * location of this file can be configured in the <tt>openfire.xml</tt>
* file. An entry in that file would look like the following: * file. An entry in that file would look like the following:
* * <p/>
* <pre> * <pre>
* &lt;unix&gt; * &lt;unix&gt;
* &lt;k5login&gt; /home/{0}/.k5login &lt;/k5login&gt; * &lt;k5login&gt; /home/{0}/.k5login &lt;/k5login&gt;
* &lt;/unix&gt;</pre> * &lt;/unix&gt;</pre>
* * <p/>
* The string <tt>{0}</tt> will be replaced with the username. * The string <tt>{0}</tt> will be replaced with the username.
* *
* @author Jay Kline * @author Jay Kline
*/ */
public class UnixK5LoginProvider extends AbstractAuthorizationProvider implements AuthorizationPolicy { public class UnixK5LoginProvider extends AbstractAuthorizationProvider {
/** /**
* Returns true if the principal is explicity authorized to the JID * Returns true if the principal is explicity authorized to the JID
* *
* @param username The username requested. * @param username The username requested.
* @param principal The principal requesting the username. * @param principal The principal requesting the username.
* @return true is the user is authorized to be principal * @return true is the user is authorized to be principal
*/ */
public boolean authorize(String username, String principal) { public boolean authorize(String username, String principal) {
return getAuthorized(username).contains(principal); return getAuthorized(username).contains(principal);
} }
/** /**
* Returns a String Collection of principals that are authorized to use * Returns a String Collection of principals that are authorized to use
* the named user. * the named user.
...@@ -59,63 +59,64 @@ public class UnixK5LoginProvider extends AbstractAuthorizationProvider implement ...@@ -59,63 +59,64 @@ public class UnixK5LoginProvider extends AbstractAuthorizationProvider implement
public Collection<String> getAuthorized(String username) { public Collection<String> getAuthorized(String username) {
Collection<String> authorized = new ArrayList<String>(); Collection<String> authorized = new ArrayList<String>();
try { try {
String filename = JiveGlobals.getXMLProperty("unix.k5login","/home/{0}/.k5login"); String filename = JiveGlobals.getXMLProperty("unix.k5login", "/home/{0}/.k5login");
filename = filename.replace("{0}",username); filename = filename.replace("{0}", username);
File k5login = new File(filename); File k5login = new File(filename);
FileInputStream fis = new FileInputStream(k5login); FileInputStream fis = new FileInputStream(k5login);
DataInputStream dis = new DataInputStream(fis); DataInputStream dis = new DataInputStream(fis);
String line; String line;
while ( (line = dis.readLine() ) != null) { while ((line = dis.readLine()) != null) {
authorized.add(line); authorized.add(line);
} }
} catch (IOException e) { }
catch (IOException e) {
//?? //??
} }
return authorized; return authorized;
} }
/** /**
* Returns false, this implementation is not writeable. * Returns false, this implementation is not writeable.
* *
* @return False. * @return False.
*/ */
public boolean isWritable() { public boolean isWritable() {
return false; return false;
} }
/** /**
* Always throws UnsupportedOperationException. * Always throws UnsupportedOperationException.
* *
* @param username The username. * @param username The username.
* @param principal The principal authorized to use the named user. * @param principal The principal authorized to use the named user.
* @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated. * @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated.
*/ */
public void addAuthorized(String username, String principal) throws UnsupportedOperationException { public void addAuthorized(String username, String principal) throws UnsupportedOperationException {
throw new UnsupportedOperationException(); throw new UnsupportedOperationException();
} }
/** /**
* Always throws UnsupportedOperationException. * Always throws UnsupportedOperationException.
* *
* @param username The username. * @param username The username.
* @param principals The Collection of principals authorized to use the named user. * @param principals The Collection of principals authorized to use the named user.
*/ */
public void addAuthorized(String username, Collection<String> principals) throws UnsupportedOperationException { public void addAuthorized(String username, Collection<String> principals) throws UnsupportedOperationException {
throw new UnsupportedOperationException(); throw new UnsupportedOperationException();
} }
/** /**
* Always throws UnsupportedOperationException. * Always throws UnsupportedOperationException.
* *
* @param username The username. * @param username The username.
* @param principals The Collection of principals authorized to use the named user. * @param principals The Collection of principals authorized to use the named user.
* @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated. * @throws UnsupportedOperationException If this AuthorizationProvider cannot be updated.
*/ */
public void setAuthorized(String username, Collection<String> principals) throws UnsupportedOperationException { public void setAuthorized(String username, Collection<String> principals) throws UnsupportedOperationException {
throw new UnsupportedOperationException(); throw new UnsupportedOperationException();
} }
/** /**
* Returns the short name of the Policy * Returns the short name of the Policy
* *
...@@ -124,7 +125,7 @@ public class UnixK5LoginProvider extends AbstractAuthorizationProvider implement ...@@ -124,7 +125,7 @@ public class UnixK5LoginProvider extends AbstractAuthorizationProvider implement
public String name() { public String name() {
return "Unix .k5login"; return "Unix .k5login";
} }
/** /**
* Returns a description of the Policy * Returns a description of the Policy
* *
...@@ -133,4 +134,4 @@ public class UnixK5LoginProvider extends AbstractAuthorizationProvider implement ...@@ -133,4 +134,4 @@ public class UnixK5LoginProvider extends AbstractAuthorizationProvider implement
public String description() { public String description() {
return "Checks if the authenticated principal is in the user's .k5login file. A traditional Unix Kerberos methodology."; return "Checks if the authenticated principal is in the user's .k5login file. A traditional Unix Kerberos methodology.";
} }
} }
\ No newline at end of file
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment