X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~ian/git?a=blobdiff_plain;f=normalise;h=0af8f19623d021aa3d9c34f912fc3be0584fc908;hb=49ea4a7c89f1a34aff1714a3dcbdc3a52f5990e6;hp=df9e46763b3918bab064a9002538478dfc3f02f5;hpb=622789750886b5b2e39f2fadf7efe3c962ead67e;p=appendix-a6.git diff --git a/normalise b/normalise index df9e467..0af8f19 100755 --- a/normalise +++ b/normalise @@ -1,24 +1,49 @@ #!/usr/bin/perl -w +# +# usage: +# normalise [NORM-OPTIONS...] [--] INPUT-FILES... +# +# NORM-OPTIONS are +# +OPTNAME[=OPTVAL] Election option +# CAND=[DESCRIPTION] +# +CAND+OPTNAME[=OPTVAL] Candidate option +# -- End of options to normalise +# -... Reserved for future options to normalise use strict; our @options; our %candidates; # $candidates{CAND}{Desc}, {Opts}[] +our %candref; # $candref{CAND} => msg of why needed our @ballots; my $candvoter_re = '\w+'; +my $cands_re = '\w+(?,\w+)*'; +my $opt_re = '\w+(?:=\S*)?'; sub badinput ($) { die "bad input: $_[0]"; } -sub normalise_opts_list ($) { - my ($os) = @_; +sub normalise_opts_list ($$) { + # $ctx is one of Election Candidate Ballot + my ($os,$ctx) = @_; $os //= ''; my @o; foreach my $o (split /\s+/, $os) { if ($o =~ m/^\w+$/) { push @o, $&; + } elsif ($o =~ s/^_[Tt]ie\=//) { + $o =~ m/^($cands_re)([<>])($cands_re)$/ + or badinput "bad value \`$_' for tie option"; + my ($l,$op,$r) = ($1,$2,$3); + ($l,$op,$r) = ($r,'>',$l) if $op eq '<'; + $candref{$_} = "tie break spec" foreach $o =~ m/\w+/g; + $l = join ',', sort split /\,/, $l; + $r = join ',', sort split /\,/, $r; + $l =~ m/\b$_\b/ and badinput "reflexive tie" + foreach split /\,/, $r; + push @o, "$l$op$r"; } elsif ($o =~ m/^\w+\=\S+$/) { push @o, $&; } elsif ($o !~ m/\S/) { @@ -29,9 +54,9 @@ sub normalise_opts_list ($) { return @o; } -sub normalise_opts ($) { - my ($os) = @_; - my @o = normalise_opts_list $os; +sub normalise_opts ($$) { + my ($os,$ctx) = @_; + my @o = normalise_opts_list $os, $ctx; return " | @o"; } @@ -45,18 +70,37 @@ sub setcanddesc ($$) { } } +while (@ARGV) { + $_ = shift @ARGV; + if (m/^--$/) { + last; + } elsif (m/^(\w+)=([^|]+)$/) { + setcanddesc $1, $2; + } elsif (m/^\+($opt_re)$/) { + push @options, $1; + } elsif (m/^\+(\w+)\+($opt_re)$/) { + push @{ $candidates{$1}{Opts} }, $2; + } elsif (m/^-/) { + die "unknown normalise option \`$_'\n"; + } else { + # oh! + unshift @ARGV, $_; + last; + } +} + while (<>) { next unless m/\S/; next if m/^\#/; s/^\s+//; s/\s+$//; if (m/^\|/) { - push @options, normalise_opts_list $'; + push @options, normalise_opts_list $', 'Election'; } elsif (m/^($candvoter_re?)\s*=\s*([^|]+?)\s*\|(.*)?$/o) { use Data::Dumper; print STDERR Dumper($1,$2,$3); my ($cand,$desc,$opts) = ($1,$2,$3); - push @{ $candidates{$cand}{Opts} }, normalise_opts $opts; + push @{ $candidates{$cand}{Opts} }, normalise_opts $opts, 'Candidate'; setcanddesc $cand, $desc; } elsif (m/^($candvoter_re?)?\s*\:([^|]*)(?:\|(.*))?$/) { my ($voter,$opts) = ($1,$3); @@ -71,7 +115,7 @@ print STDERR Dumper($1,$2,$3); badinput "bad vote preference \`$p'"; } } - push @ballots, "$voter : @p".normalise_opts $opts; + push @ballots, "$voter : @p".normalise_opts $opts, 'Ballot'; } elsif (m/^\.$/) { } else { badinput "unknown line format \`$_'"; @@ -80,6 +124,11 @@ print STDERR Dumper($1,$2,$3); print "| @options\n" or die $!; +foreach my $cand (sort keys %candref) { + badinput "missing candidate $cand (ref. by $candref{$cand}" + unless defined $candidates{$cand}; +} + foreach my $cand (sort keys %candidates) { my $c = $candidates{$cand}; $c->{Desc} //= $cand;