mirror of
				https://github.com/pragma-/pbot.git
				synced 2025-11-04 00:27:23 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			96 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			96 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
# File: RunCommand.pm
 | 
						|
#
 | 
						|
# Purpose: Runs a system command, streaming each line of output in real-time.
 | 
						|
#
 | 
						|
# WARNING: The `runcmd` command will allow a user to run any command on your
 | 
						|
# system. Do not give out the `can-runcmd` capability to anyone you do not
 | 
						|
# absolutely trust 100%. Instead, make a locked-down factoid; i.e.:
 | 
						|
#
 | 
						|
#   factalias ls runcmd ls $args
 | 
						|
#   factset ls cap-override can-runcmd
 | 
						|
#   factset ls locked 1
 | 
						|
#
 | 
						|
# The above will create an `ls` alias that can only run `runcmd ls $args` and
 | 
						|
# cannot be modified by anybody. The cap-override is necessary so the factoid
 | 
						|
# itself has permission to use `runcmd` regardless of whether the user has the
 | 
						|
# `can-runcmd` capability.
 | 
						|
#
 | 
						|
# This plugin is not in data/plugin_autoload. Load at your own risk.
 | 
						|
 | 
						|
# SPDX-FileCopyrightText: 2021-2023 Pragmatic Software <pragma78@gmail.com>
 | 
						|
# SPDX-License-Identifier: MIT
 | 
						|
 | 
						|
package PBot::Plugin::RunCommand;
 | 
						|
use parent 'PBot::Plugin::Base';
 | 
						|
 | 
						|
use PBot::Imports;
 | 
						|
 | 
						|
use IPC::Run qw/start pump finish/;
 | 
						|
 | 
						|
sub initialize($self, %conf) {
 | 
						|
    $self->{pbot}->{commands}->add(
 | 
						|
        name => 'runcmd',
 | 
						|
        help => 'Executes a system command and outputs each line in real-time',
 | 
						|
        requires_cap => 1,
 | 
						|
        subref => sub { $self->cmd_runcmd(@_) },
 | 
						|
    );
 | 
						|
}
 | 
						|
 | 
						|
sub unload($self) {
 | 
						|
    $self->{pbot}->{commands}->remove('runcmd');
 | 
						|
}
 | 
						|
 | 
						|
sub cmd_runcmd($self, $context) {
 | 
						|
    my @args = $self->{pbot}->{interpreter}->split_line($context->{arguments}, strip_quotes => 1);
 | 
						|
 | 
						|
    my ($in, $out, $err);
 | 
						|
 | 
						|
    my $h = eval { start \@args, \$in, \$out, \$err };
 | 
						|
 | 
						|
    if ($@) {
 | 
						|
        return "Error starting command: $@";
 | 
						|
    }
 | 
						|
 | 
						|
    my $lines = 0;
 | 
						|
 | 
						|
    while (pump $h) {
 | 
						|
        $lines += $self->send_lines($context, \$out);
 | 
						|
        $lines += $self->send_lines($context, \$err);
 | 
						|
    }
 | 
						|
 | 
						|
    finish $h;
 | 
						|
 | 
						|
    $lines += $self->send_lines($context, \$out, 1);
 | 
						|
    $lines += $self->send_lines($context, \$err, 1);
 | 
						|
 | 
						|
    return "No output." if not $lines;
 | 
						|
}
 | 
						|
 | 
						|
sub send_lines($self, $context, $buffer, $send_all = 0) {
 | 
						|
    my $lines = 0;
 | 
						|
 | 
						|
    my $regex;
 | 
						|
 | 
						|
    if ($send_all) {
 | 
						|
        # all lines
 | 
						|
        $regex = qr/(.{1,450})/;
 | 
						|
    } else {
 | 
						|
        # lines that end with a newline
 | 
						|
        $regex = qr/^(.{1,450})\s+/;
 | 
						|
    }
 | 
						|
 | 
						|
    while ($$buffer =~ s/$regex//) {
 | 
						|
        my $line = $1;
 | 
						|
        $line =~ s/^\s+|\s+$//g;
 | 
						|
 | 
						|
        if (length $line) {
 | 
						|
            $self->{pbot}->{conn}->privmsg($context->{from}, $line);
 | 
						|
            $lines++;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return $lines;
 | 
						|
}
 | 
						|
 | 
						|
1;
 |