mirror of
https://github.com/pragma-/pbot.git
synced 2025-01-25 19:44:26 +01:00
Refactoring compiler_vm a bit (1/2)
This commit is contained in:
parent
2832298af3
commit
8bc8a7a8b1
@ -19,8 +19,6 @@ eval {
|
|||||||
use lib 'languages';
|
use lib 'languages';
|
||||||
require "$language.pm";
|
require "$language.pm";
|
||||||
} or do {
|
} or do {
|
||||||
$language =~ s/^cfact_//;
|
|
||||||
|
|
||||||
my @modules = glob 'languages/*.pm';
|
my @modules = glob 'languages/*.pm';
|
||||||
my $found = 0;
|
my $found = 0;
|
||||||
my ($languages, $comma) = ('', '');
|
my ($languages, $comma) = ('', '');
|
||||||
@ -29,9 +27,11 @@ eval {
|
|||||||
$module = basename $module;
|
$module = basename $module;
|
||||||
$module =~ s/.pm$//;
|
$module =~ s/.pm$//;
|
||||||
next if $module =~ m/^_/;
|
next if $module =~ m/^_/;
|
||||||
require "$module.pm";
|
|
||||||
|
require "$module.pm" or die $!;
|
||||||
my $mod = $module->new;
|
my $mod = $module->new;
|
||||||
|
|
||||||
|
|
||||||
if (exists $mod->{name} and $mod->{name} eq $language) {
|
if (exists $mod->{name} and $mod->{name} eq $language) {
|
||||||
$language = $module;
|
$language = $module;
|
||||||
$found = 1;
|
$found = 1;
|
||||||
@ -53,12 +53,12 @@ if (not length $h->{code}) {
|
|||||||
if (exists $h->{usage}) {
|
if (exists $h->{usage}) {
|
||||||
print "$h->{usage}\n";
|
print "$h->{usage}\n";
|
||||||
} else {
|
} else {
|
||||||
print "Usage: cc [-paste] [-lang=<language>] [-info] [language options] <code> [-input=<stdin input>]\n";
|
print "Usage: cc [-lang=<language>] [-info] [-paste] [-args \"command-line arguments\"] [-stdin \"stdin input\"] [compiler/language options] <code>\n";
|
||||||
}
|
}
|
||||||
exit;
|
exit;
|
||||||
}
|
}
|
||||||
|
|
||||||
my $lang = $language->new(%$h);
|
my $lang = $language->new(%{$h});
|
||||||
|
|
||||||
$lang->{local} = $ENV{CC_LOCAL};
|
$lang->{local} = $ENV{CC_LOCAL};
|
||||||
|
|
||||||
|
@ -211,28 +211,6 @@ sub preprocess_code {
|
|||||||
$prelude .= "\n#include <prelude.h>\n";
|
$prelude .= "\n#include <prelude.h>\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (defined $self->{arguments}) {
|
|
||||||
my $qargs = quotemeta $self->{arguments};
|
|
||||||
$qargs =~ s/\\ / /g;
|
|
||||||
my @args = shellwords($self->{arguments});
|
|
||||||
$prelude .= "\nint arglen = " . (scalar @args) . ";\n";
|
|
||||||
|
|
||||||
if (@args) {
|
|
||||||
$prelude .= "char *args[] = { ";
|
|
||||||
|
|
||||||
my $comma = "";
|
|
||||||
foreach my $arg (@args) {
|
|
||||||
$arg =~ s/"/\\"/g;
|
|
||||||
$prelude .= "$comma\"$arg\"";
|
|
||||||
$comma = ", ";
|
|
||||||
}
|
|
||||||
|
|
||||||
$prelude .= " };\n";
|
|
||||||
} else {
|
|
||||||
$prelude .= "char *args[] = {0};\n";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
print "*** prelude: [$prelude]\n precode: [$precode]\n" if $self->{debug};
|
print "*** prelude: [$prelude]\n precode: [$precode]\n" if $self->{debug};
|
||||||
|
|
||||||
my $preprecode = $precode;
|
my $preprecode = $precode;
|
||||||
|
@ -14,6 +14,8 @@ use LWP::UserAgent;
|
|||||||
use Time::HiRes qw/gettimeofday/;
|
use Time::HiRes qw/gettimeofday/;
|
||||||
use Text::Balanced qw/extract_delimited/;
|
use Text::Balanced qw/extract_delimited/;
|
||||||
use JSON;
|
use JSON;
|
||||||
|
use Getopt::Long qw/GetOptionsFromArray :config pass_through/;
|
||||||
|
|
||||||
|
|
||||||
my $EXECUTE_PORT = '3333';
|
my $EXECUTE_PORT = '3333';
|
||||||
|
|
||||||
@ -40,11 +42,6 @@ sub new {
|
|||||||
$self->{lang} =~ s/^\s+|\s+$//g if defined $self->{lang};
|
$self->{lang} =~ s/^\s+|\s+$//g if defined $self->{lang};
|
||||||
$self->{code} =~ s/^\s+|\s+$//g if defined $self->{code};
|
$self->{code} =~ s/^\s+|\s+$//g if defined $self->{code};
|
||||||
|
|
||||||
if (defined $self->{arguments}) {
|
|
||||||
$self->{arguments} = quotemeta $self->{arguments};
|
|
||||||
$self->{arguments} =~ s/\\ / /g;
|
|
||||||
}
|
|
||||||
|
|
||||||
$self->initialize(%conf);
|
$self->initialize(%conf);
|
||||||
|
|
||||||
return $self;
|
return $self;
|
||||||
@ -70,10 +67,14 @@ sub preprocess_code {
|
|||||||
unless($self->{got_run} and $self->{copy_code}) {
|
unless($self->{got_run} and $self->{copy_code}) {
|
||||||
open FILE, ">> log.txt";
|
open FILE, ">> log.txt";
|
||||||
print FILE localtime() . "\n";
|
print FILE localtime() . "\n";
|
||||||
print FILE "$self->{nick} $self->{channel}: " . $self->{cmdline_options} . "$self->{code}\n";
|
print FILE "$self->{nick} $self->{channel}: [" . $self->{arguments} . "] " . $self->{cmdline_options} . "$self->{code}\n";
|
||||||
close FILE;
|
close FILE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (exists $self->{prelude}) {
|
||||||
|
$self->{code} = "$self->{prelude}\n$self->{code}";
|
||||||
|
}
|
||||||
|
|
||||||
# replace \n outside of quotes with literal newline
|
# replace \n outside of quotes with literal newline
|
||||||
my $new_code = "";
|
my $new_code = "";
|
||||||
|
|
||||||
@ -334,6 +335,11 @@ sub execute {
|
|||||||
$input =~ s/\s+$//;
|
$input =~ s/\s+$//;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$input =~ s/(?<!\\)\\n/\n/mg;
|
||||||
|
$input =~ s/(?<!\\)\\r/\r/mg;
|
||||||
|
$input =~ s/(?<!\\)\\t/\t/mg;
|
||||||
|
$input =~ s/(?<!\\)\\b/\b/mg;
|
||||||
|
|
||||||
my $pretty_code = $self->pretty_format($self->{code});
|
my $pretty_code = $self->pretty_format($self->{code});
|
||||||
|
|
||||||
my $cmdline = $self->{cmdline};
|
my $cmdline = $self->{cmdline};
|
||||||
@ -371,8 +377,16 @@ sub execute {
|
|||||||
print FILE localtime() . "\n";
|
print FILE localtime() . "\n";
|
||||||
print FILE "$cmdline\n$input\n$pretty_code\n";
|
print FILE "$cmdline\n$input\n$pretty_code\n";
|
||||||
|
|
||||||
my $compile_in = { lang => $self->{lang}, sourcefile => $self->{sourcefile}, execfile => $self->{execfile},
|
my $compile_in = {
|
||||||
cmdline => $cmdline, input => $input, date => $date, arguments => $self->{arguments}, code => $pretty_code };
|
lang => $self->{lang},
|
||||||
|
sourcefile => $self->{sourcefile},
|
||||||
|
execfile => $self->{execfile},
|
||||||
|
cmdline => $cmdline,
|
||||||
|
input => $input,
|
||||||
|
date => $date,
|
||||||
|
arguments => $self->{arguments},
|
||||||
|
code => $pretty_code
|
||||||
|
};
|
||||||
|
|
||||||
$compile_in->{'factoid'} = $self->{'factoid'} if length $self->{'factoid'};
|
$compile_in->{'factoid'} = $self->{'factoid'} if length $self->{'factoid'};
|
||||||
$compile_in->{'persist-key'} = $self->{'persist-key'} if length $self->{'persist-key'};
|
$compile_in->{'persist-key'} = $self->{'persist-key'} if length $self->{'persist-key'};
|
||||||
@ -384,7 +398,7 @@ sub execute {
|
|||||||
my $sent = 0;
|
my $sent = 0;
|
||||||
my $chunk_max = 4096;
|
my $chunk_max = 4096;
|
||||||
my $chunk_size = $length < $chunk_max ? $length : $chunk_max;
|
my $chunk_size = $length < $chunk_max ? $length : $chunk_max;
|
||||||
my $chunks_sent;
|
my $chunks_sent = 0;
|
||||||
|
|
||||||
#print FILE "Sending $length bytes [$compile_json] to vm_server\n";
|
#print FILE "Sending $length bytes [$compile_json] to vm_server\n";
|
||||||
|
|
||||||
@ -457,9 +471,30 @@ sub add_option {
|
|||||||
|
|
||||||
sub process_standard_options {
|
sub process_standard_options {
|
||||||
my $self = shift;
|
my $self = shift;
|
||||||
my $code = $self->{code};
|
|
||||||
|
|
||||||
if ($code =~ s/(?:^|(?<=\s))-info\s*//i) {
|
my @opt_args = $self->split_line($self->{code});
|
||||||
|
|
||||||
|
use Data::Dumper;
|
||||||
|
print STDERR Dumper \@opt_args;
|
||||||
|
|
||||||
|
my $getopt_error;
|
||||||
|
local $SIG{__WARN__} = sub {
|
||||||
|
$getopt_error = shift;
|
||||||
|
chomp $getopt_error;
|
||||||
|
};
|
||||||
|
|
||||||
|
my ($info, $input, $arguments, $paste);
|
||||||
|
my ($ret, $rest) = GetOptionsFromArray(\@opt_args,
|
||||||
|
'info!' => \$info,
|
||||||
|
'stdin|input=s' => \$input,
|
||||||
|
'args|arguments=s' => \$arguments,
|
||||||
|
'paste!' => \$paste);
|
||||||
|
|
||||||
|
print STDERR "getopt: ret: [$ret]: rest: [$rest], info: $info, input: $input, args: $arguments, paste: $paste\n";
|
||||||
|
|
||||||
|
print STDERR Dumper @opt_args;
|
||||||
|
|
||||||
|
if ($info) {
|
||||||
my $cmdline = $self->{cmdline};
|
my $cmdline = $self->{cmdline};
|
||||||
if (length $self->{default_options}) {
|
if (length $self->{default_options}) {
|
||||||
$cmdline =~ s/\$options/$self->{default_options}/;
|
$cmdline =~ s/\$options/$self->{default_options}/;
|
||||||
@ -473,15 +508,25 @@ sub process_standard_options {
|
|||||||
exit;
|
exit;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($code =~ s/-(?:input|stdin)=(.*)$//i) {
|
if (defined $input) {
|
||||||
$self->add_option("-input", $1);
|
if (not $input =~ s/^"(.*)"$/$1/) {
|
||||||
|
$input =~ s/^'(.*)'$/$1/;
|
||||||
|
}
|
||||||
|
$self->add_option("-input", $input);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($code =~ s/(?:^|(?<=\s))-paste\s*//i) {
|
if (defined $arguments) {
|
||||||
|
if (not $arguments =~ s/^"(.*)"$/$1/) {
|
||||||
|
$arguments =~ s/^'(.*)'$/$1/;
|
||||||
|
}
|
||||||
|
$self->{arguments} = $arguments;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($paste) {
|
||||||
$self->add_option("-paste");
|
$self->add_option("-paste");
|
||||||
}
|
}
|
||||||
|
|
||||||
$self->{code} = $code;
|
$self->{code} = join ' ', @opt_args;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub process_custom_options {
|
sub process_custom_options {
|
||||||
@ -1002,4 +1047,108 @@ sub process_interactive_edit {
|
|||||||
$self->{code} = $code;
|
$self->{code} = $code;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# splits line into quoted arguments while preserving quotes. handles
|
||||||
|
# unbalanced quotes gracefully by treating them as part of the argument
|
||||||
|
# they were found within.
|
||||||
|
sub split_line {
|
||||||
|
my ($self, $line, %opts) = @_;
|
||||||
|
|
||||||
|
my %default_opts = (
|
||||||
|
strip_quotes => 0,
|
||||||
|
keep_spaces => 0
|
||||||
|
);
|
||||||
|
|
||||||
|
%opts = (%default_opts, %opts);
|
||||||
|
|
||||||
|
my @chars = split //, $line;
|
||||||
|
|
||||||
|
my @args;
|
||||||
|
my $escaped = 0;
|
||||||
|
my $quote;
|
||||||
|
my $token = '';
|
||||||
|
my $ch = ' ';
|
||||||
|
my $last_ch;
|
||||||
|
my $i = 0;
|
||||||
|
my $pos;
|
||||||
|
my $ignore_quote = 0;
|
||||||
|
my $spaces = 0;
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
$last_ch = $ch;
|
||||||
|
|
||||||
|
if ($i >= @chars) {
|
||||||
|
if (defined $quote) {
|
||||||
|
# reached end, but unbalanced quote... reset to beginning of quote and ignore it
|
||||||
|
$i = $pos;
|
||||||
|
$ignore_quote = 1;
|
||||||
|
$quote = undef;
|
||||||
|
$last_ch = ' ';
|
||||||
|
$token = '';
|
||||||
|
} else {
|
||||||
|
# add final token and exit
|
||||||
|
push @args, $token if length $token;
|
||||||
|
last;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$ch = $chars[$i++];
|
||||||
|
|
||||||
|
$spaces = 0 if $ch ne ' ';
|
||||||
|
|
||||||
|
if ($escaped) {
|
||||||
|
$token .= "\\$ch";
|
||||||
|
$escaped = 0;
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($ch eq '\\') {
|
||||||
|
$escaped = 1;
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (defined $quote) {
|
||||||
|
if ($ch eq $quote) {
|
||||||
|
# closing quote
|
||||||
|
$token .= $ch unless $opts{strip_quotes};
|
||||||
|
push @args, $token;
|
||||||
|
$quote = undef;
|
||||||
|
$token = '';
|
||||||
|
} else {
|
||||||
|
# still within quoted argument
|
||||||
|
$token .= $ch;
|
||||||
|
}
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($last_ch eq ' ' and not defined $quote and ($ch eq "'" or $ch eq '"')) {
|
||||||
|
if ($ignore_quote) {
|
||||||
|
# treat unbalanced quote as part of this argument
|
||||||
|
$token .= $ch;
|
||||||
|
$ignore_quote = 0;
|
||||||
|
} else {
|
||||||
|
# begin potential quoted argument
|
||||||
|
$pos = $i - 1;
|
||||||
|
$quote = $ch;
|
||||||
|
$token .= $ch unless $opts{strip_quotes};
|
||||||
|
}
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($ch eq ' ') {
|
||||||
|
if (++$spaces > 1 and $opts{keep_spaces}) {
|
||||||
|
$token .= $ch;
|
||||||
|
next;
|
||||||
|
} else {
|
||||||
|
push @args, $token if length $token;
|
||||||
|
$token = '';
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$token .= $ch;
|
||||||
|
}
|
||||||
|
|
||||||
|
return @args;
|
||||||
|
}
|
||||||
|
|
||||||
1;
|
1;
|
||||||
|
@ -16,8 +16,6 @@ sub initialize {
|
|||||||
$self->{options_nopaste} = '-fno-diagnostics-show-caret';
|
$self->{options_nopaste} = '-fno-diagnostics-show-caret';
|
||||||
$self->{cmdline} = 'gcc -ggdb -g3 $sourcefile $options -o $execfile';
|
$self->{cmdline} = 'gcc -ggdb -g3 $sourcefile $options -o $execfile';
|
||||||
|
|
||||||
# $self->{default_options} .= ' -Werror' if defined $self->{nick} && $self->{nick} =~ m/marchelz/i;
|
|
||||||
|
|
||||||
$self->{prelude} = <<'END';
|
$self->{prelude} = <<'END';
|
||||||
#define _XOPEN_SOURCE 9001
|
#define _XOPEN_SOURCE 9001
|
||||||
#define __USE_XOPEN
|
#define __USE_XOPEN
|
||||||
|
@ -16,8 +16,6 @@ sub initialize {
|
|||||||
$self->{options_nopaste} = '-fno-diagnostics-show-caret';
|
$self->{options_nopaste} = '-fno-diagnostics-show-caret';
|
||||||
$self->{cmdline} = 'gcc -ggdb -g3 $sourcefile $options -o $execfile';
|
$self->{cmdline} = 'gcc -ggdb -g3 $sourcefile $options -o $execfile';
|
||||||
|
|
||||||
# $self->{default_options} .= ' -Werror' if defined $self->{nick} && $self->{nick} =~ m/marchelz/i;
|
|
||||||
|
|
||||||
$self->{prelude} = <<'END';
|
$self->{prelude} = <<'END';
|
||||||
#define _XOPEN_SOURCE 9001
|
#define _XOPEN_SOURCE 9001
|
||||||
#define __USE_XOPEN
|
#define __USE_XOPEN
|
||||||
|
@ -15,37 +15,6 @@ sub initialize {
|
|||||||
$self->{execfile} = 'prog.pl';
|
$self->{execfile} = 'prog.pl';
|
||||||
$self->{default_options} = '-w';
|
$self->{default_options} = '-w';
|
||||||
$self->{cmdline} = 'perl $options $sourcefile';
|
$self->{cmdline} = 'perl $options $sourcefile';
|
||||||
|
|
||||||
if (length $self->{arguments}) {
|
|
||||||
$self->{cmdline} .= " $self->{arguments}";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sub preprocess_code {
|
|
||||||
my $self = shift;
|
|
||||||
$self->SUPER::preprocess_code;
|
|
||||||
|
|
||||||
if (defined $self->{arguments}) {
|
|
||||||
my @args = shellwords($self->{arguments});
|
|
||||||
my $prelude .= "\nmy \$arglen = " . (scalar @args) . ";\n";
|
|
||||||
|
|
||||||
if (@args) {
|
|
||||||
$prelude .= "my \@args = (";
|
|
||||||
|
|
||||||
my $comma = "";
|
|
||||||
foreach my $arg (@args) {
|
|
||||||
$arg = quotemeta $arg;
|
|
||||||
$prelude .= "$comma\"$arg\"";
|
|
||||||
$comma = ", ";
|
|
||||||
}
|
|
||||||
|
|
||||||
$prelude .= ");\n";
|
|
||||||
} else {
|
|
||||||
$prelude .= "my \@args;\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
$self->{code} = "$prelude\n$self->{code}";
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sub postprocess_output {
|
sub postprocess_output {
|
||||||
|
@ -8,7 +8,55 @@ use parent '_default';
|
|||||||
|
|
||||||
sub preprocess {
|
sub preprocess {
|
||||||
my $self = shift;
|
my $self = shift;
|
||||||
$self->SUPER::preprocess;
|
|
||||||
|
my $input = $self->{input};
|
||||||
|
$input = "" if not defined $input;
|
||||||
|
|
||||||
|
print "writing input [$input]\n";
|
||||||
|
open(my $fh, '>', '.input');
|
||||||
|
print $fh "$input\n";
|
||||||
|
close $fh;
|
||||||
|
|
||||||
|
$self->execute(10, undef, 'date', '-s', "\@$self->{date}");
|
||||||
|
|
||||||
|
my @cmd = $self->split_line($self->{cmdline}, strip_quotes => 1);
|
||||||
|
|
||||||
|
if ($self->{code} =~ m/print_last_statement\(.*\);$/m) {
|
||||||
|
# remove print_last_statement wrapper in order to get warnings/errors from last statement line
|
||||||
|
my $code = $self->{code};
|
||||||
|
$code =~ s/print_last_statement\((.*)\);$/$1;/mg;
|
||||||
|
open(my $fh, '>', $self->{sourcefile}) or die $!;
|
||||||
|
print $fh $code . "\n";
|
||||||
|
close $fh;
|
||||||
|
|
||||||
|
print "Executing [$self->{cmdline}] without print_last_statement\n";
|
||||||
|
my ($retval, $stdout, $stderr) = $self->execute(60, undef, @cmd);
|
||||||
|
$self->{output} = $stderr;
|
||||||
|
$self->{output} .= ' ' if length $self->{output};
|
||||||
|
$self->{output} .= $stdout;
|
||||||
|
$self->{error} = $retval;
|
||||||
|
|
||||||
|
# now compile with print_last_statement intact, ignoring compile results
|
||||||
|
if (not $self->{error}) {
|
||||||
|
open(my $fh, '>', $self->{sourcefile}) or die $!;
|
||||||
|
print $fh $self->{code} . "\n";
|
||||||
|
close $fh;
|
||||||
|
|
||||||
|
print "Executing [$self->{cmdline}] with print_last_statement\n";
|
||||||
|
$self->execute(60, undef, @cmd);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
open(my $fh, '>', $self->{sourcefile}) or die $!;
|
||||||
|
print $fh $self->{code} . "\n";
|
||||||
|
close $fh;
|
||||||
|
|
||||||
|
print "Executing [$self->{cmdline}]\n";
|
||||||
|
my ($retval, $stdout, $stderr) = $self->execute(60, undef, @cmd);
|
||||||
|
$self->{output} = $stderr;
|
||||||
|
$self->{output} .= ' ' if length $self->{output};
|
||||||
|
$self->{output} .= $stdout;
|
||||||
|
$self->{error} = $retval;
|
||||||
|
}
|
||||||
|
|
||||||
if ($self->{cmdline} =~ m/--(?:version|analyze)/) {
|
if ($self->{cmdline} =~ m/--(?:version|analyze)/) {
|
||||||
$self->{done} = 1;
|
$self->{done} = 1;
|
||||||
@ -27,22 +75,17 @@ sub postprocess {
|
|||||||
}
|
}
|
||||||
|
|
||||||
print "Executing gdb\n";
|
print "Executing gdb\n";
|
||||||
my ($retval, $result) = $self->execute(60, "date -s \@$self->{date}; ulimit -t 5; compiler_watchdog.pl $self->{arguments} > .output");
|
my @args = $self->split_line($self->{arguments}, strip_quotes => 1);
|
||||||
|
my ($exitval, $stdout, $stderr) = $self->execute(60, undef, 'compiler_watchdog.pl', @args);
|
||||||
|
|
||||||
$result = "";
|
my $result = $stderr;
|
||||||
open(FILE, '.output');
|
$result .= ' ' if length $result;
|
||||||
while(<FILE>) {
|
$result .= $stdout;
|
||||||
$result .= $_;
|
|
||||||
last if length $result >= 1024 * 20;
|
|
||||||
}
|
|
||||||
close(FILE);
|
|
||||||
|
|
||||||
$result =~ s/\s+$//;
|
|
||||||
|
|
||||||
if (not length $result) {
|
if (not length $result) {
|
||||||
$self->{no_output} = 1;
|
$self->{no_output} = 1;
|
||||||
} elsif ($self->{code} =~ m/print_last_statement\(.*\);$/m
|
} elsif ($self->{code} =~ m/print_last_statement\(.*\);$/m
|
||||||
&& ($result =~ m/A syntax error in expression/ || $result =~ m/No symbol.*in current context/ || $result =~ m/has unknown return type; cast the call to its declared/ || $result =~ m/Can't take address of.*which isn't an lvalue/)) {
|
&& ($result =~ m/A syntax error in expression/ || $result =~ m/No symbol.*in current context/ || $result =~ m/has unknown return type; cast the call to its declared return/ || $result =~ m/Can't take address of.*which isn't an lvalue/)) {
|
||||||
# strip print_last_statement and rebuild/re-run
|
# strip print_last_statement and rebuild/re-run
|
||||||
$self->{code} =~ s/print_last_statement\((.*)\);/$1;/mg;
|
$self->{code} =~ s/print_last_statement\((.*)\);/$1;/mg;
|
||||||
$self->preprocess;
|
$self->preprocess;
|
||||||
|
@ -1,30 +1,32 @@
|
|||||||
#!/usr/bin/perl
|
#!/usr/bin/perl
|
||||||
|
|
||||||
|
package _default;
|
||||||
|
|
||||||
use warnings;
|
use warnings;
|
||||||
use strict;
|
use strict;
|
||||||
use feature "switch";
|
|
||||||
|
|
||||||
|
use feature "switch";
|
||||||
no if $] >= 5.018, warnings => "experimental::smartmatch";
|
no if $] >= 5.018, warnings => "experimental::smartmatch";
|
||||||
|
|
||||||
package _default;
|
use IPC::Run qw/run timeout/;
|
||||||
|
use Data::Dumper;
|
||||||
|
|
||||||
sub new {
|
sub new {
|
||||||
my ($class, %conf) = @_;
|
my ($class, %conf) = @_;
|
||||||
my $self = bless {}, $class;
|
my $self = bless {}, $class;
|
||||||
|
|
||||||
$self->{debug} = $conf{debug} // 0;
|
$self->{debug} = $conf{debug} // 0;
|
||||||
$self->{sourcefile} = $conf{sourcefile};
|
$self->{sourcefile} = $conf{sourcefile};
|
||||||
$self->{execfile} = $conf{execfile};
|
$self->{execfile} = $conf{execfile};
|
||||||
$self->{code} = $conf{code};
|
$self->{code} = $conf{code};
|
||||||
$self->{cmdline} = $conf{cmdline};
|
$self->{cmdline} = $conf{cmdline};
|
||||||
$self->{input} = $conf{input};
|
$self->{input} = $conf{input};
|
||||||
$self->{date} = $conf{date};
|
$self->{date} = $conf{date};
|
||||||
$self->{arguments} = $conf{arguments};
|
$self->{arguments} = $conf{arguments};
|
||||||
$self->{factoid} = $conf{factoid};
|
$self->{factoid} = $conf{factoid};
|
||||||
$self->{'persist-key'} = $conf{'persist-key'};
|
$self->{'persist-key'} = $conf{'persist-key'};
|
||||||
|
|
||||||
$self->initialize(%conf);
|
$self->initialize(%conf);
|
||||||
|
|
||||||
return $self;
|
return $self;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -35,52 +37,21 @@ sub initialize {
|
|||||||
sub preprocess {
|
sub preprocess {
|
||||||
my $self = shift;
|
my $self = shift;
|
||||||
|
|
||||||
my $input = $self->{input};
|
open(my $fh, '>', $self->{sourcefile}) or die $!;
|
||||||
$input = "" if not defined $input;
|
print $fh $self->{code} . "\n";
|
||||||
|
|
||||||
print "writing input [$input]\n";
|
|
||||||
|
|
||||||
$input =~ s/(?<!\\)\\n/\n/mg;
|
|
||||||
$input =~ s/(?<!\\)\\r/\r/mg;
|
|
||||||
$input =~ s/(?<!\\)\\t/\t/mg;
|
|
||||||
$input =~ s/(?<!\\)\\b/\b/mg;
|
|
||||||
|
|
||||||
open(my $fh, '>', '.input');
|
|
||||||
print $fh "$input\n";
|
|
||||||
close $fh;
|
close $fh;
|
||||||
|
|
||||||
if ($self->{code} =~ m/print_last_statement\(.*\);$/m) {
|
$self->execute(10, undef, 'date', '-s', "\@$self->{date}");
|
||||||
# remove print_last_statement wrapper in order to get warnings/errors from last statement line
|
|
||||||
my $code = $self->{code};
|
|
||||||
$code =~ s/print_last_statement\((.*)\);$/$1;/mg;
|
|
||||||
open(my $fh, '>', $self->{sourcefile}) or die $!;
|
|
||||||
print $fh $code . "\n";
|
|
||||||
close $fh;
|
|
||||||
|
|
||||||
print "Executing [$self->{cmdline}] without print_last_statement\n";
|
print "Executing [$self->{cmdline}] with args [$self->{arguments}]\n";
|
||||||
my ($retval, $result) = $self->execute(60, "date -s \@$self->{date} > /dev/null; ulimit -t 5; $self->{cmdline}");
|
my @cmdline = $self->split_line($self->{cmdline}, strip_quotes => 1);
|
||||||
$self->{output} = $result;
|
push @cmdline, $self->split_line($self->{arguments}, strip_quotes => 1);
|
||||||
$self->{error} = $retval;
|
|
||||||
|
|
||||||
# now compile with print_last_statement intact, ignoring compile results
|
my ($retval, $stdout, $stderr) = $self->execute(60, $self->{input}, @cmdline);
|
||||||
if (not $self->{error}) {
|
$self->{output} = $stderr;
|
||||||
open(my $fh, '>', $self->{sourcefile}) or die $!;
|
$self->{output} .= ' ' if length $self->{output};
|
||||||
print $fh $self->{code} . "\n";
|
$self->{output} .= $stdout;
|
||||||
close $fh;
|
$self->{error} = $retval;
|
||||||
|
|
||||||
print "Executing [$self->{cmdline}] with print_last_statement\n";
|
|
||||||
my ($retval, $result) = $self->execute(60, "date -s \@$self->{date} > /dev/null; ulimit -t 5; $self->{cmdline}");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
open(my $fh, '>', $self->{sourcefile}) or die $!;
|
|
||||||
print $fh $self->{code} . "\n";
|
|
||||||
close $fh;
|
|
||||||
|
|
||||||
print "Executing [$self->{cmdline}]\n";
|
|
||||||
my ($retval, $result) = $self->execute(60, "date -s \@$self->{date} > /dev/null; ulimit -t 5; $self->{cmdline} < .input");
|
|
||||||
$self->{output} = $result;
|
|
||||||
$self->{error} = $retval;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sub postprocess {
|
sub postprocess {
|
||||||
@ -88,42 +59,130 @@ sub postprocess {
|
|||||||
}
|
}
|
||||||
|
|
||||||
sub execute {
|
sub execute {
|
||||||
my $self = shift;
|
my ($self, $timeout, $stdin, @cmdline) = @_;
|
||||||
my $timeout = shift;
|
|
||||||
my ($cmdline) = @_;
|
|
||||||
|
|
||||||
my ($ret, $result);
|
$stdin //= '';
|
||||||
|
print "execute($timeout) [$stdin] ", Dumper \@cmdline, "\n";
|
||||||
|
|
||||||
($ret, $result) = eval {
|
my ($exitval, $stdout, $stderr) = eval {
|
||||||
print "eval\n";
|
my ($stdout, $stderr);
|
||||||
|
run \@cmdline, \$stdin, \$stdout, \$stderr, timeout($timeout);
|
||||||
my $result = '';
|
my $exitval = $? >> 8;
|
||||||
|
return ($exitval, $stdout, $stderr);
|
||||||
my $pid = open(my $fh, '-|', "$cmdline 2>&1");
|
|
||||||
|
|
||||||
local $SIG{ALRM} = sub { print "Time out\n"; kill 'TERM', $pid; die "$result [Timed-out]\n"; };
|
|
||||||
local $SIG{CHLD} = 'IGNORE';
|
|
||||||
alarm($timeout);
|
|
||||||
|
|
||||||
while(my $line = <$fh>) {
|
|
||||||
$result .= $line;
|
|
||||||
}
|
|
||||||
|
|
||||||
close $fh;
|
|
||||||
my $ret = $? >> 8;
|
|
||||||
alarm 0;
|
|
||||||
return ($ret, $result);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
print "done eval\n";
|
if ($@) {
|
||||||
alarm 0;
|
my $error = $@;
|
||||||
|
$error = "[Timed-out]" if $error =~ m/timeout on timer/;
|
||||||
if($@ =~ /Timed-out/) {
|
($exitval, $stdout, $stderr) = (-1, '', $error);
|
||||||
return (-1, $@);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
print "[$ret, $result]\n";
|
print "exitval $exitval stdout [$stdout]\nstderr [$stderr]\n";
|
||||||
return ($ret, $result);
|
return ($exitval, $stdout, $stderr);
|
||||||
|
}
|
||||||
|
|
||||||
|
# splits line into quoted arguments while preserving quotes. handles
|
||||||
|
# unbalanced quotes gracefully by treating them as part of the argument
|
||||||
|
# they were found within.
|
||||||
|
sub split_line {
|
||||||
|
my ($self, $line, %opts) = @_;
|
||||||
|
|
||||||
|
my %default_opts = (
|
||||||
|
strip_quotes => 0,
|
||||||
|
keep_spaces => 0
|
||||||
|
);
|
||||||
|
|
||||||
|
%opts = (%default_opts, %opts);
|
||||||
|
|
||||||
|
my @chars = split //, $line;
|
||||||
|
|
||||||
|
my @args;
|
||||||
|
my $escaped = 0;
|
||||||
|
my $quote;
|
||||||
|
my $token = '';
|
||||||
|
my $ch = ' ';
|
||||||
|
my $last_ch;
|
||||||
|
my $i = 0;
|
||||||
|
my $pos;
|
||||||
|
my $ignore_quote = 0;
|
||||||
|
my $spaces = 0;
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
$last_ch = $ch;
|
||||||
|
|
||||||
|
if ($i >= @chars) {
|
||||||
|
if (defined $quote) {
|
||||||
|
# reached end, but unbalanced quote... reset to beginning of quote and ignore it
|
||||||
|
$i = $pos;
|
||||||
|
$ignore_quote = 1;
|
||||||
|
$quote = undef;
|
||||||
|
$last_ch = ' ';
|
||||||
|
$token = '';
|
||||||
|
} else {
|
||||||
|
# add final token and exit
|
||||||
|
push @args, $token if length $token;
|
||||||
|
last;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$ch = $chars[$i++];
|
||||||
|
|
||||||
|
$spaces = 0 if $ch ne ' ';
|
||||||
|
|
||||||
|
if ($escaped) {
|
||||||
|
$token .= "\\$ch";
|
||||||
|
$escaped = 0;
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($ch eq '\\') {
|
||||||
|
$escaped = 1;
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (defined $quote) {
|
||||||
|
if ($ch eq $quote) {
|
||||||
|
# closing quote
|
||||||
|
$token .= $ch unless $opts{strip_quotes};
|
||||||
|
push @args, $token;
|
||||||
|
$quote = undef;
|
||||||
|
$token = '';
|
||||||
|
} else {
|
||||||
|
# still within quoted argument
|
||||||
|
$token .= $ch;
|
||||||
|
}
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($last_ch eq ' ' and not defined $quote and ($ch eq "'" or $ch eq '"')) {
|
||||||
|
if ($ignore_quote) {
|
||||||
|
# treat unbalanced quote as part of this argument
|
||||||
|
$token .= $ch;
|
||||||
|
$ignore_quote = 0;
|
||||||
|
} else {
|
||||||
|
# begin potential quoted argument
|
||||||
|
$pos = $i - 1;
|
||||||
|
$quote = $ch;
|
||||||
|
$token .= $ch unless $opts{strip_quotes};
|
||||||
|
}
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($ch eq ' ') {
|
||||||
|
if (++$spaces > 1 and $opts{keep_spaces}) {
|
||||||
|
$token .= $ch;
|
||||||
|
next;
|
||||||
|
} else {
|
||||||
|
push @args, $token if length $token;
|
||||||
|
$token = '';
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$token .= $ch;
|
||||||
|
}
|
||||||
|
|
||||||
|
return @args;
|
||||||
}
|
}
|
||||||
|
|
||||||
1;
|
1;
|
||||||
|
Loading…
Reference in New Issue
Block a user