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

org.apache.log.output.io.rotate.RevolvingFileStrategy Maven / Gradle / Ivy

The newest version!
/* 
 * 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.log.output.io.rotate;

import java.io.File;
import java.text.DecimalFormat;
import java.text.FieldPosition;
import java.text.NumberFormat;

/**
 * strategy for naming log files based on appending revolving suffix.
 * If the initial rotation is not specified then the class will attempt to
 * calculate the rotation number via the following algorithm.
 *
 * It will search for the file with the highest number in the rotation. It will
 * then increment its rotation number and use that number. If all files in rotation
 * are present then it will then set the initial rotation to the next rotation after
 * the most recently created file.
 *
 * @author Avalon Development Team
 * @author Bernhard Huber
 * @author Peter Donald
 * @author David Gray
 */
public class RevolvingFileStrategy
    implements FileStrategy
{
    ///revolving suffix formatting pattern. ie. "'.'000000"
    private static final String PATTERN = "'.'000000";

    ///a revolving suffix formatter
    private DecimalFormat m_decimalFormat;

    ///current revolving suffix
    private int m_rotation;

    ///max revolving value.
    private int m_maxRotations;

    ///the base file name.
    private File m_baseFile;

    /**
     * Creation of a new instane ofthe revolving file strategy.
     * @param baseFile the base file
     * @param maxRotations the maximum number of rotations ??
     */
    public RevolvingFileStrategy( final File baseFile, final int maxRotations )
    {
        this( baseFile, -1, maxRotations );
    }

    /**
     * Creation of a new instane ofthe revolving file strategy.
     * @param baseFile the base file
     * @param initialRotation the number of initial rotations ??
     * @param maxRotations the maximum number of rotations??
     */
    public RevolvingFileStrategy( final File baseFile,
                                  final int initialRotation,
                                  final int maxRotations )
    {
        m_decimalFormat = new DecimalFormat( PATTERN );

        m_baseFile = baseFile;
        m_rotation = initialRotation;
        m_maxRotations = maxRotations;

        if( -1 == m_maxRotations )
        {
            m_maxRotations = Integer.MAX_VALUE;
        }

        if( -1 == initialRotation )
        {
            m_rotation = calculateInitialRotation();
        }

        if( m_rotation > m_maxRotations )
        {
            m_rotation = m_maxRotations;
        }

        if( m_rotation < 0 )
        {
            m_rotation = 0;
        }
    }

    /**
     * Calculate the real file name from the base filename.
     *
     * @return File the calculated file name
     */
    public File nextFile()
    {
        final StringBuffer sb = new StringBuffer();
        final FieldPosition position =
            new FieldPosition( NumberFormat.INTEGER_FIELD );
        sb.append( m_baseFile );

        final StringBuffer result =
            m_decimalFormat.format( m_rotation, sb, position );
        m_rotation += 1;

        if( m_rotation >= m_maxRotations )
        {
            m_rotation = 0;
        }

        return new File( result.toString() );
    }

    /**
     * Retrieve the current rotation number.
     *
     * @return the current rotation number.
     */
    public int getCurrentRotation()
    {
        return m_rotation;
    }

    /**
     * Method that searches through files that
     * match the pattern for resolving file and determine
     * the last generation written to.
     *
     * @return the initial rotation
     */
    private int calculateInitialRotation()
    {
        final File[] matchingFiles = getMatchingFiles();
        if( null == matchingFiles || 0 == matchingFiles.length )
        {
            return 0;
        }

        final int[] rotations = calculateRotations( matchingFiles );

        //First we go through and look for maximum rotation
        int maxRotation = 0;
        for( int i = 0; i < rotations.length; i++ )
        {
            final int rotation = rotations[ i ];
            if( rotation > maxRotation )
            {
                maxRotation = rotation;
            }
        }

        //Now that we know maximum rotation, find minimum rotation, and index in array
        int minRotation = maxRotation; //don't assume min is 0
        int minRotationIndex = 0; //not correct now, but we will set in the loop below
        for( int i = 0; i < rotations.length; i++ )
        {
            final int rotation = rotations[ i ];
            if (rotation <= minRotation) 
            {
                minRotation = rotation;
                minRotationIndex = i;
            }
        }

        //If the max rotation present on filessytem
        //is less than max rotation possible then return that
        //rotation
        if( m_maxRotations != maxRotation )
        {
            return maxRotation + 1;
        }

        //Okay now we need to calculate the youngest file for our rotation
        long time = matchingFiles[ minRotationIndex ].lastModified();

        //index of oldest file
        int oldest = rotations[ minRotationIndex ];
        for( int i = 0; i < matchingFiles.length; i++ )
        {
            final File file = matchingFiles[ i ];
            final long lastModified = file.lastModified();
            if( lastModified < time )
            {
                time = lastModified;
                oldest = rotations[ i ];
            }
        }

        return oldest;
    }

    /**
     * Generate an array of rotation numbers for all the files specified.
     *
     * @param matchingFiles the files to generate rotation numbers for
     * @return the array containing rotations
     */
    private int[] calculateRotations( final File[] matchingFiles )
    {
        final int[] results = new int[ matchingFiles.length ];

        for( int i = 0; i < matchingFiles.length; i++ )
        {
            final File file = matchingFiles[ i ];

            // The files may be returned in any order
            //therefore calc the rotation number
            try
            {
                results[ i ] = calculateRotationForFile( file );
            }
            catch( final NumberFormatException nfe )
            {
                //If bad log file detected then set to -1
                results[ i ] = -1;
            }
        }
        return results;
    }

    /**
     * Return the rotation for the specified file
     *
     * @param file the file to check
     * @return the rotation of the file
     */
    private int calculateRotationForFile( final File file )
    {
        final String filename = file.toString();
        final int length = filename.length();
        final int minDigits =
            m_decimalFormat.getMinimumIntegerDigits();
        final String rotation = filename.substring( length - minDigits );

        return Integer.parseInt( rotation );
    }

    /**
     * Get a list of files that could have been part of the rotation.
     *
     * @return the list of files that match
     */
    private File[] getMatchingFiles()
    {
        // First get the path of the base file. Note that this path includes
        // the path and the base of the file name itself.
        final String fullFilePathName = m_baseFile.getPath();

        // Try to find the last path separator (if it exists)
        final int fileSeparatorPosition = fullFilePathName.lastIndexOf( File.separator );

        // If the last path separator does not exist the baseFile is a pure file name
        File basePath;
        String baseFileName;
        if( fileSeparatorPosition < 0 )
        {
            // assume the current directory
            basePath = new File( "." );
            baseFileName = fullFilePathName;
        }
        else
        {
            // Extract the sub-directory structure
            String m_parentPath = fullFilePathName.substring( 0, fileSeparatorPosition );
            baseFileName = fullFilePathName.substring( fileSeparatorPosition + 1 );
            basePath = new File( m_parentPath );
        }

        return basePath.listFiles( new BaseFileNameFilter( baseFileName ) );
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy