# See copyright, etc in below POD section. ###################################################################### package IPC::PidStat; require 5.004; use IPC::Locker; use Socket; use Time::HiRes qw(gettimeofday tv_interval); use IO::Socket; use Sys::Hostname; use Net::Domain; use POSIX; use strict; use vars qw($VERSION $Debug $Stat_Of_Pid_Supported %Local_Hostnames); use Carp; our @_Local_Responses; ###################################################################### #### Configuration Section # Other configurable settings. $Debug = 0; $VERSION = '1.491'; # True if pid existance can be detected by looking at /proc filesystem $Stat_Of_Pid_Supported = -e "/proc/1"; %Local_Hostnames = ('localhost' => 1, hostname() => 1, hostfqdn() => 1); ###################################################################### #### Creator sub new { # Establish the server @_ >= 1 or croak 'usage: IPC::PidStat->new ({options})'; my $proto = shift; my $class = ref($proto) || $proto; my $self = { socket=>undef, # IO::Socket handle of open socket tries=>5, # Documented port=>$IPC::Locker::Default_PidStat_Port, # Internal _host_ips => {}, # Resolved IP address of hosts @_,}; bless $self, $class; return $self; } sub open_socket { my $self = shift; # Open the socket return if $self->{_socket_fh}; $self->{_socket_fh} = IO::Socket::INET->new( Proto => 'udp') or die "$0: %Error, socket: $!"; } sub fh { my $self = shift; # Return socket file handle, for external select loops $self->open_socket(); #open if not already return $self->{_socket_fh}; } sub pid_request { my $self = shift; my %params = (host=>'localhost', pid=>$$, return_exist=>1, return_doesnt=>1, return_unknown=>1, @_); $self->open_socket(); #open if not already my $res; if ($Local_Hostnames{$params{host}}) { # No need to go via server, instead check locally my $res = $self->_local_response($params{pid}, $params{host}); push @_Local_Responses, $res if $res; # If unknown (undef response), forward to the server } if (!defined $res) { my $reqval = (($params{return_exist}?1:0) | ($params{return_doesnt}?2:0) | ($params{return_unknown}?4:0)); my $out_msg = "PIDR $params{pid} $params{host} $reqval\n"; my $ipnum = $self->{_host_ips}->{$params{host}}; if (!$ipnum) { # inet_aton("name") calls gethostbyname(), which chats with the # NS cache socket and NIS server. Too costly in a polling loop. $ipnum = inet_aton($params{host}) or die "%Error: Can't find host $params{host}\n"; $self->{_host_ips}->{$params{host}} = $ipnum; } my $dest = sockaddr_in($self->{port}, $ipnum); $self->fh->send($out_msg,0,$dest); } } sub recv_stat { my $self = shift; my $in_msg; if ($#_Local_Responses >= 0) { $in_msg = shift @_Local_Responses; print "Got local response $in_msg\n" if $Debug; } else { $self->fh->recv($in_msg, 8192) or return undef; print "Got server response $in_msg\n" if $Debug; } if ($in_msg =~ /^EXIS (\d+) (\d+) (\S+)/) { # PID server response my $pid=$1; my $exists = $2; my $hostname = $3; print " Pid $pid Exists on $hostname? $exists\n" if $Debug; return ($pid, $exists, $hostname); } elsif ($in_msg =~ /^UNKN (\d+) (\s+) (\S+)/) { # PID not determinate return undef; } return undef; } sub pid_request_recv { my $self = shift; my @params = @_; for (my $try=0; $try<$self->{tries}; $try++) { $self->pid_request(@params); my @recved; eval { local $SIG{ALRM} = sub { die "Timeout\n"; }; alarm(1); @recved = $self->recv_stat(); alarm(0); }; alarm(0) if $@; return @recved if defined $recved[0]; } return undef; } ###################################################################### #### Status checking sub ping_status { my $self = shift; my %params = (pid => 1, # Init. host => $self->{host}, @_, ); # Return OK and status message, for nagios like checks my $start_time = [gettimeofday()]; my ($epid, $eexists, $ehostname) = eval { return $self->pid_request_recv(%params); }; my $elapsed = tv_interval ( $start_time, [gettimeofday]); if (!$eexists) { return ({ok=>undef,status=>"No response from pidstatd on $self->{host}:$self->{port}"}); } else { return ({ok=>1,status=>sprintf("%1.3f second response on $self->{host}:$self->{port}", $elapsed)}); } } ###################################################################### #### Local messages sub _local_response { my $self = shift; my $pid = shift; my $host = shift; my $exists = IPC::PidStat::local_pid_exists($pid); if ($exists) { return "EXIS $pid $exists $host"; # PID response } elsif (defined $exists) { # Known not to exist return "EXIS $pid $exists $host"; # PID response } else { # Perhaps we're not running as root? return undef; } } ###################################################################### #### Static Accessors our $_Hostfqdn; sub hostfqdn { # Return hostname() including domain name $_Hostfqdn = Net::Domain::hostfqdn() if !defined $_Hostfqdn; return $_Hostfqdn; } ###################################################################### #### Utilities sub local_pid_doesnt_exist { my $result = local_pid_exists(@_); # Return 0 if a pid exists, 1 if not, undef (or second argument) if unknown return undef if !defined $result; return !$result; } sub local_pid_exists { my $pid = shift; # Return 1 if a pid exists, 0 if not, undef (or second argument) if unknown # We can't just call kill, because if there's a different user running the # process, we'll get an error instead of a result. $! = undef; my $exists = (kill (0,$pid))?1:0; if ($!) { if ($! == POSIX::ESRCH) { $exists = 0; } elsif ($! == POSIX::EPERM # Sigh, different user? && $Stat_Of_Pid_Supported ) { # This system supports /proc $exists = (-e "/proc/$pid") ? 1:0; } else { $exists = undef; # Unknown reason } } return $exists; } ###################################################################### #### Package return 1; =pod =head1 NAME IPC::PidStat - Process ID existence test =head1 SYNOPSIS use IPC::PidStat; my $exister = new IPC::PidStat( port=>1234, ); $exister->pid_request(host=>'foo', pid=>$pid) while (1) { # Poll receiving callbacks my ($epid, $eexists, $ehostname) = $exister->recv_stat(); print "Pid $epid ",($eexists?'exists':'dead'),"\n" if $ehostname; } =head1 DESCRIPTION L allows remote requests to be made to the L, to determine if a PID is running on the daemon's machine. PidStat uses UDP, and as such results are fast but may be unreliable. Furthermore, the pidstatd may not even be running on the remote machine, so responses should never be required before an application program makes progress. =head1 METHODS =over 4 =item new ([parameter=>value ...]); Creates a new object for later use. See the PARAMETERS section. =item pid_request (host=>$host, pid=>$pid); Sends a request to the specified host's server to see if the specified PID exists. The optional parameters return_exist=>0, return_doesnt=>0 and return_unknown=>0 improve performance by suppressing return messages if the specified pid exists, doesn't exist, or has unknown state respectively. Pidstatd versions before 1.480 ignore this flag, so the return code from recv_stat should not assume the undesired return types will be suppressed. =item pid_request_recv (host=>$host, pid=>$pid); Calls pid_request and returns the recv_stat reply. If the response fails to return in one second, it is retried up to 5 times, then undef is returned. =item recv_stat() Blocks waiting for any return from the server. Returns undef if none is found, or a 2 element array with the PID and existence flag. Generally this would be called inside a IO::Poll loop. =back =head1 STATIC METHODS =over 4 =item local_pid_doesnt_exist() Static call, not a method call. Return 0 if a pid exists, 1 if not. Return undef if it can't be determined. =item local_pid_exists() Static call, not a method call. Return 1 if a pid exists, 0 if not. Return undef if it can't be determined. =back =head1 PARAMETERS =over 4 =item port The port number (INET) of the pidstatd server. Defaults to 'pidstatd' looked up via /etc/services, else 1752. =back =head1 DISTRIBUTION The latest version is available from CPAN and from L. Copyright 2002-2012 by Wilson Snyder. This package is free software; you can redistribute it and/or modify it under the terms of either the GNU Lesser General Public License Version 3 or the Perl Artistic License Version 2.0. =head1 AUTHORS Wilson Snyder =head1 SEE ALSO L, L, L, L L =cut ######################################################################