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,10 +15,6 @@ import org.dom4j.DocumentHelper; ...@@ -15,10 +15,6 @@ import org.dom4j.DocumentHelper;
import org.dom4j.Element; import org.dom4j.Element;
import org.dom4j.Namespace; import org.dom4j.Namespace;
import org.dom4j.QName; import org.dom4j.QName;
import org.jivesoftware.util.CertificateManager;
import org.jivesoftware.util.JiveGlobals;
import org.jivesoftware.util.Log;
import org.jivesoftware.util.StringUtils;
import org.jivesoftware.openfire.XMPPServer; import org.jivesoftware.openfire.XMPPServer;
import org.jivesoftware.openfire.auth.AuthFactory; import org.jivesoftware.openfire.auth.AuthFactory;
import org.jivesoftware.openfire.auth.AuthToken; import org.jivesoftware.openfire.auth.AuthToken;
...@@ -26,21 +22,31 @@ import org.jivesoftware.openfire.auth.UnauthorizedException; ...@@ -26,21 +22,31 @@ import org.jivesoftware.openfire.auth.UnauthorizedException;
import org.jivesoftware.openfire.session.ClientSession; import org.jivesoftware.openfire.session.ClientSession;
import org.jivesoftware.openfire.session.IncomingServerSession; import org.jivesoftware.openfire.session.IncomingServerSession;
import org.jivesoftware.openfire.session.Session; import org.jivesoftware.openfire.session.Session;
import org.jivesoftware.util.CertificateManager;
import org.jivesoftware.util.JiveGlobals;
import org.jivesoftware.util.Log;
import org.jivesoftware.util.StringUtils;
import org.xmpp.packet.JID; import org.xmpp.packet.JID;
import javax.net.ssl.SSLPeerUnverifiedException; import javax.net.ssl.SSLPeerUnverifiedException;
import javax.security.sasl.Sasl; import javax.security.sasl.Sasl;
import javax.security.sasl.SaslException; import javax.security.sasl.SaslException;
import javax.security.sasl.SaslServer; import javax.security.sasl.SaslServer;
import java.io.UnsupportedEncodingException; import java.io.UnsupportedEncodingException;
import java.security.cert.Certificate; import java.security.cert.Certificate;
import java.security.cert.X509Certificate; import java.security.cert.X509Certificate;
import java.util.*; import java.util.HashSet;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import java.util.StringTokenizer;
import java.util.TreeMap;
/** /**
* SASLAuthentication is responsible for returning the available SASL mechanisms to use and for * SASLAuthentication is responsible for returning the available SASL mechanisms to use and for
* actually performing the SASL authentication.<p> * actually performing the SASL authentication.<p>
* * <p/>
* The list of available SASL mechanisms is determined by: * The list of available SASL mechanisms is determined by:
* <ol> * <ol>
* <li>The type of {@link org.jivesoftware.openfire.user.UserProvider} being used since * <li>The type of {@link org.jivesoftware.openfire.user.UserProvider} being used since
...@@ -94,7 +100,8 @@ public class SASLAuthentication { ...@@ -94,7 +100,8 @@ public class SASLAuthentication {
} }
} }
@SuppressWarnings({"UnnecessarySemicolon"}) // Support for QDox Parser @SuppressWarnings({"UnnecessarySemicolon"})
// Support for QDox Parser
public enum Status { public enum Status {
/** /**
* Entity needs to respond last challenge. Session is still negotiating * Entity needs to respond last challenge. Session is still negotiating
...@@ -244,7 +251,7 @@ public class SASLAuthentication { ...@@ -244,7 +251,7 @@ public class SASLAuthentication {
break; break;
case RESPONSE: case RESPONSE:
// Store the requested SASL mechanism by the client // Store the requested SASL mechanism by the client
mechanism = (String) session.getSessionData("SaslMechanism"); mechanism = (String)session.getSessionData("SaslMechanism");
if (mechanism.equalsIgnoreCase("PLAIN") && if (mechanism.equalsIgnoreCase("PLAIN") &&
mechanisms.contains("PLAIN")) { mechanisms.contains("PLAIN")) {
status = doPlainAuthentication(session, doc); status = doPlainAuthentication(session, doc);
...@@ -256,7 +263,7 @@ public class SASLAuthentication { ...@@ -256,7 +263,7 @@ public class SASLAuthentication {
status = doSharedSecretAuthentication(session, doc); status = doSharedSecretAuthentication(session, doc);
} }
else if (mechanisms.contains(mechanism)) { else if (mechanisms.contains(mechanism)) {
SaslServer ss = (SaslServer) session.getSessionData("SaslServer"); SaslServer ss = (SaslServer)session.getSessionData("SaslServer");
if (ss != null) { if (ss != null) {
boolean ssComplete = ss.isComplete(); boolean ssComplete = ss.isComplete();
String response = doc.getTextTrim(); String response = doc.getTextTrim();
...@@ -469,10 +476,10 @@ public class SASLAuthentication { ...@@ -469,10 +476,10 @@ public class SASLAuthentication {
return Status.authenticated; return Status.authenticated;
} }
// Check that hostname matches the one provided in a certificate // Check that hostname matches the one provided in a certificate
SocketConnection connection = (SocketConnection) session.getConnection(); SocketConnection connection = (SocketConnection)session.getConnection();
try { try {
for (Certificate certificate : connection.getSSLSession().getPeerCertificates()) { for (Certificate certificate : connection.getSSLSession().getPeerCertificates()) {
if (CertificateManager.getPeerIdentities((X509Certificate) certificate) if (CertificateManager.getPeerIdentities((X509Certificate)certificate)
.contains(hostname)) { .contains(hostname)) {
authenticationSuccessful(session, hostname, null); authenticationSuccessful(session, hostname, null);
return Status.authenticated; return Status.authenticated;
...@@ -487,8 +494,7 @@ public class SASLAuthentication { ...@@ -487,8 +494,7 @@ public class SASLAuthentication {
} }
private static Status doSharedSecretAuthentication(Session session, Element doc) private static Status doSharedSecretAuthentication(Session session, Element doc)
throws UnsupportedEncodingException throws UnsupportedEncodingException {
{
String secretDigest; String secretDigest;
String response = doc.getTextTrim(); String response = doc.getTextTrim();
if (response == null || response.length() == 0) { if (response == null || response.length() == 0) {
...@@ -541,7 +547,7 @@ public class SASLAuthentication { ...@@ -541,7 +547,7 @@ public class SASLAuthentication {
session.getConnection().deliverRawText(reply.toString()); session.getConnection().deliverRawText(reply.toString());
// We only support SASL for c2s // We only support SASL for c2s
if (session instanceof ClientSession) { if (session instanceof ClientSession) {
((ClientSession) session).setAuthToken(new AuthToken(username)); ((ClientSession)session).setAuthToken(new AuthToken(username));
} }
else if (session instanceof IncomingServerSession) { else if (session instanceof IncomingServerSession) {
String hostname = username; String hostname = username;
...@@ -549,7 +555,7 @@ public class SASLAuthentication { ...@@ -549,7 +555,7 @@ public class SASLAuthentication {
session.setAddress(new JID(null, hostname, null)); session.setAddress(new JID(null, hostname, null));
// Add the validated domain as a valid domain. The remote server can // Add the validated domain as a valid domain. The remote server can
// now send packets from this address // now send packets from this address
((IncomingServerSession) session).addValidatedDomain(hostname); ((IncomingServerSession)session).addValidatedDomain(hostname);
} }
} }
...@@ -559,7 +565,7 @@ public class SASLAuthentication { ...@@ -559,7 +565,7 @@ public class SASLAuthentication {
reply.append("<not-authorized/></failure>"); reply.append("<not-authorized/></failure>");
session.getConnection().deliverRawText(reply.toString()); session.getConnection().deliverRawText(reply.toString());
// Give a number of retries before closing the connection // Give a number of retries before closing the connection
Integer retries = (Integer) session.getSessionData("authRetries"); Integer retries = (Integer)session.getSessionData("authRetries");
if (retries == null) { if (retries == null) {
retries = 1; retries = 1;
} }
...@@ -567,7 +573,7 @@ public class SASLAuthentication { ...@@ -567,7 +573,7 @@ public class SASLAuthentication {
retries = retries + 1; retries = retries + 1;
} }
session.setSessionData("authRetries", retries); session.setSessionData("authRetries", retries);
if (retries >= JiveGlobals.getIntProperty("xmpp.auth.retries", 3) ) { if (retries >= JiveGlobals.getIntProperty("xmpp.auth.retries", 3)) {
// Close the connection // Close the connection
session.getConnection().close(); session.getConnection().close();
} }
...@@ -576,7 +582,7 @@ public class SASLAuthentication { ...@@ -576,7 +582,7 @@ public class SASLAuthentication {
/** /**
* Adds a new SASL mechanism to the list of supported SASL mechanisms by the server. The * Adds a new SASL mechanism to the list of supported SASL mechanisms by the server. The
* new mechanism will be offered to clients and connection managers as stream features.<p> * new mechanism will be offered to clients and connection managers as stream features.<p>
* * <p/>
* Note: this method simply registers the SASL mechanism to be advertised as a supported * Note: this method simply registers the SASL mechanism to be advertised as a supported
* mechanism by Openfire. Actual SASL handling is done by Java itself, so you must add * mechanism by Openfire. Actual SASL handling is done by Java itself, so you must add
* the provider to Java. * the provider to Java.
...@@ -607,7 +613,7 @@ public class SASLAuthentication { ...@@ -607,7 +613,7 @@ public class SASLAuthentication {
public static Set<String> getSupportedMechanisms() { public static Set<String> getSupportedMechanisms() {
Set<String> answer = new HashSet<String>(mechanisms); Set<String> answer = new HashSet<String>(mechanisms);
// Clean up not-available mechanisms // Clean up not-available mechanisms
for (Iterator<String> it=answer.iterator(); it.hasNext();) { for (Iterator<String> it = answer.iterator(); it.hasNext();) {
String mech = it.next(); String mech = it.next();
if (mech.equals("CRAM-MD5") || mech.equals("DIGEST-MD5")) { if (mech.equals("CRAM-MD5") || mech.equals("DIGEST-MD5")) {
// Check if the user provider in use supports passwords retrieval. Accessing // Check if the user provider in use supports passwords retrieval. Accessing
...@@ -652,8 +658,7 @@ public class SASLAuthentication { ...@@ -652,8 +658,7 @@ public class SASLAuthentication {
mech.equals("DIGEST-MD5") || mech.equals("DIGEST-MD5") ||
mech.equals("CRAM-MD5") || mech.equals("CRAM-MD5") ||
mech.equals("GSSAPI") || mech.equals("GSSAPI") ||
mech.equals("JIVE-SHAREDSECRET")) mech.equals("JIVE-SHAREDSECRET")) {
{
Log.debug("SASLAuthentication: Added " + mech + " to mech list"); Log.debug("SASLAuthentication: Added " + mech + " to mech list");
mechanisms.add(mech); mechanisms.add(mech);
} }
...@@ -661,12 +666,9 @@ public class SASLAuthentication { ...@@ -661,12 +666,9 @@ public class SASLAuthentication {
if (mechanisms.contains("GSSAPI")) { if (mechanisms.contains("GSSAPI")) {
if (JiveGlobals.getXMLProperty("sasl.gssapi.config") != null) { if (JiveGlobals.getXMLProperty("sasl.gssapi.config") != null) {
System.setProperty("java.security.krb5.debug", System.setProperty("java.security.krb5.debug", JiveGlobals.getXMLProperty("sasl.gssapi.debug", "false"));
JiveGlobals.getXMLProperty("sasl.gssapi.debug", "false")); System.setProperty("java.security.auth.login.config", JiveGlobals.getXMLProperty("sasl.gssapi.config"));
System.setProperty("java.security.auth.login.config", System.setProperty("javax.security.auth.useSubjectCredsOnly", JiveGlobals.getXMLProperty("sasl.gssapi.useSubjectCredsOnly", "false"));
JiveGlobals.getXMLProperty("sasl.gssapi.config"));
System.setProperty("javax.security.auth.useSubjectCredsOnly",
JiveGlobals.getXMLProperty("sasl.gssapi.useSubjectCredsOnly", "false"));
} }
else { else {
//Not configured, remove the option. //Not configured, remove the option.
......
...@@ -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) {
StringTokenizer st = new StringTokenizer(classList, " ,\t\n\r\f");
while (st.hasMoreTokens()) {
String s_provider = st.nextToken();
try { try {
Class c_provider = ClassUtils.forName(className); Class c_provider = ClassUtils.forName(s_provider);
AuthorizationPolicy provider = (AuthorizationPolicy)(c_provider.newInstance()); AuthorizationPolicy provider =
Log.debug("AuthorizationManager: Loaded " + provider); (AuthorizationPolicy)(c_provider.newInstance());
authorizationProvider = provider; Log.debug("AuthorizationManager: Loaded " + s_provider);
authorizationPolicies.add(provider);
} }
catch (Exception e) { catch (Exception e) {
Log.error("Error loading AuthorizationProvider: " + className + "\n" + e); Log.error("Error loading AuthorizationProvider: " + s_provider + "\n" + e);
} }
} }
}
if (authorizationProvider == null) { if (authorizationPolicies.isEmpty()) {
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 false;
return true;
} }
} }
/**
* $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,17 +26,17 @@ import java.util.Collection; ...@@ -26,17 +26,17 @@ 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
...@@ -59,17 +59,18 @@ public class UnixK5LoginProvider extends AbstractAuthorizationProvider implement ...@@ -59,17 +59,18 @@ 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;
......
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