]> git.sesse.net Git - remoteglot-book/blobdiff - www/opening-stats.pl
Indent fix.
[remoteglot-book] / www / opening-stats.pl
index f9e85484eb0ca749df78fbd604b9a10a7f748319..58a0f9be02a9ed230808fb00ce281abeef3531e9 100755 (executable)
@@ -6,6 +6,7 @@ use JSON::XS;
 use lib '..';
 use Position;
 use IPC::Open2;
+use Chess::PGN::Parse;
 
 our %openings = ();
 read_openings();
@@ -16,40 +17,97 @@ my $pid = IPC::Open2::open2($chld_out, $chld_in, "../binlookup", "../open.mtbl",
 
 # Root position. Basically ignore everything except the opening (and later some root game stuff).
 my $fen = $cgi->param('fen') // 'rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 0 1';
+my $includetransp = $cgi->param('includetransp') // 1;
+
 my $pos = Position->from_fen($fen);
-my $hex = unpack('H*', $pos->bitpacked_fen);
-print $chld_in $hex, "\n";
-chomp (my $line = <$chld_out>);
+my ($json_root_pos, $root_aux_data) = get_json_move($pos, undef, $chld_in, $chld_out);
+
+my $opening = $openings{$json_root_pos->{'opening_num'}} // 'A00: Start position';
+my @json_moves = ($json_root_pos);
+
+my $root_game;
+eval {
+       if (!exists($root_aux_data->{'pgn_file_number'}) ||
+           !exists($root_aux_data->{'pgn_start_position'})) {
+               die "Missing PGN position data."
+       }
+       my $pgntext = read_root_pgn($root_aux_data->{'pgn_file_number'}, $root_aux_data->{'pgn_start_position'});
+       my $pgn = Chess::PGN::Parse->new(undef, $pgntext);
+       $pgn->read_game() or die;
+       $pgn->parse_game() or die;
 
-my ($white, $draw, $black, $opening_num, $white_avg_elo, $black_avg_elo, $num_elo, $timestamp, @moves) = split / /, $line;
-my $opening = $openings{$opening_num} // 'A00: Start position';
+       my $tags = $pgn->tags;
+       $root_game = {};
+       $root_game->{'white'} = $pgn->white;
+       $root_game->{'white_elo'} = $tags->{'WhiteElo'};
+       $root_game->{'black'} = $pgn->black;
+       $root_game->{'black_elo'} = $tags->{'BlackElo'};
+       $root_game->{'event'} = $pgn->event;
+       $root_game->{'date'} = $pgn->date;
+       $root_game->{'result'} = $pgn->result;
+       $root_game->{'eco'} = $pgn->eco;
+       $root_game->{'moves'} = int(((scalar @{$pgn->moves}) + 1) / 2);
+};
+if ($@) {
+       print STDERR "Error while getting root move: $@\n";
+}
 
 # Explore one move out.
-my @json_moves = ();
-for my $move (@moves) {
+my $white_left = $json_root_pos->{'white'};
+my $draw_left = $json_root_pos->{'draw'};
+my $black_left = $json_root_pos->{'black'};
+for my $move (@{$root_aux_data->{'moves'}}) {
        my ($np, $uci_move) = $pos->make_pretty_move($move);
-       my $hex = unpack('H*', $np->bitpacked_fen);
-       print $chld_in $hex, "\n";
-       my $line = <$chld_out>;
-       my ($white, $draw, $black, $opening_num, $white_avg_elo, $black_avg_elo, $num_elo) = split / /, $line;
-       push @json_moves, {
-               move => $move,
-               white => $white * 1,
-               draw => $draw * 1,
-               black => $black * 1,
-               white_avg_elo => $white_avg_elo * 1,
-               black_avg_elo => $black_avg_elo * 1,
-               num_elo => $num_elo * 1
+       my $json_pos;
+
+       my ($json_pos_only_this_root, undef) = get_json_move($np, $root_aux_data->{'pos_hash'}, $chld_in, $chld_out);
+       $white_left -= $json_pos_only_this_root->{'white'};
+       $draw_left -= $json_pos_only_this_root->{'draw'};
+       $black_left -= $json_pos_only_this_root->{'black'};
+
+       if ($includetransp) {
+               ($json_pos, undef) = get_json_move($np, undef, $chld_in, $chld_out);
+
+               # See if this move exists only due to transpositions.
+               if (!defined($json_pos_only_this_root)) {
+                       $json_pos->{'transpose_only'} = 1;
+               }
+       } else {
+               $json_pos = $json_pos_only_this_root;
+       }
+       $json_pos->{'move'} = $move;
+       push @json_moves, $json_pos;
+}
+
+# If there are any positions that are not accounted for by any moves,
+# these have to be games that end here. Add them as pseudo-moves so as
+# not to confuse the user.
+for my $result (['1-0', 'white', $white_left], ['1/2-1/2', 'draw', $draw_left], ['0-1', 'black', $black_left]) {
+       next if ($result->[2] == 0);
+       my $move = {
+               move => $result->[0],
+               white => 0,
+               draw => 0,
+               black => 0
        };
+       $move->{$result->[1]} = $result->[2];
+       push @json_moves, $move;
 }
 
-print $cgi->header(-type=>'application/json');
-print JSON::XS::encode_json({ moves => \@json_moves, opening => $opening });
+# Get stats for the root position, for the human index.
+my $start_pos = Position->start_pos("white", "black");
+my ($json_start_pos, undef) = get_json_move($start_pos, 0, $chld_in, $chld_out);
+my $total_games = $json_start_pos->{'white'} + $json_start_pos->{'draw'} + $json_start_pos->{'black'};
+my $computer_games = $json_start_pos->{'computer'} * 1;
 
-sub num {
-       my $x = shift;
-       return $x->{'white'} + $x->{'draw'} + $x->{'black'};
-}
+print $cgi->header(-type=>'application/json');
+print JSON::XS::encode_json({
+       moves => \@json_moves,
+       opening => $opening,
+       root_game => $root_game,
+       total_games => $total_games,
+       computer_games => $computer_games
+});
 
 sub read_openings {
        open my $fh, "../openings.txt"
@@ -65,3 +123,71 @@ sub read_openings {
        }
        close $fh;
 }
+
+sub read_root_pgn {
+       my ($pgn_file_number, $pgn_start_position) = @_;
+       my @pgnnames;
+       open my $pgnnamesfh, "<", "../pgnnames.txt"
+               or die "../pgnnames.txt: $!";
+       while (<$pgnnamesfh>) {
+               chomp;
+               s/^comp://;
+               push @pgnnames, $_;
+       }
+       close $pgnnamesfh;
+
+       if ($pgn_file_number > $#pgnnames) {
+               die "Unknown PGN file number $pgn_file_number";
+       }
+
+       my $root_pgn;
+       open my $pgnfh, "<", "../" . $pgnnames[$pgn_file_number]
+               or die $pgnnames[$pgn_file_number] . ": $!";
+       sysseek($pgnfh, $pgn_start_position, 0)
+               or die "Could not seek to $pgn_start_position: $!";
+       sysread($pgnfh, $root_pgn, 32768)
+               or die "Could not read PGN from $pgn_start_position at $pgnnames[$pgn_file_number]: $!";
+       close $pgnfh;
+       $root_pgn =~ s/^.*?(\[Event )/$1/s;
+       $root_pgn =~ s/^(.+?)\[Event .*/$1/s;
+
+       return $root_pgn;
+}
+
+sub get_json_move {
+       my ($pos, $filter_prev_pos_hash, $chld_in, $chld_out) = @_;
+       my $bpfen_hex = unpack('H*', $pos->bitpacked_fen);
+       my $prev_pos_hash_hex = '';
+       if (defined($filter_prev_pos_hash)) {
+               $prev_pos_hash_hex .= unpack('H*', pack('S', $filter_prev_pos_hash));
+       }
+       print $chld_in $bpfen_hex, "\n", $prev_pos_hash_hex, "\n";
+
+       # Read the hash of this position.
+       chomp (my $pos_hash = <$chld_out>);
+
+       chomp (my $line = <$chld_out>);
+       if ($line eq '-') {
+               warn "Missing pos '" . $pos->fen . "' " . $filter_prev_pos_hash;
+               return (undef, undef);
+       }
+
+       my ($white, $draw, $black, $computer, $opening_num, $white_sum_elo, $black_sum_elo, $num_elo, $timestamp, $pgn_file_number, $pgn_start_position, @moves) = split / /, $line;
+       my $json_pos = {
+               white => int($white),
+               draw => int($draw),
+               black => int($black),
+               computer => int($computer),
+               white_avg_elo => $num_elo == 0 ? undef : $white_sum_elo / $num_elo,
+               black_avg_elo => $num_elo == 0 ? undef : $black_sum_elo / $num_elo,
+               num_elo => int($num_elo),
+               opening_num => $opening_num,  # Keep as string.
+       };
+       my $aux_data = {  # Only relevant for the root.
+               pos_hash => $pos_hash * 1,
+               moves => \@moves,
+               pgn_file_number => $pgn_file_number,
+               pgn_start_position => $pgn_start_position,
+       };
+       return ($json_pos, $aux_data);
+}