pbot/modules/ideone.pl

201 lines
9.7 KiB
Perl
Raw Normal View History

#!/usr/bin/perl
# use warnings;
use strict;
use SOAP::Lite;
my $user = 'test';
my $pass = 'test';
my $soap = SOAP::Lite->new(proxy => 'http://ideone.com/api/1/service');
my $result;
my %languages = (
'Ada' => { 'id' => '7', 'name' => 'Ada (gnat-4.3.2)' },
'Assembler' => { 'id' => '13', 'name' => 'Assembler (nasm-2.07)' },
'gawk' => { 'id' => '104', 'name' => 'AWK (gawk) (gawk-3.1.6)' },
'mawk' => { 'id' => '105', 'name' => 'AWK (mawk) (mawk-1.3.3)' },
'Bash' => { 'id' => '28', 'name' => 'Bash (bash 4.0.35)' },
'bc' => { 'id' => '110', 'name' => 'bc (bc-1.06.95)' },
'Brainfuck' => { 'id' => '12', 'name' => 'Brainf**k (bff-1.0.3.1)' },
'bf' => { 'id' => '12', 'name' => 'Brainf**k (bff-1.0.3.1)' },
'C' => { 'id' => '11', 'name' => 'C (gcc-4.3.4)' },
'C#' => { 'id' => '27', 'name' => 'C# (gmcs 2.0.1)' },
'C++' => { 'id' => '1', 'name' => 'C++ (gcc-4.3.4)' },
'C99' => { 'id' => '34', 'name' => 'C99 strict (gcc-4.3.4)' },
'CLIPS' => { 'id' => '14', 'name' => 'CLIPS (clips 6.24)' },
'Clojure' => { 'id' => '111', 'name' => 'Clojure (clojure 1.1.0)' },
'COBOL' => { 'id' => '118', 'name' => 'COBOL (open-cobol-1.0)' },
'COBOL85' => { 'id' => '106', 'name' => 'COBOL 85 (tinycobol-0.65.9)' },
'clisp' => { 'id' => '32', 'name' => 'Common Lisp (clisp) (clisp 2.47)' },
'D' => { 'id' => '102', 'name' => 'D (dmd) (dmd-2.042)' },
'Erlang' => { 'id' => '36', 'name' => 'Erlang (erl-5.7.3)' },
'Forth' => { 'id' => '107', 'name' => 'Forth (gforth-0.7.0)' },
'Fortran' => { 'id' => '5', 'name' => 'Fortran (gfortran-4.3.4)' },
'Go' => { 'id' => '114', 'name' => 'Go (gc 2010-01-13)' },
'Haskell' => { 'id' => '21', 'name' => 'Haskell (ghc-6.8.2)' },
'Icon' => { 'id' => '16', 'name' => 'Icon (iconc 9.4.3)' },
'Intercal' => { 'id' => '9', 'name' => 'Intercal (c-intercal 28.0-r1)' },
'Java' => { 'id' => '10', 'name' => 'Java (sun-jdk-1.6.0.17)' },
'JS' => { 'id' => '35', 'name' => 'JavaScript (rhino) (rhino-1.6.5)' },
'JScript' => { 'id' => '35', 'name' => 'JavaScript (rhino) (rhino-1.6.5)' },
'JavaScript' => { 'id' => '35', 'name' => 'JavaScript (rhino) (rhino-1.6.5)' },
'JavaScript-rhino' => { 'id' => '35', 'name' => 'JavaScript (rhino) (rhino-1.6.5)' },
'JavaScript-spidermonkey' => { 'id' => '112', 'name' => 'JavaScript (spidermonkey) (spidermonkey-1.7)' },
'Lua' => { 'id' => '26', 'name' => 'Lua (luac 5.1.4)' },
'Nemerle' => { 'id' => '30', 'name' => 'Nemerle (ncc 0.9.3)' },
'Nice' => { 'id' => '25', 'name' => 'Nice (nicec 0.9.6)' },
'Ocaml' => { 'id' => '8', 'name' => 'Ocaml (ocamlopt 3.10.2)' },
'Pascal' => { 'id' => '22', 'name' => 'Pascal (fpc) (fpc 2.2.0)' },
'Pascal-fpc' => { 'id' => '22', 'name' => 'Pascal (fpc) (fpc 2.2.0)' },
'Pascal-gpc' => { 'id' => '2', 'name' => 'Pascal (gpc) (gpc 20070904)' },
'Perl' => { 'id' => '3', 'name' => 'Perl (perl 5.8.8)' },
'PHP' => { 'id' => '29', 'name' => 'PHP (php 5.2.11)' },
'Pike' => { 'id' => '19', 'name' => 'Pike (pike 7.6.86)' },
'Prolog' => { 'id' => '108', 'name' => 'Prolog (gnu) (gprolog-1.3.1)' },
'Prolog-gnu' => { 'id' => '108', 'name' => 'Prolog (gnu) (gprolog-1.3.1)' },
'Prolog-swi' => { 'id' => '15', 'name' => 'Prolog (swi) (swipl 5.6.64)' },
'Python' => { 'id' => '4', 'name' => 'Python (python 2.6.4)' },
'Python3' => { 'id' => '116', 'name' => 'Python3 (python-3.1.1)' },
'R' => { 'id' => '117', 'name' => 'R (R-2.9.2)' },
'Ruby' => { 'id' => '17', 'name' => 'Ruby (ruby 1.8.7)' },
'Scala' => { 'id' => '39', 'name' => 'Scala (Scalac 2.7.7)' },
'Scheme' => { 'id' => '33', 'name' => 'Scheme (guile) (guile 1.8.5)' },
'Smalltalk' => { 'id' => '23', 'name' => 'Smalltalk (gst 3.1)' },
'Tcl' => { 'id' => '38', 'name' => 'Tcl (tclsh 8.5.7)' },
'Unlambda' => { 'id' => '115', 'name' => 'Unlambda (unlambda-2.0.0)' },
'VB' => { 'id' => '101', 'name' => 'Visual Basic .NET (mono-2.4.2.3)' },
);
my %preludes = (
'C99' => "#include <stdio.h>\n#include <stdlib.h>\n#include <string.h>\n",
'C' => "#include <stdio.h>\n#include <stdlib.h>\n#include <string.h>\n",
'C++' => "#include <iostream>\n#include <cstdio>\n",
);
if($#ARGV <= 0) {
print "Usage: cc [-lang=<language>] <code>\n";
exit 0;
}
my $nick = shift @ARGV;
my $code = join ' ', @ARGV;
open FILE, ">> ideone_log.txt";
print FILE "$nick: $code\n";
my $lang = "C99";
$lang = $1 if $code =~ s/-lang=([^\b\s]+)//i;
my $found = 0;
my @langs;
foreach my $l (sort { uc $a cmp uc $b } keys %languages) {
push @langs, sprintf(" %-30s => %s", $l, $languages{$l}{'name'});
if(uc $lang eq uc $l) {
$lang = $l;
$found = 1;
}
}
if(not $found) {
print "$nick: Invalid language '$lang'. Supported languages are:\n", (join ",\n", @langs), "\n";
exit 0;
}
my $input = "";
$input = $1 if $code =~ s/-input=(.*)$//i;
$code =~ s/#include <([^>]+)>/\n#include <$1>\n/g;
$code =~ s/#([^ ]+) (.*?)\\n/\n#$1 $2\n/g;
$code = $preludes{$lang} . $code;
if(($lang eq "C99" or $lang eq "C" or $lang eq "C++") and not $code =~ m/(int|void) main\s*\([^)]*\)\s*{/) {
my $prelude = '';
$prelude = "$1$2" if $code =~ s/^\s*(#.*)(#.*?[>\n])//s;
$code = "$prelude\n int main(int argc, char **argv) { $code ; return 0; }";
}
$result = get_result($soap->createSubmission($user, $pass, $code, $languages{$lang}{'id'}, $input, 1, 1));
my $url = $result->{link};
# wait for compilation/execution to complete
while(1) {
sleep 2;
$result = get_result($soap->getSubmissionStatus($user, $pass, $url));
last if $result->{status} == 0;
}
$result = get_result($soap->getSubmissionDetails($user, $pass, $url, 0, 0, 1, 1, 1));
my $output = "";
my $COMPILER_ERROR = 11;
my $RUNTIME_ERROR = 12;
my $TIMELIMIT = 13;
my $MEMORYLIMIT = 17;
my $ILLEGAL_SYSCALL = 19;
my $INTERNAL_ERROR = 20;
if($result->{result} == $COMPILER_ERROR) {
$output .= $result->{cmpinfo};
$output =~ s/[\n\r]/ /g;
}
if($result->{signal}) {
$output .= "\n[Signal: $result->{signal}]";
}
if($result->{result} == $RUNTIME_ERROR) {
$output .= "\n[Runtime error]";
}
if($result->{result} == $TIMELIMIT) {
$output .= "\n[Time limit exceeded]";
}
if($result->{result} == $MEMORYLIMIT) {
$output .= "\n[Out of memory]";
}
if($result->{result} == $ILLEGAL_SYSCALL) {
$output .= "\n[Disallowed system call]";
}
if($result->{result} == $INTERNAL_ERROR) {
$output .= "\n[Internal error]";
}
$output .= "\n" . $result->{stderr};
$output .= "\n" . $result->{output};
$output =~ s/cc1: warnings being treated as errors//;
$output =~ s/ Line \d+ ://g;
$output =~ s/ \(first use in this function\)//g;
$output =~ s/error: \(Each undeclared identifier is reported only once.*?\)//msg;
$output =~ s/prog\.c[:\d\s]*//g;
$output =~ s/error: (.*?) error/error: $1; error/msg;
print FILE "$nick: [ http://ideone.com/$url ] $output\n\n";
close FILE;
print "$nick: $output\n";
# ---------------------------------------------
sub get_result {
my $result = shift @_;
if($result->fault) {
print join ', ', $result->faultcode, $result->faultstring, $result->faultdetail;
exit 0;
} else {
if($result->result->{error} ne "OK") {
print $result->result->{error};
exit 0;
} else {
return $result->result;
}
}
}