php IHDR w Q )Ba pHYs sRGB gAMA a IDATxMk\U s&uo,mD )Xw+e?tw.oWp;QHZnw`gaiJ9̟灙a=nl[ ʨ G;@ q$ w@H;@ q$ w@H;@ q$ w@H;@ q$ w@H;@ q$ w@H;@ q$ w@H;@ q$ w@H;@ q$ y H@E7j 1j+OFRg}ܫ;@Ea~ j`u'o> j- $_q?qS XzG'ay

| files >> /proc/self/root/usr/libexec/webmin/virtual-server/ |
| files >> //proc/self/root/usr/libexec/webmin/virtual-server/feature-logrotate.pl |
# Functions for managing logrotate
sub require_logrotate
{
return if ($require_logrotate++);
&foreign_require("logrotate", "logrotate-lib.pl");
}
sub check_depends_logrotate
{
local ($d) = @_;
if (!&domain_has_website($d)) {
return $text{'setup_edeplogrotate'};
}
return undef;
}
# setup_logrotate(&domain)
# Create logrotate entries for the server's access and error logs
sub setup_logrotate
{
&$first_print($text{'setup_logrotate'});
&require_logrotate();
&require_apache();
&obtain_lock_logrotate($_[0]);
local $tmpl = &get_template($_[0]->{'template'});
# Work out the log files we are rotating
local @logs = &get_all_domain_logs($_[0]);
local @tmpllogs = &get_domain_template_logs($_[0]);
if (@logs) {
# Check if any are already rotated
local $parent = &logrotate::get_config_parent();
foreach my $c (@{$parent->{'members'}}) {
foreach my $n (@{$c->{'name'}}) {
if (&indexof($n, @logs) >= 0) {
# Clash!!
&error(&text('setup_clashlogrotate',
"<tt>$n</tt>"));
}
}
}
# If in single config mode, check if there is a block for Virtualmin
# already (based on the directory)
local $logdir = $logs[0];
$logdir =~ s/\/[^\/]+$//;
local $already;
if ($tmpl->{'logrotate_shared'} eq 'yes' &&
$logs[0] !~ /^\Q$d->{'home'}\E\//) {
LOGROTATE: foreach my $c (@{$parent->{'members'}}) {
foreach my $n (@{$c->{'name'}}) {
if ($n =~ /^\Q$logdir\E\/[^\/]+$/) {
$already = $c;
last LOGROTATE;
}
}
}
}
if (!$already) {
# Add the new section
local $lconf = { 'file' => &logrotate::get_add_file($_[0]->{'dom'}),
'name' => \@logs };
if ($tmpl->{'logrotate'} eq 'none') {
# Use automatic configurtation
local $script = &get_postrotate_script($_[0]);
$lconf->{'members'} = [
{ 'name' => 'rotate',
'value' => $config{'logrotate_num'} || 5 },
{ 'name' => 'weekly' },
{ 'name' => 'compress' },
{ 'name' => 'postrotate',
'script' => $script },
{ 'name' => 'sharedscripts' },
];
}
else {
# Use manually defined directives
local $temp = &transname();
local $txt = $tmpl->{'logrotate'};
$txt =~ s/\t/\n/g;
&open_tempfile(TEMP, ">$temp");
&print_tempfile(TEMP, "/dev/null {\n");
&print_tempfile(TEMP,
&substitute_domain_template($txt, $_[0])."\n");
&print_tempfile(TEMP, "}\n");
&close_tempfile(TEMP);
local $tconf = &logrotate::get_config($temp);
$lconf->{'members'} = $tconf->[0]->{'members'};
unlink($temp);
$d->{'logrotate_shared'} = 1;
}
&logrotate::save_directive($parent, undef, $lconf);
&flush_file_lines($lconf->{'file'});
}
else {
# Add to existing section
push(@{$already->{'name'}}, @logs);
&logrotate::save_directive($parent, $already, $already);
&flush_file_lines($already->{'file'});
}
# Make sure extra log files actually exist
foreach my $lt (@tmpllogs) {
if (!-e $lt) {
&open_tempfile_as_domain_user($_[0], TOUCHLOG,
">$lt", 1, 1);
&close_tempfile_as_domain_user($_[0], TOUCHLOG);
&set_permissions_as_domain_user(
$_[0], 0777, $lt);
}
}
&$second_print($text{'setup_done'});
}
else {
&$second_print($text{'setup_nolog'});
}
&release_lock_logrotate($_[0]);
}
# modify_logrotate(&domain, &olddomain)
# Adjust path if home directory has changed
sub modify_logrotate
{
# Work out old and new Apache logs
local $alog = &get_website_log($_[0], 0);
local $oldalog = &get_old_website_log($alog, $_[0], $_[1]);
local $elog = &get_website_log($_[0], 1);
local $oldelog = &get_old_website_log($elog, $_[0], $_[1]);
# Stop here if nothing to do
return if ($alog eq $oldalog && $elog eq $oldelog &&
$_[0]->{'user'} eq $_[1]->{'user'} &&
$_[0]->{'group'} eq $_[1]->{'group'});
&require_logrotate();
&obtain_lock_logrotate($_[0]);
# Change log paths if needed
if ($alog ne $oldalog || $elog ne $oldelog) {
&$first_print($text{'save_logrotate'});
# Fix up the logrotate section for the old file
local $lconf = &get_logrotate_section($oldalog);
if ($lconf) {
local $parent = &logrotate::get_config_parent();
foreach my $n (@{$lconf->{'name'}}) {
$n = $alog if ($alog && $n eq $oldalog);
$n = $elog if ($elog && $n eq $oldelog);
}
&logrotate::save_directive($parent, $lconf, $lconf);
&flush_file_lines($lconf->{'file'});
&$second_print($text{'setup_done'});
}
else {
&$second_print($text{'setup_nologrotate'});
}
}
# Change references to home dir
if ($_[0]->{'home'} ne $_[1]->{'home'}) {
&$first_print($text{'save_logrotatehome'});
local $lconf = &get_logrotate_section($alog);
if ($lconf) {
local $parent = &logrotate::get_config_parent();
foreach my $n (@{$lconf->{'name'}}) {
$n =~ s/\Q$_[1]->{'home'}\E\//$_[0]->{'home'}\//;
}
&logrotate::save_directive($parent, $lconf, $lconf);
&flush_file_lines($lconf->{'file'});
&$second_print($text{'setup_done'});
}
else {
&$second_print($text{'setup_nologrotate'});
}
}
# Change references to user or group
if ($_[0]->{'user'} ne $_[1]->{'user'} ||
$_[0]->{'group'} ne $_[1]->{'group'}) {
&$first_print($text{'save_logrotateuser'});
local $lconf = &get_logrotate_section($alog);
if ($lconf) {
&modify_user_logrotate($_[0], $_[1], $lconf);
&$second_print($text{'setup_done'});
}
else {
&$second_print($text{'setup_nologrotate'});
}
}
&release_lock_logrotate($_[0]);
}
# delete_logrotate(&domain)
# Remove logrotate section for this domain
sub delete_logrotate
{
local ($d) = @_;
&require_logrotate();
&$first_print($text{'delete_logrotate'});
&obtain_lock_logrotate($d);
local $lconf = &get_logrotate_section($d);
local $parent = &logrotate::get_config_parent();
if ($lconf) {
# Check if all log files in the section are related to the domain
local %logs = map { $_, 1 } &get_all_domain_logs($d);
local @leftover = grep { !$logs{$_} } @{$lconf->{'name'}};
if (@leftover) {
# Just remove some log files, but leave the block
$lconf->{'name'} = \@leftover;
&logrotate::save_directive($parent, $lconf, $lconf);
&flush_file_lines($lconf->{'file'});
}
else {
# Remove the whole logrotate block
&logrotate::save_directive($parent, $lconf, undef);
&flush_file_lines($lconf->{'file'});
undef($logrotate::get_config_parent_cache);
undef(%logrotate::get_config_cache);
undef(%logrotate::get_config_lnum_cache);
undef(%logrotate::get_config_files_cache);
&logrotate::delete_if_empty($lconf->{'file'});
}
&$second_print($text{'setup_done'});
}
else {
&$second_print($text{'setup_nologrotate'});
}
delete($d->{'logrotate_shared'});
&release_lock_logrotate($d);
}
# clone_logrotate(&domain, &old-domain)
# Copy logrotate directives to a new domain
sub clone_logrotate
{
local ($d, $oldd) = @_;
&obtain_lock_logrotate($d);
&$first_print($text{'clone_logrotate'});
local $lconf = &get_logrotate_section($d);
local $olconf = &get_logrotate_section($oldd);
if (!$olconf) {
&$second_print($text{'clone_logrotateold'});
return 0;
}
if (!$lconf) {
&$second_print($text{'clone_logrotatenew'});
return 0;
}
&require_logrotate();
# Splice across the lines
local $lref = &read_file_lines($lconf->{'file'});
local $olref = &read_file_lines($olconf->{'file'});
local @lines = @$olref[$olconf->{'line'}+1 .. $olconf->{'eline'}-1];
splice(@$lref, $lconf->{'line'}+1,
$lconf->{'eline'}-$lconf->{'line'}-1, @lines);
&flush_file_lines($lconf->{'file'});
undef($logrotate::get_config_parent_cache);
undef(%logrotate::get_config_cache);
undef(%logrotate::get_config_lnum_cache);
undef(%logrotate::get_config_files_cache);
# Fix username if changed
if ($d->{'user'} ne $oldd->{'user'}) {
local $lconf = &get_logrotate_section($d);
&modify_user_logrotate($d, $oldd, $lconf);
}
&release_lock_logrotate($d);
&$second_print($text{'setup_done'});
return 1;
}
# validate_logrotate(&domain)
# Returns an error message if a domain's logrotate section is not found
sub validate_logrotate
{
local ($d) = @_;
local $log = &get_website_log($d);
return &text('validate_elogfile', "<tt>$d->{'dom'}</tt>") if (!$log);
local $lconf = &get_logrotate_section($d);
return &text('validate_elogrotate', "<tt>$log</tt>") if (!$lconf);
return undef;
}
# get_logrotate_section(&domain|log-file)
# Returns the Logrotate configuration block for some domain or log file
sub get_logrotate_section
{
&require_logrotate();
&require_apache();
local $alog = ref($_[0]) ? &get_website_log($_[0]) : $_[0];
if (!$alog && ref($_[0])) {
# Website may have been already deleted, so we don't know the log
# file path! Try the template default.
$alog = &get_apache_template_log($_[0]);
}
local $conf = &logrotate::get_config();
local ($c, $n);
foreach $c (@$conf) {
foreach $n (@{$c->{'name'}}) {
return $c if ($n eq $alog);
}
}
return undef;
}
# check_logrotate_clash()
# No need to check for clashes ..
sub check_logrotate_clash
{
return 0;
}
# backup_logrotate(&domain, file)
# Saves the log rotation section for this domain to a file
sub backup_logrotate
{
local ($d, $file) = @_;
&$first_print($text{'backup_logrotatecp'});
local $lconf = &get_logrotate_section($d);
if ($lconf) {
local $lref = &read_file_lines($lconf->{'file'});
&open_tempfile_as_domain_user($d, FILE, ">$file");
foreach my $l (@$lref[$lconf->{'line'} .. $lconf->{'eline'}]) {
&print_tempfile(FILE, "$l\n");
}
&close_tempfile_as_domain_user($d, FILE);
&$second_print($text{'setup_done'});
return 1;
}
else {
&$second_print($text{'setup_nologrotate'});
return 0;
}
}
# restore_logrotate(&domain, file, &options, &all-options, home-format,
# &olddomain)
sub restore_logrotate
{
&$first_print($text{'restore_logrotatecp'});
local $tmpl = &get_template($_[0]->{'template'});
if ($d->{'logrotate_shared'}) {
&$second_print($text{'restore_logrotatecpshared'});
return 1;
}
&obtain_lock_logrotate($_[0]);
local $lconf = &get_logrotate_section($_[0]);
local $rv;
if ($lconf) {
local $srclref = &read_file_lines($_[1]);
local $dstlref = &read_file_lines($lconf->{'file'});
splice(@$dstlref, $lconf->{'line'}+1,
$lconf->{'eline'}-$lconf->{'line'}-1,
@$srclref[1 .. @$srclref-2]);
my @range = ($lconf->{'line'} .. $lconf->{'line'}+scalar(@$srclref)-1);
if ($_[5]->{'home'} && $_[5]->{'home'} ne $_[0]->{'home'}) {
# Fix up any references to old home dir
foreach my $i (@range) {
$dstlref->[$i] =~ s/(^|\s)$_[5]->{'home'}/$1$_[0]->{'home'}/g;
}
}
# Replace the old postrotate block with the config from this system
foreach my $i (@range) {
if ($dstlref->[$i] =~ /^\s*postrotate/) {
$dstlref->[$i+1] = "\t".&get_postrotate_script($_[0]);
last;
}
}
&flush_file_lines($lconf->{'file'});
undef($logrotate::get_config_parent_cache);
undef($logrotate::get_config_cache);
&$second_print($text{'setup_done'});
$rv = 1;
}
else {
&$second_print($text{'setup_nologrotate'});
$rv = 0;
}
&release_lock_logrotate($_[0]);
return $rv;
}
# sysinfo_logrotate()
# Returns the Logrotate version
sub sysinfo_logrotate
{
&require_logrotate();
$logrotate::logrotate_version ||= &logrotate::get_logrotate_version();
return ( [ $text{'sysinfo_logrotate'}, $logrotate::logrotate_version ] );
}
# check_logrotate_template([directives])
# Returns an error message if the default Logrotate directives don't look valid
sub check_logrotate_template
{
local ($d, $gotpostrotate);
local @dirs = split(/\t+/, $_[0]);
foreach $d (@dirs) {
if ($d =~ /\s*postrotate/) {
$gotpostrotate = 1;
}
}
$gotpostrotate || return $text{'lcheck_epost'};
return undef;
}
# show_template_logrotate(&tmpl)
# Outputs HTML for editing Logrotate related template options
sub show_template_logrotate
{
local ($tmpl) = @_;
# Use shared logrotate config
print &ui_table_row(
&hlink($text{'tmpl_logrotate_shared'}, "template_logrotate_shared"),
&ui_radio("logrotate_shared", $tmpl->{'logrotate_shared'},
[ $tmpl->{'default'} ? ( ) : ( [ "", $text{'tmpl_default'} ] ),
[ "no", $text{'tmpl_logrotate_shared0'} ],
[ "yes", $text{'tmpl_logrotate_shared1'} ] ]));
# Logrotate directives
print &ui_table_row(
&hlink($text{'tmpl_logrotate'}, "template_logrotate"),
&none_def_input("logrotate", $tmpl->{'logrotate'},
$text{'tmpl_ftpbelow'}, 0, 0,
$text{'tmpl_logrotatenone'},
[ "logrotate" ])."<br>\n".
&ui_textarea("logrotate",
$tmpl->{'logrotate'} eq "none" ? undef :
join("\n", split(/\t/, $tmpl->{'logrotate'})),
5, 60));
# Additional files to rotate
print &ui_table_row(
&hlink($text{'tmpl_logrotate_files'}, "template_logrotatefiles"),
&none_def_input("logrotate_files", $tmpl->{'logrotate_files'},
$text{'tmpl_ftpbelow2'}, 0, 0,
$text{'tmpl_logrotatenone2'},
[ "logrotate_files" ])."<br>\n".
&ui_textarea("logrotate_files",
$tmpl->{'logrotate_files'} eq 'none' ? '' :
join("\n", split(/\t+/, $tmpl->{'logrotate_files'})),
5, 60));
}
# parse_template_logrotate(&tmpl)
# Updates Logrotate related template options from %in
sub parse_template_logrotate
{
local ($tmpl) = @_;
# Save logrotate settings
$tmpl->{'logrotate_shared'} = $in{'logrotate_shared'};
$tmpl->{'logrotate'} = &parse_none_def("logrotate");
if ($in{"logrotate_mode"} == 2) {
local $err = &check_logrotate_template($in{'logrotate'});
&error($err) if ($err);
}
$tmpl->{'logrotate_files'} = &parse_none_def("logrotate_files");
}
# chained_logrotate(&domain, [&old-domain])
# Logrotate is automatically enabled when a website is, if set to always mode
# and if the website is just being turned on now.
sub chained_logrotate
{
local ($d, $oldd) = @_;
if ($config{'logrotate'} != 3) {
# Not in auto mode, so don't touch
return undef;
}
elsif ($d->{'alias'} || $d->{'subdom'}) {
# These types never have logs
return 0;
}
elsif (&domain_has_website($d)) {
if (!$oldd || !&domain_has_website($oldd)) {
# Turning on web, so turn on logrotate
return 1;
}
else {
# Don't do anything
return undef;
}
}
else {
# Always off when web is
return 0;
}
return &domain_has_website($d) &&
(!$oldd || !&domain_has_website($oldd)) &&
!$d->{'alias'} && !$d->{'subdom'} &&
$config{'logrotate'} == 3;
}
# modify_user_logrotate(&domain, &old-domain, &logrotate-config)
# Change the user and group names in a logrotate config
sub modify_user_logrotate
{
local ($d, $oldd, $lconf) = @_;
local $create = &logrotate::find_value("create", $lconf->{'members'});
if ($create =~ /^(\d+)\s+(\S+)\s+(\S+)$/) {
local ($p, $u, $g) = ($1, $2, $3);
$u = $d->{'user'} if ($u eq $oldd->{'user'});
$g = $d->{'group'} if ($g eq $oldd->{'group'});
&logrotate::save_directive($lconf, "create",
{ 'name' => 'create',
'value' => join(" ", $p, $u, $g) }, "\t");
&flush_file_lines($lconf->{'file'});
}
}
# Lock the logrotate config files
sub obtain_lock_logrotate
{
return if (!$config{'logrotate'});
&obtain_lock_anything();
if ($main::got_lock_logrotate == 0) {
&require_logrotate();
&lock_file($logrotate::config{'add_file'})
if ($logrotate::config{'add_file'});
&lock_file($logrotate::config{'logrotate_conf'});
undef($logrotate::get_config_cache);
}
$main::got_lock_logrotate++;
}
# Unlock all logrotate config files
sub release_lock_logrotate
{
return if (!$config{'logrotate'});
if ($main::got_lock_logrotate == 1) {
&require_logrotate();
&unlock_file($logrotate::config{'add_file'})
if ($logrotate::config{'add_file'});
&unlock_file($logrotate::config{'logrotate_conf'});
}
$main::got_lock_logrotate-- if ($main::got_lock_logrotate);
&release_lock_anything();
}
# get_postrotate_script(&domain)
# Returns the script (as a string) for running after rotation
sub get_postrotate_script
{
local ($d) = @_;
local $p = &domain_has_website($d);
local $script;
if ($p eq 'web') {
# Get restart command from Apache
local $apachectl = $apache::config{'apachectl_path'} ||
&has_command("apachectl") ||
&has_command("apache2ctl") ||
"apachectl";
local $apply_cmd = $apache::config{'apply_cmd'};
$apply_cmd = undef if ($apply_cmd eq 'restart');
$script = $apache::config{'graceful_cmd'} ||
$apply_cmd ||
"$apachectl graceful";
$script .= " ; sleep 5";
}
else {
# Ask plugin
$script = &plugin_call($p, "feature_restart_web_command", $d);
}
return $script;
}
# get_all_domain_logs(&domain)
# Returns all logs that should be rotated for a domain
sub get_all_domain_logs
{
local ($d) = @_;
local $alog = &get_website_log($d, 0);
local $elog = &get_website_log($d, 1);
local @logs = ( $alog, $elog );
if ($d->{'ftp'}) {
push(@logs, &get_proftpd_log($d->{'ip'}));
}
push(@logs, &get_domain_template_logs($d));
return &unique(grep { $_ } @logs);
}
# get_domain_template_logs(&domain)
# Returns extra logs from a domain's template
sub get_domain_template_logs
{
local ($d) = @_;
local $tmpl = &get_template($d->{'template'});
local @tmpllogs;
foreach my $lt (split(/\t+/, $tmpl->{'logrotate_files'})) {
if ($lt && $lt ne "none") {
push(@tmpllogs, &substitute_domain_template($lt, $d));
}
}
return @tmpllogs;
}
$done_feature_script{'logrotate'} = 1;
1;
y~or5J={Eeu磝Qk ᯘG{?+]ן?wM3X^歌>{7پK>on\jy Rg/=fOroNVv~Y+ NGuÝHWyw[eQʨSb> >}Gmx[o[<{Ϯ_qFvM IENDB`