in reply to Timer Question

fork a child, then monitor (alarm+waitpid) the child in the parent. I'm no expert at working with child processes, but I came up with the following. Once the child has run for $timeout seconds, the parent asks the child to terminate. If the child is still running 10 seconds later, the parent forcefully kills the child.
my @command = "sleep 10; echo awoke"; my $timeout = 5; my $cpid = fork(); die("Unable to fork: $!\n") if not defined $cpid; if (not $cpid) { # In child exec(@command); die("Unable to exec: $!\n"); } eval { local $SIG{ALRM} = sub { die "alarm\n" }; alarm($timeout); waitpid($cpid, 0); }; alarm(0); if ($@) { die $@ if $@ ne "alarm\n"; warn("Long running child. Attempting to terminate.\n"); kill 'TERM', $cpid; eval { local $SIG{ALRM} = sub { die "alarm\n" }; alarm(10); waitpid($cpid, 0); }; alarm(0); if ($@) { die $@ if $@ ne "alarm\n"; warn("Long running child. Attempting to kill.\n"); kill 'KILL', $cpid; eval { local $SIG{ALRM} = sub { die "alarm\n" }; alarm(5); waitpid($cpid, 0); }; alarm(0); if ($@) { die $@ if $@ ne "alarm\n"; warn("Unable to kill long running child.\n"); } else { warn("Long running child successfully killed.\n"); } } else { warn("Long running child successfully terminated.\n"); } }
Output:
$ perl script.pl Long running child. Attempting to terminate. Long running child successfully terminated.