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

org.apache.wiki.auth.user.JDBCUserDatabase Maven / Gradle / Ivy

/* 
    Licensed to the Apache Software Foundation (ASF) under one
    or more contributor license agreements.  See the NOTICE file
    distributed with this work for additional information
    regarding copyright ownership.  The ASF licenses this file
    to you 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.apache.wiki.auth.user;

import org.apache.commons.lang3.StringUtils;
import org.apache.wiki.api.core.Engine;
import org.apache.wiki.api.exceptions.NoRequiredPropertyException;
import org.apache.wiki.auth.NoSuchPrincipalException;
import org.apache.wiki.auth.WikiPrincipal;
import org.apache.wiki.auth.WikiSecurityException;
import org.apache.wiki.util.Serializer;

import javax.naming.Context;
import javax.naming.InitialContext;
import javax.naming.NamingException;
import javax.sql.DataSource;
import java.io.IOException;
import java.io.Serializable;
import java.security.Principal;
import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Timestamp;
import java.util.Date;
import java.util.HashSet;
import java.util.Map;
import java.util.Properties;
import java.util.Set;

/**
 * 

* Implementation of UserDatabase that persists {@link DefaultUserProfile} * objects to a JDBC DataSource, as might typically be provided by a web * container. This implementation looks up the JDBC DataSource using JNDI. The * JNDI name of the datasource, backing table and mapped columns used by this * class can be overridden by adding settings in jspwiki.properties. *

*

* Configurable properties are these: *

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
PropertyDefaultDefinition
jspwiki.userdatabase.datasourcejdbc/UserDatabaseThe JNDI name of the DataSource
jspwiki.userdatabase.tableusersThe table that stores the user profiles
jspwiki.userdatabase.attributesattributesThe CLOB column containing the profile's custom attributes, stored as key/value strings, each separated by newline.
jspwiki.userdatabase.createdcreatedThe column containing the profile's creation timestamp
jspwiki.userdatabase.emailemailThe column containing the user's e-mail address
jspwiki.userdatabase.fullNamefull_nameThe column containing the user's full name
jspwiki.userdatabase.loginNamelogin_nameThe column containing the user's login id
jspwiki.userdatabase.passwordpasswordThe column containing the user's password
jspwiki.userdatabase.modifiedmodifiedThe column containing the profile's last-modified timestamp
jspwiki.userdatabase.uiduidThe column containing the profile's unique identifier, as a long integer
jspwiki.userdatabase.wikiNamewiki_nameThe column containing the user's wiki name
jspwiki.userdatabase.lockExpirylock_expiryThe column containing the date/time when the profile, if locked, should be unlocked.
jspwiki.userdatabase.roleTablerolesThe table that stores user roles. When a new user is created, a new * record is inserted containing user's initial role. The table will have an ID * column whose name and values correspond to the contents of the user table's * login name column. It will also contain a role column (see next row).
jspwiki.userdatabase.roleroleThe column in the role table that stores user roles. When a new user is * created, this column will be populated with the value * Authenticated. Once created, JDBCUserDatabase does not use * this column again; it is provided strictly for the convenience of * container-managed authentication services.
*

* This class hashes passwords using SHA-1. All of the underying SQL commands * used by this class are implemented using prepared statements, so it is immune * to SQL injection attacks. *

*

* This class is typically used in conjunction with a web container's JNDI * resource factory. For example, Tomcat provides a basic * JNDI factory for registering DataSources. To give JSPWiki access to the JNDI * resource named by , you would declare the datasource resource * similar to this: *

*
<Context ...>
*   ...
*   <Resource name="jdbc/UserDatabase" auth="Container"
*     type="javax.sql.DataSource" username="dbusername" password="dbpassword"
*     driverClassName="org.hsql.jdbcDriver" url="jdbc:HypersonicSQL:database"
*     maxActive="8" maxIdle="4"/>
*  ...
* </Context>
*

* To configure JSPWiki to use JDBC support, first create a database * with a structure similar to that provided by the HSQL and PostgreSQL * scripts in src/main/config/db. If you have different table or column * names you can either alias them with a database view and have JSPWiki * use the views, or alter the WEB-INF/jspwiki.properties file: the * jspwiki.userdatabase.* and jspwiki.groupdatabase.* properties change the * names of the tables and columns that JSPWiki uses. *

*

* A JNDI datasource (named jdbc/UserDatabase by default but can be configured * in the jspwiki.properties file) will need to be created in your servlet container. * JDBC driver JARs should be added, e.g. in Tomcat's lib * directory. For more Tomcat JNDI configuration examples, see * http://tomcat.apache.org/tomcat-7.0-doc/jndi-resources-howto.html. * Once done, restart JSPWiki in the servlet container for it to read the * new properties and switch to JDBC authentication. *

*

* JDBCUserDatabase commits changes as transactions if the back-end database * supports them. Changes are made immediately (during the {@link #save(UserProfile)} method). *

* * @since 2.3 */ public class JDBCUserDatabase extends AbstractUserDatabase { private static final String NOTHING = ""; public static final String DEFAULT_DB_ATTRIBUTES = "attributes"; public static final String DEFAULT_DB_CREATED = "created"; public static final String DEFAULT_DB_EMAIL = "email"; public static final String DEFAULT_DB_FULL_NAME = "full_name"; public static final String DEFAULT_DB_JNDI_NAME = "jdbc/UserDatabase"; public static final String DEFAULT_DB_LOCK_EXPIRY = "lock_expiry"; public static final String DEFAULT_DB_MODIFIED = "modified"; public static final String DEFAULT_DB_ROLE = "role"; public static final String DEFAULT_DB_ROLE_TABLE = "roles"; public static final String DEFAULT_DB_TABLE = "users"; public static final String DEFAULT_DB_LOGIN_NAME = "login_name"; public static final String DEFAULT_DB_PASSWORD = "password"; public static final String DEFAULT_DB_UID = "uid"; public static final String DEFAULT_DB_WIKI_NAME = "wiki_name"; public static final String PROP_DB_ATTRIBUTES = "jspwiki.userdatabase.attributes"; public static final String PROP_DB_CREATED = "jspwiki.userdatabase.created"; public static final String PROP_DB_EMAIL = "jspwiki.userdatabase.email"; public static final String PROP_DB_FULL_NAME = "jspwiki.userdatabase.fullName"; public static final String PROP_DB_DATASOURCE = "jspwiki.userdatabase.datasource"; public static final String PROP_DB_LOCK_EXPIRY = "jspwiki.userdatabase.lockExpiry"; public static final String PROP_DB_LOGIN_NAME = "jspwiki.userdatabase.loginName"; public static final String PROP_DB_MODIFIED = "jspwiki.userdatabase.modified"; public static final String PROP_DB_PASSWORD = "jspwiki.userdatabase.password"; public static final String PROP_DB_UID = "jspwiki.userdatabase.uid"; public static final String PROP_DB_ROLE = "jspwiki.userdatabase.role"; public static final String PROP_DB_ROLE_TABLE = "jspwiki.userdatabase.roleTable"; public static final String PROP_DB_TABLE = "jspwiki.userdatabase.table"; public static final String PROP_DB_WIKI_NAME = "jspwiki.userdatabase.wikiName"; private DataSource m_ds; private String m_deleteUserByLoginName; private String m_deleteRoleByLoginName; private String m_findByEmail; private String m_findByFullName; private String m_findByLoginName; private String m_findByUid; private String m_findByWikiName; private String m_renameProfile; private String m_renameRoles; private String m_updateProfile; private String m_findAll; private String m_findRoles; private String m_insertProfile; private String m_insertRole; private String m_attributes; private String m_email; private String m_fullName; private String m_lockExpiry; private String m_loginName; private String m_password; private String m_uid; private String m_wikiName; private String m_created; private String m_modified; private boolean m_supportsCommits; /** * Looks up and deletes the first {@link UserProfile} in the user database * that matches a profile having a given login name. If the user database * does not contain a user with a matching attribute, throws a * {@link NoSuchPrincipalException}. This method is intended to be atomic; * results cannot be partially committed. If the commit fails, it should * roll back its state appropriately. Implementing classes that persist to * the file system may wish to make this method synchronized. * * @param loginName the login name of the user profile that shall be deleted */ @Override public void deleteByLoginName( final String loginName ) throws WikiSecurityException { // Get the existing user; if not found, throws NoSuchPrincipalException findByLoginName( loginName ); try( final Connection conn = m_ds.getConnection() ; final PreparedStatement ps1 = conn.prepareStatement( m_deleteUserByLoginName ); final PreparedStatement ps2 = conn.prepareStatement( m_deleteRoleByLoginName ) ) { // Open the database connection if( m_supportsCommits ) { conn.setAutoCommit( false ); } // Delete user record ps1.setString( 1, loginName ); ps1.execute(); // Delete role record ps2.setString( 1, loginName ); ps2.execute(); // Commit and close connection if( m_supportsCommits ) { conn.commit(); } } catch( final SQLException e ) { throw new WikiSecurityException( e.getMessage(), e ); } } /** * @see org.apache.wiki.auth.user.UserDatabase#findByEmail(java.lang.String) */ @Override public UserProfile findByEmail( final String index ) throws NoSuchPrincipalException { return findByPreparedStatement( m_findByEmail, index ); } /** * @see org.apache.wiki.auth.user.UserDatabase#findByFullName(java.lang.String) */ @Override public UserProfile findByFullName( final String index ) throws NoSuchPrincipalException { return findByPreparedStatement( m_findByFullName, index ); } /** * @see org.apache.wiki.auth.user.UserDatabase#findByLoginName(java.lang.String) */ @Override public UserProfile findByLoginName( final String index ) throws NoSuchPrincipalException { return findByPreparedStatement( m_findByLoginName, index ); } /** * @see org.apache.wiki.auth.user.UserDatabase#findByWikiName(String) */ @Override public UserProfile findByUid( final String uid ) throws NoSuchPrincipalException { return findByPreparedStatement( m_findByUid, uid ); } /** * @see org.apache.wiki.auth.user.UserDatabase#findByWikiName(String) */ @Override public UserProfile findByWikiName( final String index ) throws NoSuchPrincipalException { return findByPreparedStatement( m_findByWikiName, index ); } /** * Returns all WikiNames that are stored in the UserDatabase as an array of * WikiPrincipal objects. If the database does not contain any profiles, * this method will return a zero-length array. * * @return the WikiNames */ @Override public Principal[] getWikiNames() throws WikiSecurityException { final Set principals = new HashSet<>(); try( final Connection conn = m_ds.getConnection(); final PreparedStatement ps = conn.prepareStatement( m_findAll ); final ResultSet rs = ps.executeQuery() ) { while( rs.next() ) { final String wikiName = rs.getString( m_wikiName ); if( StringUtils.isEmpty( wikiName ) ) { LOG.warn( "Detected null or empty wiki name for {} in JDBCUserDataBase. Check your user database.", rs.getString( m_loginName ) ); } else { final Principal principal = new WikiPrincipal( wikiName, WikiPrincipal.WIKI_NAME ); principals.add( principal ); } } } catch( final SQLException e ) { throw new WikiSecurityException( e.getMessage(), e ); } return principals.toArray( new Principal[0] ); } /** * @see org.apache.wiki.auth.user.UserDatabase#initialize(org.apache.wiki.api.core.Engine, java.util.Properties) */ @Override public void initialize( final Engine engine, final Properties props ) throws NoRequiredPropertyException, WikiSecurityException { final String jndiName = props.getProperty( PROP_DB_DATASOURCE, DEFAULT_DB_JNDI_NAME ); try { final Context initCtx = new InitialContext(); final Context ctx = (Context) initCtx.lookup( "java:comp/env" ); m_ds = (DataSource) ctx.lookup( jndiName ); // Prepare the SQL selectors final String userTable = props.getProperty( PROP_DB_TABLE, DEFAULT_DB_TABLE ); m_email = props.getProperty( PROP_DB_EMAIL, DEFAULT_DB_EMAIL ); m_fullName = props.getProperty( PROP_DB_FULL_NAME, DEFAULT_DB_FULL_NAME ); m_lockExpiry = props.getProperty( PROP_DB_LOCK_EXPIRY, DEFAULT_DB_LOCK_EXPIRY ); m_loginName = props.getProperty( PROP_DB_LOGIN_NAME, DEFAULT_DB_LOGIN_NAME ); m_password = props.getProperty( PROP_DB_PASSWORD, DEFAULT_DB_PASSWORD ); m_uid = props.getProperty( PROP_DB_UID, DEFAULT_DB_UID ); m_wikiName = props.getProperty( PROP_DB_WIKI_NAME, DEFAULT_DB_WIKI_NAME ); m_created = props.getProperty( PROP_DB_CREATED, DEFAULT_DB_CREATED ); m_modified = props.getProperty( PROP_DB_MODIFIED, DEFAULT_DB_MODIFIED ); m_attributes = props.getProperty( PROP_DB_ATTRIBUTES, DEFAULT_DB_ATTRIBUTES ); m_findAll = "SELECT * FROM " + userTable; m_findByEmail = "SELECT * FROM " + userTable + " WHERE " + m_email + "=?"; m_findByFullName = "SELECT * FROM " + userTable + " WHERE " + m_fullName + "=?"; m_findByLoginName = "SELECT * FROM " + userTable + " WHERE " + m_loginName + "=?"; m_findByUid = "SELECT * FROM " + userTable + " WHERE " + m_uid + "=?"; m_findByWikiName = "SELECT * FROM " + userTable + " WHERE " + m_wikiName + "=?"; // The user insert SQL prepared statement m_insertProfile = "INSERT INTO " + userTable + " (" + m_uid + "," + m_email + "," + m_fullName + "," + m_password + "," + m_wikiName + "," + m_modified + "," + m_loginName + "," + m_attributes + "," + m_created + ") VALUES (?,?,?,?,?,?,?,?,?)"; // The user update SQL prepared statement m_updateProfile = "UPDATE " + userTable + " SET " + m_uid + "=?," + m_email + "=?," + m_fullName + "=?," + m_password + "=?," + m_wikiName + "=?," + m_modified + "=?," + m_loginName + "=?," + m_attributes + "=?," + m_lockExpiry + "=? " + "WHERE " + m_loginName + "=?"; // Prepare the role insert SQL final String roleTable = props.getProperty( PROP_DB_ROLE_TABLE, DEFAULT_DB_ROLE_TABLE ); final String role = props.getProperty( PROP_DB_ROLE, DEFAULT_DB_ROLE ); m_insertRole = "INSERT INTO " + roleTable + " (" + m_loginName + "," + role + ") VALUES (?,?)"; m_findRoles = "SELECT * FROM " + roleTable + " WHERE " + m_loginName + "=?"; // Prepare the user delete SQL m_deleteUserByLoginName = "DELETE FROM " + userTable + " WHERE " + m_loginName + "=?"; // Prepare the role delete SQL m_deleteRoleByLoginName = "DELETE FROM " + roleTable + " WHERE " + m_loginName + "=?"; // Prepare the rename user/roles SQL m_renameProfile = "UPDATE " + userTable + " SET " + m_loginName + "=?," + m_modified + "=? WHERE " + m_loginName + "=?"; m_renameRoles = "UPDATE " + roleTable + " SET " + m_loginName + "=? WHERE " + m_loginName + "=?"; } catch( final NamingException e ) { LOG.error( "JDBCUserDatabase initialization error: " + e.getMessage() ); throw new NoRequiredPropertyException( PROP_DB_DATASOURCE, "JDBCUserDatabase initialization error: " + e.getMessage() ); } // Test connection by doing a quickie select try( final Connection conn = m_ds.getConnection(); final PreparedStatement ps = conn.prepareStatement( m_findAll ) ) { } catch( final SQLException e ) { LOG.error( "DB connectivity error: " + e.getMessage() ); throw new WikiSecurityException("DB connectivity error: " + e.getMessage(), e ); } LOG.info( "JDBCUserDatabase initialized from JNDI DataSource: {}", jndiName ); // Determine if the datasource supports commits try( final Connection conn = m_ds.getConnection() ) { final DatabaseMetaData dmd = conn.getMetaData(); if( dmd.supportsTransactions() ) { m_supportsCommits = true; conn.setAutoCommit( false ); LOG.info( "JDBCUserDatabase supports transactions. Good; we will use them." ); } } catch( final SQLException e ) { LOG.warn( "JDBCUserDatabase warning: user database doesn't seem to support transactions. Reason: {}", e.getMessage() ); } } /** * @see org.apache.wiki.auth.user.UserDatabase#rename(String, String) */ @Override public void rename( final String loginName, final String newName ) throws DuplicateUserException, WikiSecurityException { // Get the existing user; if not found, throws NoSuchPrincipalException final UserProfile profile = findByLoginName( loginName ); // Get user with the proposed name; if found, it's a collision try { final UserProfile otherProfile = findByLoginName( newName ); if( otherProfile != null ) { throw new DuplicateUserException( "security.error.cannot.rename", newName ); } } catch( final NoSuchPrincipalException e ) { // Good! That means it's safe to save using the new name } try( final Connection conn = m_ds.getConnection(); final PreparedStatement ps1 = conn.prepareStatement( m_renameProfile ); final PreparedStatement ps2 = conn.prepareStatement( m_renameRoles ) ) { if( m_supportsCommits ) { conn.setAutoCommit( false ); } final Timestamp ts = new Timestamp( System.currentTimeMillis() ); final Date modDate = new Date( ts.getTime() ); // Change the login ID for the user record ps1.setString( 1, newName ); ps1.setTimestamp( 2, ts ); ps1.setString( 3, loginName ); ps1.execute(); // Change the login ID for the role records ps2.setString( 1, newName ); ps2.setString( 2, loginName ); ps2.execute(); // Set the profile name and mod time profile.setLoginName( newName ); profile.setLastModified( modDate ); // Commit and close connection if( m_supportsCommits ) { conn.commit(); } } catch( final SQLException e ) { throw new WikiSecurityException( e.getMessage(), e ); } } /** * @see org.apache.wiki.auth.user.UserDatabase#save(org.apache.wiki.auth.user.UserProfile) */ @Override public void save( final UserProfile profile ) throws WikiSecurityException { final String initialRole = "Authenticated"; // Figure out which prepared statement to use & execute it final String loginName = profile.getLoginName(); UserProfile existingProfile = null; try { existingProfile = findByLoginName( loginName ); } catch( final NoSuchPrincipalException e ) { // Existing profile will be null } // Get a clean password from the passed profile. // Blank password is the same as null, which means we re-use the existing one. String password = profile.getPassword(); final String existingPassword = (existingProfile == null) ? null : existingProfile.getPassword(); if( NOTHING.equals( password ) ) { password = null; } if( password == null ) { password = existingPassword; } // If password changed, hash it before we save if( !StringUtils.equals( password, existingPassword ) ) { password = getHash( password ); } try( final Connection conn = m_ds.getConnection(); final PreparedStatement ps1 = conn.prepareStatement( m_insertProfile ); final PreparedStatement ps2 = conn.prepareStatement( m_findRoles ); final PreparedStatement ps3 = conn.prepareStatement( m_insertRole ); final PreparedStatement ps4 = conn.prepareStatement( m_updateProfile ) ) { if( m_supportsCommits ) { conn.setAutoCommit( false ); } final Timestamp ts = new Timestamp( System.currentTimeMillis() ); final Date modDate = new Date( ts.getTime() ); final java.sql.Date lockExpiry = profile.getLockExpiry() == null ? null : new java.sql.Date( profile.getLockExpiry().getTime() ); if( existingProfile == null ) { // User is new: insert new user record ps1.setString( 1, profile.getUid() ); ps1.setString( 2, profile.getEmail() ); ps1.setString( 3, profile.getFullname() ); ps1.setString( 4, password ); ps1.setString( 5, profile.getWikiName() ); ps1.setTimestamp( 6, ts ); ps1.setString( 7, profile.getLoginName() ); try { ps1.setString( 8, Serializer.serializeToBase64( profile.getAttributes() ) ); } catch ( final IOException e ) { throw new WikiSecurityException( "Could not save user profile attribute. Reason: " + e.getMessage(), e ); } ps1.setTimestamp( 9, ts ); ps1.execute(); // Insert new role record ps2.setString( 1, profile.getLoginName() ); int roles = 0; try ( final ResultSet rs = ps2.executeQuery() ) { while ( rs.next() ) { roles++; } } if( roles == 0 ) { ps3.setString( 1, profile.getLoginName() ); ps3.setString( 2, initialRole ); ps3.execute(); } // Set the profile creation time profile.setCreated( modDate ); } else { // User exists: modify existing record ps4.setString( 1, profile.getUid() ); ps4.setString( 2, profile.getEmail() ); ps4.setString( 3, profile.getFullname() ); ps4.setString( 4, password ); ps4.setString( 5, profile.getWikiName() ); ps4.setTimestamp( 6, ts ); ps4.setString( 7, profile.getLoginName() ); try { ps4.setString( 8, Serializer.serializeToBase64( profile.getAttributes() ) ); } catch ( final IOException e ) { throw new WikiSecurityException( "Could not save user profile attribute. Reason: " + e.getMessage(), e ); } ps4.setDate( 9, lockExpiry ); ps4.setString( 10, profile.getLoginName() ); ps4.execute(); } // Set the profile mod time profile.setLastModified( modDate ); // Commit and close connection if( m_supportsCommits ) { conn.commit(); } } catch( final SQLException e ) { throw new WikiSecurityException( e.getMessage(), e ); } } /** * Private method that returns the first {@link UserProfile} matching a * named column's value. This method will also set the UID if it has not yet been set. * @param sql the SQL statement that should be prepared; it must have one parameter * to set (either a String or a Long) * @param index the value to match * @return the resolved UserProfile * @throws NoSuchPrincipalException problems accessing the database */ private UserProfile findByPreparedStatement( final String sql, final Object index ) throws NoSuchPrincipalException { UserProfile profile = null; boolean found = false; boolean unique = true; try( final Connection conn = m_ds.getConnection(); final PreparedStatement ps = conn.prepareStatement( sql ) ) { if( m_supportsCommits ) { conn.setAutoCommit( false ); } // Set the parameter to search by if( index instanceof String ) { ps.setString( 1, ( String )index ); } else if ( index instanceof Long ) { ps.setLong( 1, ( Long )index ); } else { throw new IllegalArgumentException( "Index type not recognized!" ); } // Go and get the record! try( final ResultSet rs = ps.executeQuery() ) { while ( rs.next() ) { if( profile != null ) { unique = false; break; } profile = newProfile(); // Fetch the basic user attributes profile.setUid( rs.getString( m_uid ) ); if ( profile.getUid() == null ) { profile.setUid( generateUid( this ) ); } profile.setCreated( rs.getTimestamp( m_created ) ); profile.setEmail( rs.getString( m_email ) ); profile.setFullname( rs.getString( m_fullName ) ); profile.setLastModified( rs.getTimestamp( m_modified ) ); final Date lockExpiry = rs.getDate( m_lockExpiry ); profile.setLockExpiry( rs.wasNull() ? null : lockExpiry ); profile.setLoginName( rs.getString( m_loginName ) ); profile.setPassword( rs.getString( m_password ) ); // Fetch the user attributes final String rawAttributes = rs.getString( m_attributes ); if ( rawAttributes != null ) { try { final Map attributes = Serializer.deserializeFromBase64( rawAttributes ); profile.getAttributes().putAll( attributes ); } catch ( final IOException e ) { LOG.error( "Could not parse user profile attributes!", e ); } } found = true; } } } catch( final SQLException e ) { throw new NoSuchPrincipalException( e.getMessage() ); } if( !found ) { throw new NoSuchPrincipalException( "Could not find profile in database!" ); } if( !unique ) { throw new NoSuchPrincipalException( "More than one profile in database!" ); } return profile; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy