2023-08-26 14:16:48 +00:00
|
|
|
use v5.32;
|
2023-08-26 14:20:14 +00:00
|
|
|
|
2023-08-26 14:16:48 +00:00
|
|
|
use HTML::TableExtract;
|
|
|
|
use File::Slurp;
|
|
|
|
use Data::Dumper;
|
|
|
|
use Text::SimpleTable::AutoWidth;
|
2023-08-26 14:20:14 +00:00
|
|
|
use WWW::Mechanize ();
|
2023-08-26 15:22:19 +00:00
|
|
|
use List::AllUtils qw/ sum pairmap /;
|
2023-08-26 16:36:25 +00:00
|
|
|
use List::UtilsBy qw/ nsort_by partition_by /;
|
2023-08-26 14:16:48 +00:00
|
|
|
|
2023-08-26 14:28:29 +00:00
|
|
|
use experimental qw/ signatures /;
|
|
|
|
|
2023-08-26 16:36:25 +00:00
|
|
|
unless (caller) {
|
2023-08-26 14:29:15 +00:00
|
|
|
my $ts = get_mersenne_results();
|
|
|
|
|
|
|
|
generate_output_table($ts)->draw;
|
|
|
|
}
|
|
|
|
|
2023-08-26 14:24:39 +00:00
|
|
|
sub get_mersenne_results() {
|
2023-08-26 14:28:29 +00:00
|
|
|
|
2023-08-26 14:24:39 +00:00
|
|
|
# log in to Mersenne.org and get results for the last year
|
|
|
|
# excluding everything but PHP and DD results
|
|
|
|
|
|
|
|
my $mech = WWW::Mechanize->new;
|
|
|
|
|
|
|
|
my $url =
|
2023-08-26 14:28:29 +00:00
|
|
|
'https://www.mersenne.org/results/?extf=1&exp1=1&execm=1&excert=1&exp_lo=2&exp_hi=&limit=10000';
|
2023-08-26 14:24:39 +00:00
|
|
|
|
|
|
|
$mech->get($url);
|
|
|
|
|
|
|
|
$mech->submit_form(
|
|
|
|
form_number => 1,
|
|
|
|
fields => {
|
|
|
|
user_login => $ENV{'MERSENNE_USER'},
|
|
|
|
user_password => $ENV{'MERSENNE_PASSWORD'},
|
|
|
|
} );
|
|
|
|
|
|
|
|
# load the results into a table object
|
|
|
|
my $html_string = $mech->content;
|
|
|
|
|
2023-08-26 16:50:17 +00:00
|
|
|
return extract_first_table($html_string);
|
|
|
|
}
|
|
|
|
|
2023-08-26 16:36:25 +00:00
|
|
|
sub extract_first_table ($html) {
|
2023-08-26 16:50:17 +00:00
|
|
|
$html =~ s/\n//g;
|
2023-08-26 14:24:39 +00:00
|
|
|
|
|
|
|
my $te = HTML::TableExtract->new( depth => 0, count => 2 );
|
|
|
|
|
2023-08-26 16:50:17 +00:00
|
|
|
$te->parse($html);
|
2023-08-26 14:24:39 +00:00
|
|
|
|
|
|
|
return $te->first_table_found;
|
2023-08-26 16:50:17 +00:00
|
|
|
|
2023-08-26 14:24:39 +00:00
|
|
|
}
|
|
|
|
|
2023-08-26 14:28:29 +00:00
|
|
|
sub generate_output_table ($ts) {
|
|
|
|
|
2023-08-26 16:50:48 +00:00
|
|
|
my $tbl = Text::SimpleTable::AutoWidth->new(
|
2023-08-26 14:28:29 +00:00
|
|
|
captions => [qw/ Computer GHZDaysPerDay /] );
|
2023-08-26 16:50:48 +00:00
|
|
|
|
2023-08-26 16:36:25 +00:00
|
|
|
# group GHZ Days results by computer, compute GHZ Days per Day (GHZ Days / Days)
|
|
|
|
$tbl->row(@$_) for
|
2023-08-26 15:22:19 +00:00
|
|
|
reverse
|
2023-08-26 16:36:25 +00:00
|
|
|
nsort_by { $_->[1] }
|
|
|
|
pairmap { [ $a, $b ] } # group for the sorting
|
|
|
|
pairmap { $a => sprintf "%.0f", sum(@$b) / @$b } # take the mean
|
|
|
|
pairmap {
|
|
|
|
# calculate our stats
|
|
|
|
$a => [ map { $_->{ghz_days} / $_->{days} } @$b ]
|
|
|
|
}
|
|
|
|
partition_by { $_->{machine} } # group by machine
|
|
|
|
grep { $_->{days} > 0 } # skip entries with no days
|
|
|
|
map { # get the info
|
|
|
|
my $machine = $_->[0];
|
|
|
|
my $days = $_->[4] =~ s/\s//gr;
|
|
|
|
my $ghz_days = $_->[6] =~ s/\s//gr;
|
|
|
|
|
|
|
|
+{ days => $days, ghz_days => $ghz_days, machine => $machine }
|
|
|
|
}
|
|
|
|
$ts->rows;
|
2023-08-26 16:50:48 +00:00
|
|
|
|
|
|
|
return $tbl;
|
2023-08-26 14:16:48 +00:00
|
|
|
}
|