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

org.apache.wiki.auth.authorize.XMLGroupDatabase 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.authorize;

import org.apache.commons.lang3.StringUtils;
import org.apache.commons.text.StringEscapeUtils;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
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.TextUtil;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NodeList;
import org.xml.sax.SAXException;

import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.security.Principal;
import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Collection;
import java.util.Date;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.ConcurrentHashMap;


/**
 * 

* GroupDatabase implementation for loading, persisting and storing wiki groups, * using an XML file for persistence. Group entries are simple * <group> elements under the root. Each group member is * representated by a <member> element. For example: *

*
* <groups>
*   <group name="TV" created="Jun 20, 2006 2:50:54 PM" lastModified="Jan 21, 2006 2:50:54 PM">
*     <member principal="Archie Bunker" />
*     <member principal="BullwinkleMoose" />
*     <member principal="Fred Friendly" />
*   </group>
*   <group name="Literature" created="Jun 22, 2006 2:50:54 PM" lastModified="Jan 23, 2006 2:50:54 PM">
*     <member principal="Charles Dickens" />
*     <member principal="Homer" />
*   </group>
* </groups> *
* @since 2.4.17 */ public class XMLGroupDatabase implements GroupDatabase { private static final Logger LOG = LogManager.getLogger( XMLGroupDatabase.class ); /** The jspwiki.properties property specifying the file system location of the group database. */ public static final String PROP_DATABASE = "jspwiki.xmlGroupDatabaseFile"; private static final String DEFAULT_DATABASE = "groupdatabase.xml"; private static final String CREATED = "created"; private static final String CREATOR = "creator"; private static final String GROUP_TAG = "group"; private static final String GROUP_NAME = "name"; private static final String LAST_MODIFIED = "lastModified"; private static final String MODIFIER = "modifier"; private static final String MEMBER_TAG = "member"; private static final String PRINCIPAL = "principal"; private static final String DATE_FORMAT = "yyyy.MM.dd 'at' HH:mm:ss:SSS z"; private Document m_dom; private final DateFormat m_defaultFormat = DateFormat.getDateTimeInstance(); private File m_file; private Engine m_engine; private final Map m_groups = new ConcurrentHashMap<>(); /** * Looks up and deletes a {@link Group} from the group database. If the * group database does not contain the supplied Group. this method throws a * {@link NoSuchPrincipalException}. The method commits the results * of the delete to persistent storage. * @param group the group to remove * @throws WikiSecurityException if the database does not contain the * supplied group (thrown as {@link NoSuchPrincipalException}) or if * the commit did not succeed */ @Override public void delete( final Group group ) throws WikiSecurityException { final String index = group.getName(); final boolean exists = m_groups.containsKey( index ); if ( !exists ) { throw new NoSuchPrincipalException( "Not in database: " + group.getName() ); } m_groups.remove( index ); // Commit to disk saveDOM(); } /** * Returns all wiki groups that are stored in the GroupDatabase as an array * of Group objects. If the database does not contain any groups, this * method will return a zero-length array. This method causes back-end * storage to load the entire set of group; thus, it should be called * infrequently (e.g., at initialization time). * @return the wiki groups * @throws WikiSecurityException if the groups cannot be returned by the back-end */ @Override public Group[] groups() throws WikiSecurityException { buildDOM(); final Collection groups = m_groups.values(); return groups.toArray( new Group[0] ); } /** * Initializes the group database based on values from a Properties object. * The properties object must contain a file path to the XML database file * whose key is {@link #PROP_DATABASE}. * @param engine the wiki engine * @param props the properties used to initialize the group database * @throws NoRequiredPropertyException if the user database cannot be located, parsed, or opened * @throws WikiSecurityException if the database could not be initialized successfully */ @Override public void initialize( final Engine engine, final Properties props ) throws NoRequiredPropertyException, WikiSecurityException { m_engine = engine; final File defaultFile; if ( engine.getRootPath() == null ) { LOG.warn( "Cannot identify JSPWiki root path" ); defaultFile = new File( "WEB-INF/" + DEFAULT_DATABASE ).getAbsoluteFile(); } else { defaultFile = new File( engine.getRootPath() + "/WEB-INF/" + DEFAULT_DATABASE ); } // Get database file location final String file = TextUtil.getStringProperty(props, PROP_DATABASE , defaultFile.getAbsolutePath()); if ( file == null ) { LOG.warn( "XML group database property " + PROP_DATABASE + " not found; trying " + defaultFile ); m_file = defaultFile; } else { m_file = new File( file ); } LOG.info( "XML group database at " + m_file.getAbsolutePath() ); // Read DOM buildDOM(); } /** * Saves a Group to the group database. Note that this method must * fail, and throw an IllegalArgumentException, if the * proposed group is the same name as one of the built-in Roles: e.g., * Admin, Authenticated, etc. The database is responsible for setting * create/modify timestamps, upon a successful save, to the Group. * The method commits the results of the delete to persistent storage. * @param group the Group to save * @param modifier the user who saved the Group * @throws WikiSecurityException if the Group could not be saved successfully */ @Override public void save( final Group group, final Principal modifier ) throws WikiSecurityException { if ( group == null || modifier == null ) { throw new IllegalArgumentException( "Group or modifier cannot be null." ); } checkForRefresh(); final String index = group.getName(); final boolean isNew = !( m_groups.containsKey( index ) ); final Date modDate = new Date( System.currentTimeMillis() ); if( isNew ) { // If new, set created info group.setCreated( modDate ); group.setCreator( modifier.getName() ); } group.setModifier( modifier.getName() ); group.setLastModified( modDate ); // Add the group to the 'saved' list m_groups.put( index, group ); // Commit to disk saveDOM(); } private void buildDOM() { final DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); factory.setValidating( false ); factory.setExpandEntityReferences( false ); factory.setIgnoringComments( true ); factory.setNamespaceAware( false ); //factory.setAttribute( XMLConstants.ACCESS_EXTERNAL_DTD, "" ); //factory.setAttribute( XMLConstants.ACCESS_EXTERNAL_SCHEMA, "" ); try { m_dom = factory.newDocumentBuilder().parse( m_file ); LOG.debug( "Database successfully initialized" ); m_lastModified = m_file.lastModified(); m_lastCheck = System.currentTimeMillis(); } catch( final ParserConfigurationException e ) { LOG.error( "Configuration error: {}", e.getMessage() ); } catch( final SAXException e ) { LOG.error( "SAX error: {}", e.getMessage() ); } catch( final FileNotFoundException e ) { LOG.info( "Group database not found; creating from scratch..." ); } catch( final IOException e ) { LOG.error( "IO error: {}", e.getMessage() ); } if ( m_dom == null ) { try { // // Create the DOM from scratch // m_dom = factory.newDocumentBuilder().newDocument(); m_dom.appendChild( m_dom.createElement( "groups" ) ); } catch( final ParserConfigurationException e ) { LOG.fatal( "Could not create in-memory DOM" ); } } // Ok, now go and read this sucker in if( m_dom != null ) { final NodeList groupNodes = m_dom.getElementsByTagName( GROUP_TAG ); for( int i = 0; i < groupNodes.getLength(); i++ ) { final Element groupNode = (Element) groupNodes.item( i ); final String groupName = groupNode.getAttribute( GROUP_NAME ); if( StringUtils.isEmpty( groupName ) ) { LOG.warn( "Detected null or empty group name in XMLGroupDataBase. Check your group database." ); } else { final Group group = buildGroup( groupNode, groupName ); m_groups.put( groupName, group ); } } } } private long m_lastCheck; private long m_lastModified; private void checkForRefresh() { final long time = System.currentTimeMillis(); if( time - m_lastCheck > 60*1000L ) { final long lastModified = m_file.lastModified(); if( lastModified > m_lastModified ) { buildDOM(); } } } /** * Constructs a Group based on a DOM group node. * @param groupNode the node in the DOM containing the node * @param name the name of the group */ private Group buildGroup( final Element groupNode, final String name ) { // It's an error if either param is null (very odd) if ( groupNode == null || name == null ) { throw new IllegalArgumentException( "DOM element or name cannot be null." ); } // Construct a new group final Group group = new Group( name, m_engine.getApplicationName() ); // Get the users for this group, and add them final NodeList members = groupNode.getElementsByTagName( MEMBER_TAG ); for( int i = 0; i < members.getLength(); i++ ) { final Element memberNode = (Element) members.item( i ); final String principalName = memberNode.getAttribute( PRINCIPAL ); final Principal member = new WikiPrincipal( principalName ); group.add( member ); } // Add the created/last-modified info final String creator = groupNode.getAttribute( CREATOR ); final String created = groupNode.getAttribute( CREATED ); final String modifier = groupNode.getAttribute( MODIFIER ); final String modified = groupNode.getAttribute( LAST_MODIFIED ); try { group.setCreated( new SimpleDateFormat( DATE_FORMAT ).parse( created ) ); group.setLastModified( new SimpleDateFormat( DATE_FORMAT ).parse( modified ) ); } catch ( final ParseException e ) { // If parsing failed, use the platform default try { group.setCreated( m_defaultFormat.parse( created ) ); group.setLastModified( m_defaultFormat.parse( modified ) ); } catch ( final ParseException e2 ) { LOG.warn( "Could not parse 'created' or 'lastModified' " + "attribute for " + " group'" + group.getName() + "'." + " It may have been tampered with." ); } } group.setCreator( creator ); group.setModifier( modifier ); return group; } private void saveDOM() throws WikiSecurityException { if ( m_dom == null ) { LOG.fatal( "Group database doesn't exist in memory." ); } final File newFile = new File( m_file.getAbsolutePath() + ".new" ); try( final BufferedWriter io = new BufferedWriter( new OutputStreamWriter( Files.newOutputStream( newFile.toPath() ), StandardCharsets.UTF_8 ) ) ) { // Write the file header and document root io.write( "\n" ); io.write( "\n" ); // Write each profile as a node for( final Group group : m_groups.values() ) { io.write( " <" + GROUP_TAG + " " ); io.write( GROUP_NAME ); io.write( "=\"" + StringEscapeUtils.escapeXml11( group.getName() )+ "\" " ); io.write( CREATOR ); io.write( "=\"" + StringEscapeUtils.escapeXml11( group.getCreator() ) + "\" " ); io.write( CREATED ); io.write( "=\"" + new SimpleDateFormat( DATE_FORMAT ).format( group.getCreated() ) + "\" " ); io.write( MODIFIER ); io.write( "=\"" + group.getModifier() + "\" " ); io.write( LAST_MODIFIED ); io.write( "=\"" + new SimpleDateFormat( DATE_FORMAT ).format( group.getLastModified() ) + "\"" ); io.write( ">\n" ); // Write each member as a node for( final Principal member : group.members() ) { io.write( " <" + MEMBER_TAG + " " ); io.write( PRINCIPAL ); io.write( "=\"" + StringEscapeUtils.escapeXml11(member.getName()) + "\" " ); io.write( "/>\n" ); } // Close tag io.write( " \n" ); } io.write( "" ); } catch( final IOException e ) { throw new WikiSecurityException( e.getLocalizedMessage(), e ); } // Copy new file over old version final File backup = new File( m_file.getAbsolutePath() + ".old" ); if ( backup.exists() && !backup.delete()) { LOG.error( "Could not delete old group database backup: " + backup ); } if ( !m_file.renameTo( backup ) ) { LOG.error( "Could not create group database backup: " + backup ); } if ( !newFile.renameTo( m_file ) ) { LOG.error( "Could not save database: " + backup + " restoring backup." ); if ( !backup.renameTo( m_file ) ) { LOG.error( "Restore failed. Check the file permissions." ); } LOG.error( "Could not save database: " + m_file + ". Check the file permissions" ); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy