File: //proc/self/root/usr/share/perl5/NeedRestart.pm
# needrestart - Restart daemons after library updates.
#
# Authors:
# Thomas Liske <thomas@fiasko-nw.net>
#
# Copyright Holder:
# 2013 - 2022 (C) Thomas Liske [http://fiasko-nw.net/~thomas/]
#
# License:
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this package; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
#
package NeedRestart;
use strict;
use warnings;
use Module::Find;
use NeedRestart::Utils;
use NeedRestart::CONT;
use Sort::Naturally;
use constant {
NEEDRESTART_PRIO_NOAUTO => 0,
NEEDRESTART_PRIO_LOW => 1,
NEEDRESTART_PRIO_MEDIUM => 10,
NEEDRESTART_PRIO_HIGH => 100,
};
require Exporter;
our @ISA = qw(Exporter);
our @EXPORT = qw(
NEEDRESTART_PRIO_NOAUTO
NEEDRESTART_PRIO_LOW
NEEDRESTART_PRIO_MEDIUM
NEEDRESTART_PRIO_HIGH
needrestart_ui
needrestart_ui_list
needrestart_interp_check
needrestart_interp_source
needrestart_cont_check
needrestart_cont_get
needrestart_cont_cmd
);
our @EXPORT_OK = qw(
needrestart_ui_register
needrestart_ui_init
needrestart_interp_register
needrestart_cont_register
);
our %EXPORT_TAGS = (
ui => [qw(
NEEDRESTART_PRIO_LOW
NEEDRESTART_PRIO_MEDIUM
NEEDRESTART_PRIO_HIGH
needrestart_ui_register
needrestart_ui_init
)],
interp => [qw(
needrestart_interp_register
)],
cont => [qw(
needrestart_cont_register
)],
);
our $VERSION = '3.6';
my $LOGPREF = '[Core]';
my %UIs;
sub needrestart_ui_register($$) {
my $pkg = shift;
my $prio = shift;
$UIs{$pkg} = $prio;
}
sub needrestart_ui_init($$) {
my $verbosity = shift;
my $prefui = shift;
# load preferred UI module
if(defined($prefui)) {
return if(eval "use $prefui; 1;");
}
# autoload UI modules
foreach my $module (findsubmod NeedRestart::UI) {
unless(eval "use $module; 1;") {
warn "Error loading $module: $@\n" if($@ && ($verbosity > 1));
}
}
}
sub needrestart_ui {
my $verbosity = shift;
my $prefui = shift;
needrestart_ui_init($verbosity, $prefui) unless(%UIs);
my ($ui) = sort { ncmp($UIs{$b}, $UIs{$a}) } grep {
($UIs{$_} != NEEDRESTART_PRIO_NOAUTO) || ( defined($prefui) && ($prefui eq $_) )
} keys %UIs;
return undef unless($ui);
print STDERR "$LOGPREF Using UI '$ui'...\n" if($verbosity > 1);
return $ui->new($verbosity);
}
sub needrestart_ui_list {
my $verbosity = shift;
my $prefui = shift;
needrestart_ui_init($verbosity, $prefui) unless(%UIs);
return (sort { ncmp($UIs{$b}, $UIs{$a}) } keys %UIs);
}
my %Interps;
my %InterpCache;
my $idebug;
sub needrestart_interp_register($) {
my $pkg = shift;
$Interps{$pkg} = new $pkg($idebug);
}
sub needrestart_interp_init($) {
$idebug = shift;
# autoload Interp modules
foreach my $module (findsubmod NeedRestart::Interp) {
unless(eval "use $module; 1;") {
warn "Error loading $module: $@\n" if($@ && $idebug);
}
}
}
sub needrestart_interp_check($$$$$) {
my $debug = shift;
my $pid = shift;
my $bin = shift;
my $blacklist = shift;
my $tolerance = shift;
needrestart_interp_init($debug) unless(%Interps);
foreach my $interp (values %Interps) {
if($interp->isa($pid, $bin)) {
print STDERR "$LOGPREF #$pid is a ".(ref $interp)."\n" if($debug);
my $ps = nr_ptable_pid($pid);
my %files = $interp->files($pid, \%InterpCache);
foreach my $path (keys %files) {
next unless(scalar grep { $path =~ /$_/; } @{$blacklist});
print STDERR "$LOGPREF blacklisted: $path\n" if($debug);
delete($files{$path});
}
if(grep {!defined($_) || $_ > $ps->start + $tolerance} values %files) {
if($debug) {
print STDERR "$LOGPREF #$pid uses obsolete script file(s):";
print STDERR join("\n$LOGPREF #$pid ", '', map {(!defined($files{$_}) || $files{$_} > $ps->start ? $_ : ())} keys %files);
print STDERR "\n";
}
return 1;
}
}
}
return 0;
}
sub needrestart_interp_source($$$) {
my $debug = shift;
my $pid = shift;
my $bin = shift;
needrestart_interp_init($debug) unless(%Interps);
foreach my $interp (values %Interps) {
if($interp->isa($pid, $bin)) {
print STDERR "$LOGPREF #$pid is a ".(ref $interp)."\n" if($debug);
my $src = $interp->source($pid);
print STDERR "$LOGPREF #$pid source is ".(defined($src) ? $src : 'UNKNOWN')."\n" if($debug);
return ($src) if(defined($src));;
return ();
}
}
return ();
}
my %CONT;
my $ndebug;
sub needrestart_cont_register($) {
my $pkg = shift;
$CONT{$pkg} = new $pkg($ndebug);
}
sub needrestart_cont_init($) {
$ndebug = shift;
# autoload CONT modules
foreach my $module (findsubmod NeedRestart::CONT) {
unless(eval "use $module; 1;") {
warn "Error loading $module: $@\n" if($@ && $ndebug);
}
}
}
sub needrestart_cont_check($$$;$) {
my $debug = shift;
my $pid = shift;
my $bin = shift;
my $norestart = shift || 0;
needrestart_cont_init($debug) unless(scalar keys %CONT);
foreach my $cont (values %CONT) {
return 1 if($cont->check($pid, $bin, $norestart));
}
return 0;
}
sub needrestart_cont_get($) {
my $debug = shift;
return map {
my $cont = $_;
my $n = ref $cont;
$n =~ s/^NeedRestart::CONT:://;
my %c = $cont->get;
map {
("$n $_" => $c{$_});
} sort keys %c;
} sort { (ref $a) cmp (ref $b); } values %CONT;
}
1;