The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
# Copyright (C) 2011-2012, 2014 Rocky Bernstein <rocky@cpan.org>
# -*- coding: utf-8 -*-
use warnings; no warnings 'redefine';
use rlib '../../../..';

# FIXME: Complete doesn't work if in package.
use Devel::Trepan::Complete;
package Devel::Trepan::CmdProcessor::Command::Kill;

use if !@ISA, Devel::Trepan::CmdProcessor::Command ;

use vars qw(@ISA);

unless (@ISA) {
    eval <<'EOE';
use constant ALIASES  => ('kill!');
use constant CATEGORY => 'running';
use constant SHORT_HELP => 'Send this process a POSIX signal';
use constant MIN_ARGS   => 0; # Need at least this many
use constant MAX_ARGS   => 1; # Need at most this many - undef -> unlimited.
EOE
}
use strict;

@ISA = @CMD_ISA;
use vars @CMD_VARS;  # Value inherited from parent

our $NAME = set_name();
=pod

=head2 Synopsis:

=cut
our $HELP = <<'HELP';
=pod

B<kill>[B<!>] [I<signal-number>|I<signal-name>]

Kill execution of program being debugged.

Equivalent of C<kill('KILL', $$)>. This is an unmaskable
signal. Use this when all else fails, e.g. in thread code, use this.

If you are in interactive mode, you are prompted to confirm killing.
However when this command is aliased from a command ending in C<!>, no
questions are asked.

=head2 Examples:

 kill
 kill KILL # same as above
 kill kill # same as above
 kill -9   # same as above
 kill  9   # same as above
 kill! 9   # same as above, but no questions asked
 kill unconditionally # same as above
 kill TERM # Send "TERM" signal

=head2 See also:

See also L<C<set
confirm>|Devel::Trepan::CmdProcessor::Command::Set::Confirm> and
L<C<quit>|Devel::Trepan::CmdProcessor::Command::Quit>.

=cut
HELP

sub complete($$) {
    my ($self, $prefix) = @_;
    my @matches = Devel::Trepan::Complete::signal_complete($prefix);
    push @matches, 'unconditionally' if 0 == index('unconditionally', $prefix);
    sort @matches;
}

# This method runs the command
sub run($$) {
    my ($self, $args) = @_;
    my $unconditional = substr($args->[0], -1, 1) eq '!';
    my $sig;
    if (scalar(@$args) > 1) {
        $sig = uc($args->[1]);
        unless ( ($sig =~ /[+-]?\d+/) || exists $SIG{$sig} ) {
            $self->errmsg("Signal name '${sig}' is not a signal I know about.");
            return;
        }
    } else {
        if ($unconditional || $self->{proc}->confirm('Really quit?', 0)) {
            $sig = 'KILL';
        } else {
            $self->msg('Kill not confirmed.');
            return;
        }
    }
    if (kill(0, $$)) {
        # Force finalization on interface.
        $self->{proc}{interfaces} = [] if
            'KILL' eq $sig || 9 eq $sig || -9 eq $sig;
        if (kill($sig, $$)) {
            $self->msg("kill ${sig} successfully sent to process $$");
        } else {
            $self->errmsg("Kill ${sig} to process $$ not accepted: $!")
        }
    } else {
        $self->errmsg(["Unable kill ${sig} to process $$",
                       "Different uid and not super-user?"]);
    }
}

unless (caller()) {
    require Devel::Trepan::CmdProcessor;
    my $proc = Devel::Trepan::CmdProcessor->new;
    my $cmd = __PACKAGE__->new($proc);
    print $cmd->{help}, "\n";
    print join(', ', @{$cmd->{aliases}}), "\n";
    print "min args: ", eval('$' . __PACKAGE__ . "::MIN_ARGS"), "\n";
    for my $arg ('hu', 'HU', '', 1, '-9') {
        printf "complete($arg) => %s\n", join(", ", $cmd->complete($arg));
    }
    for my $arg (qw(fooo 100 1 -1 HUP -9)) {
      print "$NAME ${arg}\n";
      $cmd->run([$NAME, $arg]);
      my $sep = '=' x 40 . "\n";
      print $sep;
    }
}

1;