# $Id$
require 'lockfile'
module Logging::Appenders
# An appender that writes to a file and ensures that the file size or age
# never exceeds some user specified level.
#
# The goal of this class is to write log messages to a file. When the file
# age or size exceeds a given limit then the log file is closed, the name
# is changed to indicate it is an older log file, and a new log file is
# created.
#
# The name of the log file is changed by inserting the age of the log file
# (as a single number) between the log file name and the extension. If the
# file has no extension then the number is appended to the filename. Here
# is a simple example:
#
# /var/log/ruby.log => /var/log/ruby.1.log
#
# New log messages will be appended to a newly opened log file of the same
# name (/var/log/ruby.log in our example above). The age number
# for all older log files is incremented when the log file is rolled. The
# number of older log files to keep can be given, otherwise all the log
# files are kept.
#
# The actual process of rolling all the log file names can be expensive if
# there are many, many older log files to process.
#
class RollingFile < ::Logging::Appenders::IO
# call-seq:
# RollingFile.new( name, opts )
#
# Creates a new Rolling File Appender. The _name_ is the unique Appender
# name used to retrieve this appender from the Appender hash. The only
# required option is the filename to use for creating log files.
#
# [:filename] The base filename to use when constructing new log
# filenames.
#
# The following options are optional:
#
# [:layout] The Layout that will be used by this appender. The Basic
# layout will be used if none is given.
# [:truncate] When set to true any existing log files will be rolled
# immediately and a new, empty log file will be created.
# [:size] The maximum allowed size (in bytes) of a log file before
# it is rolled.
# [:age] The maximum age (in seconds) of a log file before it is
# rolled. The age can also be given as 'daily', 'weekly',
# or 'monthly'.
# [:keep] The number of rolled log files to keep.
# [:safe] When set to true, extra checks are made to ensure that
# only once process can roll the log files; this option
# should only be used when multiple processes will be
# logging to the same log file (does not work on Windows)
#
def initialize( name, opts = {} )
# raise an error if a filename was not given
@fn = opts.getopt(:filename, name)
raise ArgumentError, 'no filename was given' if @fn.nil?
::Logging::Appenders::File.assert_valid_logfile(@fn)
# grab the information we need to properly roll files
ext = ::File.extname(@fn)
bn = ::File.join(::File.dirname(@fn), ::File.basename(@fn, ext))
@rgxp = %r/\.(\d+)#{Regexp.escape(ext)}\z/
@glob = "#{bn}.*#{ext}"
@logname_fmt = "#{bn}.%d#{ext}"
# grab our options
@keep = opts.getopt(:keep, :as => Integer)
@size = opts.getopt(:size, :as => Integer)
@lockfile = if opts.getopt(:safe, false) and !::Logging::WIN32
::Lockfile.new(
@fn + '.lck',
:retries => 1,
:timeout => 2
)
end
code = 'def sufficiently_aged?() false end'
@age_fn = @fn + '.age'
case @age = opts.getopt(:age)
when 'daily'
FileUtils.touch(@age_fn) unless test(?f, @age_fn)
code = <<-CODE
def sufficiently_aged?
now = Time.now
start = ::File.mtime(@age_fn)
if (now.day != start.day) or (now - start) > 86400
return true
end
false
end
CODE
when 'weekly'
FileUtils.touch(@age_fn) unless test(?f, @age_fn)
code = <<-CODE
def sufficiently_aged?
if (Time.now - ::File.mtime(@age_fn)) > 604800
return true
end
false
end
CODE
when 'monthly'
FileUtils.touch(@age_fn) unless test(?f, @age_fn)
code = <<-CODE
def sufficiently_aged?
now = Time.now
start = ::File.mtime(@age_fn)
if (now.month != start.month) or (now - start) > 2678400
return true
end
false
end
CODE
when Integer, String
@age = Integer(@age)
FileUtils.touch(@age_fn) unless test(?f, @age_fn)
code = <<-CODE
def sufficiently_aged?
if (Time.now - ::File.mtime(@age_fn)) > @age
return true
end
false
end
CODE
end
meta = class << self; self end
meta.class_eval code
# if the truncate flag was set to true, then roll
roll_now = opts.getopt(:truncate, false)
roll_files if roll_now
super(name, open_logfile, opts)
end
private
# call-seq:
# write( event )
#
# Write the given _event_ to the log file. The log file will be rolled
# if the maximum file size is exceeded or if the file is older than the
# maximum age.
#
def write( event )
str = event.instance_of?(::Logging::LogEvent) ?
@layout.format(event) : event.to_s
return if str.empty?
check_logfile
super(str)
if roll_required?(str)
return roll unless @lockfile
begin
@lockfile.lock
check_logfile
roll if roll_required?
ensure
@lockfile.unlock
end
end
end
# call-seq:
# roll
#
# Close the currently open log file, roll all the log files, and open a
# new log file.
#
def roll
@io.close rescue nil
roll_files
open_logfile
end
# call-seq:
# roll_required?( str ) => true or false
#
# Returns +true+ if the log file needs to be rolled.
#
def roll_required?( str = nil )
# check if max size has been exceeded
s = if @size
@file_size = @stat.size if @stat.size > @file_size
@file_size += str.size if str
@file_size > @size
end
# check if max age has been exceeded
a = sufficiently_aged?
return (s || a)
end
# call-seq:
# roll_files
#
# Roll the log files. This is accomplished by renaming the log files
# starting with the oldest and working towards the youngest.
#
# test.10.log => deleted (we are only keeping 10)
# test.9.log => test.10.log
# test.8.log => test.9.log
# ...
# test.1.log => test.2.log
#
# Lastly the current log file is rolled to a numbered log file.
#
# test.log => test.1.log
#
# This method leaves no test.log file when it is done. This
# file will be created elsewhere.
#
def roll_files
return unless ::File.exist?(@fn)
files = Dir.glob(@glob).find_all {|fn| @rgxp =~ fn}
unless files.empty?
# sort the files in revese order based on their count number
files = files.sort do |a,b|
a = Integer(@rgxp.match(a)[1])
b = Integer(@rgxp.match(b)[1])
b <=> a
end
# for each file, roll its count number one higher
files.each do |fn|
cnt = Integer(@rgxp.match(fn)[1])
if @keep and cnt >= @keep
::File.delete fn
next
end
::File.rename fn, sprintf(@logname_fmt, cnt+1)
end
end
# finally reanme the base log file
::File.rename(@fn, sprintf(@logname_fmt, 1))
# touch the age file if needed
FileUtils.touch(@age_fn) if @age
end
# call-seq:
# open_logfile => io
#
# Opens the logfile and stores the current file szie and inode.
#
def open_logfile
@io = ::File.new(@fn, 'a')
@io.sync = true
@stat = ::File.stat(@fn)
@file_size = @stat.size
@inode = @stat.ino
return @io
end
#
#
def check_logfile
retry_cnt ||= 0
@stat = ::File.stat(@fn)
return unless @lockfile
return if @inode == @stat.ino
@io.close rescue nil
open_logfile
rescue SystemCallError
raise if retry_cnt > 3
retry_cnt += 1
sleep 0.08
retry
end
end # class RollingFile
end # module Logging::Appenders
# EOF