Text::Restructured::Directive::system


Text-Restructured documentation Contained in the Text-Restructured distribution.

Index


Code Index:

``:lenient:`` Causes the exit code for the subprocess to be ignored. ``:literal:`` Interpret the returned value as a literal block.

If this option is not present, the return value is interpreted as reStructuredText and is parsed again.

Any error returned by the shell generates a level 3 error message. To see the output of a command that is expected to generate an error, do::

  .. system:: <your command> 2>&1 | cat

The following defines are processed by the system directive:

-D trusted Must be specified for system directives to be processed. This requirement is to prevent a system directive in a file written elsewhere from doing destructive things on your computer. =end Description =end reST =cut


Text-Restructured documentation Contained in the Text-Restructured distribution.

# $Id: system.pm 5476 2007-12-18 15:55:04Z mnodine $
# Copyright (C) 2002-2005 Freescale Semiconductor, Inc.
# Distributed under terms of the Perl license, which is the disjunction of
# the GNU General Public License (GPL) and the Artistic License.

# This package implements the system directive for the perl implementation
# of reStructuredText.

package Text::Restructured::Directive::system;

($VERSION) = q$Revision: 5476 $ =~ /(\d+)/g;

BEGIN {
    Text::Restructured::Directive::handle_directive
	(system => \&Text::Restructured::Directive::system::main);
}

use Text::Restructured::Directive::perl;
use vars qw($DOM);
BEGIN {
    *DOM = "Text::Restructured::DOM";
}

# Plug-in handler for system directives.
# Arguments: directive name, parent, source, line number, directive text, 
#            literal text
# Returns: array of DOM objects
sub main {
    my($parser, $name, $parent, $source, $lineno, $dtext, $lit) = @_;
    my @optlist = qw(lenient literal);
    my $dhash = Text::Restructured::Directive::parse_directive
	($parser, $dtext, $lit, $source, $lineno, \@optlist);
    return $dhash if ref($dhash) eq $DOM;
    my($args, $options) = map($dhash->{$_}, qw(args options));

    return Text::Restructured::Directive::system_msg
	($parser, $name, 3, $source, $lineno, "Argument(s) required.", $lit)
	if $args =~ /^$/;

    Text::Restructured::Directive::perl::create_safe($parser, $source, $lineno);
    $args =~ s/\n/ /g;
    my $code = << "EOS";
my \$text = `\Q$args\E 2>$$.stderr`;
open ERR, "$$.stderr";
my \$errmsg = <ERR>;
close ERR;
unlink "$$.stderr";
(\$text, \$?, \$errmsg)
EOS
    $Perl::safe_world->block_stderr;
    my ($text, $syserr, $errmsg) =
      Text::Restructured::Directive::perl::evaluate_code
         ($parser, $code, $source, $lineno, $lit);
    $text = '' unless defined $text;
    my $err = $@ =~ /trapped by/ ? "$@Run with -D trusted if you believe the code is safe" : $@;
    return $parser->system_message(4, $source, $lineno,
			       qq(Error executing "$name" directive: $err.),
			       $lit)
	if $@;

    return Text::Restructured::Directive::system_msg
    ($parser, $name, 3, $source, $lineno, "Non-zero exit code: " .
     ($errmsg || $text) )
    if $syserr && ! defined $options->{lenient};
    my $newsource = qq($name directive at $source, line $lineno);
    $text .= $errmsg if defined $errmsg;
    if (defined $options->{literal} && $text ne '') {
	my $lb = $DOM->new('literal_block', %Text::Restructured::XML_SPACE,
			 source=>$newsource);
	$lb->append($DOM->newPCDATA($text));
	return $lb;
    }
    elsif ($parent->tag eq 'substitution_definition') {
	chomp $text;
	return $DOM->newPCDATA($text);
    }
    else {
	$parser->Paragraphs($parent, $text, $newsource, 1) if $text ne '';
    }

    return;
}

1;