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 copied and then
# truncated. The name of the copy indicates it is an older log file.
#
# 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 continue to be appended to the same log file
# (/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.
#
# If you do not wish to use numbered files when rolling, you can specify the
# :roll_by option as 'date'. This will use a date/time stamp to
# differentiate the older files from one another. If you configure your
# rolling file appender to roll daily and ignore the file size:
#
# /var/log/ruby.log => /var/log/ruby.20091225.log
#
# Where the date is expressed as %Y%m%d in the Time#strftime format.
#
# NOTE: this class is not safe to use when log messages are written to files
# on NFS mounts or other remote file system. It should only be used for log
# files on the local file system. The exception to this is when a single
# process is writing to the log file; remote file systems are safe to
# use in this case but still not recommended.
#
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.
# [:roll_by] How to name the rolled log files. This can be 'number' or
# 'date'.
#
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?
@fn = ::File.expand_path(@fn)
@fn_copy = @fn + '._copy_'
::Logging::Appenders::File.assert_valid_logfile(@fn)
# grab our options
@size = opts.getopt(:size, :as => Integer)
code = 'def sufficiently_aged?() false end'
@age_fn = @fn + '.age'
@age_fn_mtime = nil
case @age = opts.getopt(:age)
when 'daily'
code = <<-CODE
def sufficiently_aged?
@age_fn_mtime ||= ::File.mtime(@age_fn)
now = Time.now
if (now.day != @age_fn_mtime.day) or (now - @age_fn_mtime) > 86400
return true
end
false
end
CODE
when 'weekly'
code = <<-CODE
def sufficiently_aged?
@age_fn_mtime ||= ::File.mtime(@age_fn)
if (Time.now - @age_fn_mtime) > 604800
return true
end
false
end
CODE
when 'monthly'
code = <<-CODE
def sufficiently_aged?
@age_fn_mtime ||= ::File.mtime(@age_fn)
now = Time.now
if (now.month != @age_fn_mtime.month) or (now - @age_fn_mtime) > 2678400
return true
end
false
end
CODE
when Integer, String
@age = Integer(@age)
code = <<-CODE
def sufficiently_aged?
@age_fn_mtime ||= ::File.mtime(@age_fn)
if (Time.now - @age_fn_mtime) > @age
return true
end
false
end
CODE
end
FileUtils.touch(@age_fn) if @age and !test(?f, @age_fn)
meta = class << self; self end
meta.class_eval code, __FILE__, __LINE__
# we are opening the file in read/write mode so that a shared lock can
# be used on the file descriptor => http://pubs.opengroup.org/onlinepubs/009695399/functions/fcntl.html
super(name, ::File.new(@fn, 'a+'), opts)
# setup the file roller
@roller =
case opts.getopt(:roll_by)
when 'number'; NumberedRoller.new(@fn, opts)
when 'date'; DateRoller.new(@fn, opts)
else
(@age and !@size) ?
DateRoller.new(@fn, opts) :
NumberedRoller.new(@fn, opts)
end
# if the truncate flag was set to true, then roll
roll_now = opts.getopt(:truncate, false)
if roll_now
copy_truncate
@roller.roll_files
end
end
# Returns the path to the logfile.
#
def filename() @fn.dup end
# Reopen the connection to the underlying logging destination. If the
# connection is currently closed then it will be opened. If the connection
# is currently open then it will be closed and immediately opened.
#
def reopen
@mutex.synchronize {
if defined? @io and @io
flush
@io.close rescue nil
end
@io = ::File.new(@fn, 'a+')
}
super
self
end
private
# 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 canonical_write( str )
return self if @io.nil?
@io.flock_sh { @io.syswrite(str) }
if roll_required?
@io.flock? {
@age_fn_mtime = nil
copy_truncate if roll_required?
}
@roller.roll_files
end
self
rescue StandardError => err
self.level = :off
::Logging.log_internal {"appender #{name.inspect} has been disabled"}
::Logging.log_internal(-2) {err}
end
# Returns +true+ if the log file needs to be rolled.
#
def roll_required?
return false if ::File.exist?(@fn_copy) and (Time.now - ::File.mtime(@fn_copy)) < 180
# check if max size has been exceeded
s = @size ? ::File.size(@fn) > @size : false
# check if max age has been exceeded
a = sufficiently_aged?
return (s || a)
end
# Copy the contents of the logfile to another file. Truncate the logfile
# to zero length. This method will set the roll flag so that all the
# current logfiles will be rolled along with the copied file.
#
def copy_truncate
return unless ::File.exist?(@fn)
FileUtils.concat @fn, @fn_copy
@io.truncate 0
# touch the age file if needed
if @age
FileUtils.touch @age_fn
@age_fn_mtime = nil
end
@roller.roll = true
end
# :stopdoc:
class NumberedRoller
attr_accessor :roll
def initialize( fn, opts )
# 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}"
@fn_copy = fn + '._copy_'
@keep = opts.getopt(:keep, :as => Integer)
@roll = false
end
def roll_files
return unless @roll and ::File.exist?(@fn_copy)
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 copied log file
::File.rename(@fn_copy, sprintf(@logname_fmt, 1))
ensure
@roll = false
end
end
class DateRoller
attr_accessor :roll
def initialize( fn, opts )
@fn_copy = fn + '._copy_'
@roll = false
@keep = opts.getopt(:keep, :as => Integer)
ext = ::File.extname(fn)
bn = ::File.join(::File.dirname(fn), ::File.basename(fn, ext))
if @keep
@rgxp = %r/\.(\d+)(-\d+)?#{Regexp.escape(ext)}\z/
@glob = "#{bn}.*#{ext}"
end
if %w[daily weekly monthly].include?(opts.getopt(:age)) and !opts.getopt(:size)
@logname_fmt = "#{bn}.%Y%m%d#{ext}"
else
@logname_fmt = "#{bn}.%Y%m%d-%H%M%S#{ext}"
end
end
def roll_files
return unless @roll and ::File.exist?(@fn_copy)
# reanme the copied log file
::File.rename(@fn_copy, Time.now.strftime(@logname_fmt))
# prune old log files
if @keep
files = Dir.glob(@glob).find_all {|fn| @rgxp =~ fn}
length = files.length
if length > @keep
files.sort {|a,b| b <=> a}.last(length-@keep).each {|fn| ::File.delete fn}
end
end
ensure
@roll = false
end
end
# :startdoc:
end # class RollingFile
end # module Logging::Appenders