GitXplorerGitXplorer
l

starvote

public
11 stars
2 forks
1 issues

Commits

List of commits on branch master.
Verified
c277469ec218ff06041394c33a341f059f6c6728

Version 2.1.6! New custom serializer. (#11)

llarryhastings committed a month ago
Unverified
19c46dceaa5b1c06dd0841d401f2357aec5bcaa4

Update copyright notices to 2024.

llarryhastings committed 2 months ago
Unverified
cf6f25dd8cc52be88c1b5b93ce64f6937eaafe1d

2.1.5! Added hashed_ballots_tiebreaker.

llarryhastings committed 2 months ago
Unverified
33b124a7879348d67714222eba5619e90d7e2ac8

Add a comment on a vote line to get 100% coverage.

llarryhastings committed 5 months ago
Unverified
970c9c909207d049086359a101c1095cfe49e19e

Minor doc touch-ups.

llarryhastings committed 5 months ago
Unverified
dafb42258f96e4ef8ce3f9f3861c66950532c005

Update: the *current* version passes coverage too.

llarryhastings committed 8 months ago

README

The README file for this repository.

starvote

A STAR Voting election tabulator

Copyright 2023-2024 by Larry Hastings

# test badge # coverage badge # python versions badge

STAR voting is a relatively-new "electoral system"--a method of running an election. STAR Voting is simple--it's simple to vote, and simple to tabulate. And while a completely fair and perfect electoral system is impossible, STAR Voting's approach makes sensible tradeoffs and avoids the worst pitfalls. It's really great!

This module, starvote, implements a STAR Voting tabulator. It requires Python 3.7 or newer, but also supports CPython 3.6. (starvote relies on dictionaries preserving insertion order, which is only guaranteed as of Python 3.7, but happened to work in CPython 3.6.)

Features:

  • Supports five electoral systems:

    • STAR Voting, the snazzy new single-winner voting system.
    • Bloc STAR Voting, a multiwinner variant of STAR voting that fills multiple seats with the most popular candidates.
    • Allocated Score Voting, a proportional representation electoral system, and so far the only such system officially authorized to be a "Proportional STAR Voting" method.
    • Reweighted Range Voting (aka "RRV"), an alternative proportional representation electoral system. RRV isn't a STAR variant, but like STAR it's a form of score voting. So the ballot and instructions to the voter are identical to a STAR-PR election. The RRV algorithm is much simpler than Proportional STAR Voting, and its "never discard a voter" approach is appealing.
    • Sequentially Spent Score, a third variety of score-based proportional representation electoral system.
  • Implements the Official Tiebreaker Protocol for STAR Voting and Bloc STAR Voting elections.

  • Provides a user-configurable final tiebreaker mechanism if the election (or one round in the election) ends in a tie.

    • The default tiebreaker mechanism hashes all ballots and a monotonically increasing number, then uses that data to seed a random number generator, which shuffles a list of candidates, then breaks the tie in favor of the candidate(s) that appear earliest in the list.
    • Alternatively, you can break ties by randomly picking a candidate (or candidates) from the tied candidates, on demand.
    • You can also implement your own custom tiebreaker.
    • If you specify that there should be no tiebreaker, and the election ends in an unbreakable tie, it will raise an UnbreakableTieError exception.
  • All election tabulation calculations are performed using only integers and fractions. This guarantees results are 100% consistent and accurate across all platforms. Floating-point rounding errors are impossible--because floats are never used!

  • Supports running elections specified in CSV files using https://star.vote/ format.

  • Also supports running elections specified in a convenient custom file format called starvote format.

  • starvote passes its test suite with 100% coverage on all supported versions.

  • starvote has no external dependencies.

The current version of starvote is 2.1.5.

A quick STAR Voting primer

When you vote using STAR Voting, your ballot looks something like this:

Amy    0 1 2 3 4 5
Brad   0 1 2 3 4 5
Chuck  0 1 2 3 4 5
Darcy  0 1 2 3 4 5

Here we see the list of candidates, and next to every candidate's name are six numbers, 0 through 5. To vote, give a score to every candidate, one of those numbers from 0 to 5. 5 means you like them the most, 0 means you like them the least. (If you don't give a score to a candidate, that's the same as giving them a 0.) If you give two candidates the same score, that means you like them equally--you don't have a preference between them.

Tabulating the election is easy! You apply the STAR method: Score, Then Automatic Runoff.

In the first round, the Scoring Round, you add up the scores of all the candidates from all the ballots. The top two scoring candidates automatically advance to the second round.

In the second round, the Automatic Runoff Round, you examine every ballot to see which of the two remaining candidates they preferred. If one has a higher score, that ballot prefers that candidate. If the ballot scored both candidates the same, they have no preference. The candidate preferred by more ballots wins the election. It's that simple!

And notice--you always examine every ballot in both rounds. STAR Voting never throws away ballots. When you vote, your vote always matters, every step of the way.

What's so good about STAR Voting?

Electoral systems are a surprisingly deep topic. They've been studied for hundreds of years, and there are many many different approaches. There are a number of desirable properties and undesirable properties that electoral systems can have. And here's the bad news: a best-possible voting system, that has all the desirable properties and none of the undesirable ones, is literally impossible. There are mutually exclusive desirable properties, and there are desirable properties that bring with them undesirable side-effects. Every electoral system is a compromise. (Wikipedia has a comprehensive article on the topic.)

STAR Voting avoid the worst problems of electoral systems. The remaining undesirable properties were chosen as the least-bad option, and in practice STAR Voting generally avoids these undesirable properties anyway--they only happen in unrealistic, contrived scenarios.

Here are some desirable properites guaranteed by STAR Voting:

  • It's monotonic. Giving a candidate a higher score can never hurt them, and giving a candidate a lower score can never help them. (And yes, this isn't always true of voting systems. The increasingly popular Instant Runoff Voting, also often called "Ranked Choice Voting", fails this criterion; as strange as it sounds, it's possible to hurt a candidate you prefer by giving them a higher score.)
  • It's resolvable. Ties are unlikely.
  • It satisfies the majority loser criterion. If a majority of voters rank one candidate last, that candidate will never win a STAR Voting election.

On the other hand, here are some desirable properties STAR Voting lacks:

  • It's not a Condorcet method, which is considered a desirable property for an electoral system. Let's say you have an election with three candidates, A, B, and C. You ask each voter to vote in three head-to-head races: "which do you like better, A or B?", "which do you like better, B or C?", and "which do you like better, A or C?" If there's one candidate that wins in every such head-to-head vote in the election, they would be the "Condorcet winner", and an electoral system that guarantees the "Condorcet winner" will win the election is called a "Condorcet method". STAR isn't a Condorcet method, because Condorcet voting like this doesn't take into consideration the strength of a voter's preference. STAR permits voters to express relative preferences, which arguably can lead to a better result. (On the other hand, STAR does guarantee the opposite: a Condorcet loser will never win a STAR election. And, as a practical matter, it's frequently true that the winner of a STAR election just happens to be the Concorcet winner.)
  • It doesn't satisfy the majority criterion. The majority criterion requires: "if one candidate is ranked first by a majority of voters, that candidate must win". STAR Voting doesn't guarantee this, and it's possible to contrive an election where STAR Voting wouldn't satisfy this. In practice, STAR Voting elections do tend to satisfy this criterion.
  • It doesn't satisfy the later-no-harm criterion. Later-no-harm requires that if you've already expressed a preference for a candidate on your ballot, you shouldn't be able to harm that candidate by expressing another preference for another candidate later on the ballot. STAR fails this; giving a higher vote to a less-preferred candidate might mean that your more-preferred candidate doesn't get elected. The STAR Voting team wrote an essay on why they gave up on this criterion. In short: electoral systems that satisfy later-no-harm generally also exhibit the spoiler effect which is much worse. And achieving later-no-harm and avoiding the spoiler effect makes your electoral system even worse than that!

(If there isn't a best-possible voting system, is there a worst-possible? Maybe! If there is one, it's almost certainly Plurality voting... which is the predominant electoral system used here in the United States. Sigh.)

API

election

To use starvote, first import starvote, then call its election function:

def election(method, ballots, *,
    maximum_score=5,
    print=None,
    seats=1,
    tiebreaker=hashed_ballots_tiebreaker,
    verbosity=0,
    ):

election tabulates an election based on the arguments you pass in and returns a list containing the winners. (Even for single-winner STAR Voting--in that case, the list will only contain one element.)

method specifies which method (which "election system") you want to use in this election, via predefined Method objects. The supported values are:

  • starvote.STAR_Voting,
  • starvote.Bloc_STAR_Voting,
  • starvote.Allocated_Score_Voting,
  • starvote.Reweighted_Range_Voting, and
  • starvote.Sequentially_Spent_Score.

Since those are a lot to type, starvote also provides nicknames for these methods, respectively:

  • starvote.star,
  • starvote.bloc,
  • starvote.allocated,
  • starvote.rrv, and
  • starvote.sss.

ballots should be an iterable containing individual ballots. A ballot is a dict mapping the candidate to that ballot's score for that candidate. The candidate can be any hashable Python value; the score must be an int. (Note that tiebreakers may add additional restrictions to the candidate and score values.)

maximum_score specifies the maximum score allowed for any vote on any ballot.

seats specifies how many seats the election should fill. STAR Voting is a single-winner method, so this should be 1 when using STAR Voting; for the other methods, seats must be greater than or equal to 2.

tiebreaker specifies how to break ties. It should be either a tiebreaker function, a tiebreaker class, or an instance of a tiebreaker class. See the Ties section for a list of predefined tiebreakers.

verbosity specifies how much output you want. In most contexts, the supported values are 0 (no output) and 1 (output); some contexts support higher verbosity values to mean "print more information", e.g. hashed_ballots_tiebreaker produces incremental output for verbosity levels 2 and 3.

print lets you specify your own printing function. By default election will use builtins.print; your replacement print function should have the same signature.

Functions for specific electoral systems

starvote also exports functions implementing each of the supported electoral systems:

  • star_voting implements single-winner STAR Voting.
  • bloc_star_voting implements multiwinner Bloc STAR Voting.
  • allocated_score_voting implements Allocated Score Voting.
  • reweighted_range_voting implements Reweighted Range Voting.
  • sequentially_spent_score implements Sequentially Spent Score.

These functions have much the same signature as election, with the following changes:

  • They don't have a method parameter; the method is implicit in the function. All five only take one positional parameter, ballots, which is required.
  • star doesn't have a seats parameter. The others do accept a seats keyword-only parameter, and it's required--it doesn't have a default. (A required keyword-only parameter is pretty rare in Python!)
  • Note that, like election, these functions always return a list, even star_voting.

Reference implementation of Allocated Score Voting

starvote ships a copy of the reference implementation of Allocated Score Voting. Since this requires both NumPy and Pandas, it's not imported by default. (I didn't want starvote to have any external dependencies. The unit test suite runs correctly whether or not these external dependencies are installed.)

You can import it with import starvote.reference; the directly-callable function is starvote.reference.allocated_score_voting_reference. You can also use the Method object starvote.reference.Allocated_Score_Voting_reference, with the nickname starvote.reference.allocated_r.

If you want to integrate it into the starvote module, callstarvote.reference.monkey_patch(). This does three things:

  • Adds the function and Method objects to the starvote module directly.

    • Also adds them to starvote.__all__ so they get brought in by from starvote import *.
  • Adds the appropriate Method object to starvote.methods, using the name 'Allocated Score Voting (reference)' and the nickname allocated_r.

Note: the reference implementation doesn't support tiebreakers. allocated_score_voting_reference does accept a tiebreaker argument, but currently it must be None.

Ties

STAR Voting elections rarely result in a tie in the real world. But ties can still happen. The good news is, STAR Voting has a sensible, thorough protocol on how to break a tie. The bad news is, not all ties are breakable--some ties genuinely represent the indecisive will of the voters.

starvote gives you control over how to break ties in elections, through the tiebreaker parameter to election and the election-specific functions. starvote also has two predefined tiebreaker functions, but you can substitute your own--or none at all.

The default tiebreaker in starvote is predefined_permutation_tiebreaker, passing in candidates=None.

hashed_ballots_tiebreaker

starvote's preferred--and default--tiebreaker is hashed_ballots_tiebreaker. This is a class; you should instantiate it and pass in the instance as the tiebreaker argument when you run the election.

hashed_ballots_tiebreaker is the preferred tiebreaker for starvote because it's

  • impossible to usefully control externally,
  • impossible to predict, yet
  • completely deterministic.

Note that the default serializer used by hashed_ballots_tiebreaker requires all candidates to be str objects, and all votes have to be int objects. If you don't change any defaults, you must restrict yourself to these types (which you probably were already doing anyway).

Here's hashed_ballots_tiebreaker it works. At initialization time, this tiebreaker:

  • computes a list of all candidates, then
  • sorts the list of candidates and stores this list, then
  • sorts each ballot, then
  • sorts a list of all the sorted ballots, then
  • converts this sorted list of sorted ballots into a binary string (using a custom binary serializer by default).

Then, when it's asked to break a tie, it

  • hashes a serialized monotonically increasing counter (1 by default, incremented after every tiebreaker) followed by the binary string of sorted ballots, using a cryptographically secure hash (sha3_512 by default), then
  • uses the digest produced by the hash to seed a random number generator (Python's random.Random by default), then
  • randomly shuffles the list of candidates using that random number generator object some number of times (3 by default).

There are no random inputs into this process, which means it's completely deterministic. However, it would be impossible to predict the result of the process unless you knew the contents of all the ballots, making it as unpredictable as a method employing random numbers.

Also, the only way to usefully control this algorithm is to control the seed value--and to do that you'd have to control all the ballots. And if you control all the ballots, you don't need to influence the tiebreaker! You can just change the votes to directly produce whatever outcome you prefer.

predefined_permutation_tiebreaker

Another tiebreaker for starvote is predefined_permutation_tiebreaker. This is a class; you should instantiate it and pass in the instance as the tiebreaker argument when you run the election.

predefined_permutation_tiebreaker resolves the tie in favor of an ordered list of candidates.

Given these three variables, defined at the time it breaks the tie:

  • candidates, an ordered list of all candidates participating in the election,
  • tie, an iterable of the tied candidates, and
  • desired, the number of winners we wanted.

predefined_permutation_tiebreaker computes the winners of the tie as follows:

winners = [c for c in candidates if c in tie][:desired]

In other words, it returns a desired-length subset of tie, preferring candidates that appear earlier in candidates over those that appear later.

predefined_permutation_tiebreaker accepts a candidates argument, which should be an ordered pre-chosen list of all candidates. The default value is None, which instructs predefined_permutation_tiebreaker to scan the ballots at the start of the election, produce its own list of all the candidates, randomly shuffle that list, and use that list as the candidates list.

If you pass in your own candidates list, you may also pass in a string for the description keyword-only parameter, which should be text describing the source of this ordered list of candidates.

Note that, if you pass in your own candidates list, starvote's tabulation of the election will be 100% deterministic and repeatable. You can run that election a million times and you'll always get the same result.

predefined_permutation_tiebreaker also accepts a random keyword-only argument. This should be a random.Random object, which will be used for all randomization inside predefined_permutation_tiebreaker. Passing in an instance of random.Random with a fixed predefined seed will make predefined_permutation_tiebreaker completely deterministic. The default value of None means predefined_permutation_tiebreaker will use the random module itself for all its randomization needs.

on_demand_random_tiebreaker

If you prefer more unpredictability in your life, you can choose on_demand_random_tiebreaker to break ties. on_demand_random_tiebreaker will simply pick a random candidate (or candidates) on demand, using Python's random.sample function. on_demand_random_tiebreaker is a function; you should simply pass it in as the tiebreaker parameter.

If you prefer less unpredictability in your life, you can also use the random keyword-only argument. This argument has the same semantics as the random argument to predefined_permutation_tiebreaker. If you pass in a random number generator using a fixed seed, on_demand_random_tiebreaker will be completely deterministic.

In order to use this with starvote.election (et al), you'll have to use partial application to pre-bind the random argument, something like this:

t = functools.partial(
    starvote.on_demand_random_tiebreaker,
    random=random.Random(54321))
result = starvote.election(..., tiebreaker=t)

UnbreakableTieError

If you explicitly set tiebreaker to None, and an election results in a tie, starvote will raise an UnbreakableTieError exception. You can get a text description of the tie by calling str on the exception; you can also get a list of the tied candidates in its candidates attribute.

Writing your own tiebreaker

You can write your own tiebreaker and pass it in in the tiebreaker parameter to your chosen election function. Custom tiebreakers can be either a function or a class.

A custom tiebreaker function should have this signature:

def custom_tiebreaker(options, tie, desired, exception):

Here's what these four parameters will contain:

  • options is a starvote Options object. options has attributes mapping to the arguments you passed in to election. You should obey options.verbosity when deciding whether or not to print, and you should print using options.print. options also has the following convenience methods:

    • options.heading is a context manager that prints a heading to the output if options.verbosity is 1 or greater. You pass in the heading as a string. You can nest headings.
    • options.print_candidates will print an iterable of candidates. You pass in the iterable as the first (and only) positional argument. Also accepts a numbered parameter; if numbered is true, it prints the candidates in order, prepended by their ordinal number (starting with 1). If numbered is false, print_candidates attempts to sort the candidates before printing them.
  • tie is a list of the tied candidates.

  • desired is the desired number of winners.

  • exception is the UnbreakableTieError for this tie. If you can't break the tie, you should raise this object.

Note that starvote will also parse the tiebreaker function's docstring. The first line of the docstring will be used as a "heading" printed during election initialization, with the rest of the docstring will be printed as the body of that heading if options.verbosity is 1 or greater.

You can also write a custom tiebreaker class. The only requirement is that the class inherits from starvote.Tiebreaker, and it supports a __call__ method with the same signature as a custom tiebreaker function. (However, here the docstring is ignored; it's up to you to call options.heading and options.print.)

Tiebreaker classes can also optionally have an initialize method:

def initialize(self, options, ballots):

This will be called during initialization of the election, before any processing of the votes. Here's an explanation of the two parameters:

  • options is the same object as the options passed in to a custom tiebreaker function.

  • ballots is the iterable of ballots passed in to the election function.

Whichever kind of tiebreaker you write, you should add it to starvote.tiebreakers. That's a dict mapping names to tiebreakers. Tiebreakers you add to that dict will be available to starvote format elections parsed by parse_starvote.

Code example

Here's an example of computing a poll between Amy, Brian, and Chuck:

import starvote

ballots = [
    {'Amy': 1, 'Brian': 3, 'Chuck': 5},
    {'Amy': 5, 'Brian': 2, 'Chuck': 3},
    {'Amy': 4, 'Brian': 4, 'Chuck': 5},
]

winners = starvote.election(starvote.star, ballots, verbosity=1)

(This example is included as example.py in the starvote Git repo.)

After this example code finishes, the winners variable will contain the list ['Chuck']. The example also produces this output:

[STAR Voting]
  Tabulating 3 ballots.
  Maximum score is 5.

[STAR Voting: Initializing ordered permutation tiebreaker]
  Computing a random permutation of all the candidates.
  Permuted list of candidates:
    1. Brian
    2. Chuck
    3. Amy
  Tiebreaker candidates will be selected from this list, preferring candidates with lower numbers.

[STAR Voting: Scoring Round]
  The two highest-scoring candidates advance to the next round.
    Chuck -- 13 (average 4+1/3) -- First place
    Amy   -- 10 (average 3+1/3) -- Second place
    Brian --  9 (average 3)
  Chuck and Amy advance.

[STAR Voting: Automatic Runoff Round]
  The candidate preferred in the most head-to-head matchups wins.
    Chuck         -- 2 -- First place
    Amy           -- 1
    No Preference -- 0
  Chuck wins.

[STAR Voting: Winner]
  Chuck

starvote format

starvote also defines a custom text format for specifying elections called starvote format. It's heavily used for testing starvote itself, though it could be used to run real elections.

starvote format looks kind of like INI format, but it isn't exactly the same.

Why'd I write this? I got tired of CSV files. It's very handy to add metadata about how to run the election to the election file itself. And starvote format is so much easier to read and edit than the equivalent https://star.vote/-format CSV file.

Definition

A string that contains an election in starvote format is called a starvote format election.

starvote format is a line-oriented format. Leading and trailing whitespace per-line is ignored (and stripped). Lines that start with # are comments. Empty lines and comments are (mostly) ignored.

Non-empty lines that aren't comments are either an assignment line, a pragma line, or a section line.

A pragma line ends with a colon (:). Pragma lines are free-form; currently there's only one defined pragma. (When parsing a line, pragma take precedence over assignment.)

An assignment line must contain an equals sign (=); the text before the equals sign is the "name", and the text after the equals sign is the "value". What effect this has depends on what section we're in.

A line that starts with [ and ends with ] defines a section. You specify the name of the section between the square brackets. You can only specify a section once. Only two sections are supported: options and ballots.

The options section specifies how to run the election. Assignment lines in the options section specify options; each name maps to a parameter to the election function. Here are all the supported names:

    csv_path = <string>
    maximum score = <integer>
    method = <string>
    seats = <integer>
    starvote_path = <string>
    tiebreaker = <string | list>
    verbosity = <integer>

A starvote format election can specify each of these a maximum of once.

An assignment line in the options section can also use list mode. list mode allows you to set an option to a list of values, rather than a single value. To use, set a name to the value [. The "name" is set to an empty string, and the starvote format parser activates list mode. While in list mode, non-empty lines are appended to the list currently being defined. To deactivate list mode and finish defining the list, specify ] on a line by itself. Notes:

  • If you set a value to [], it's set to an empty list, and the parser doesn't enable list mode.
  • You can't use pragmas or change sections while in list mode.
  • You can't nest lists.

The only assignment that supports lists is the tiebreaker option in the options section. tiebreaker also supports string values.

If tiebreaker is set to a string, parse_starvote looks up that string in the options.tiebreakers dict and uses the tiebreaker found there. This string can optionally end with a list of name=value options in parentheses, separated by commas. The only option currently defined is seed. seed should be set to an integer; this integer will be used to seed a Python random.Random instance, and that instance will be used for all randomization in the tiebreaker, effectively making it completely deterministic. Example:

tiebreaker=predefined_permutation_tiebreaker(seed=12345)

If tiebreaker is set to a list, this defines a pre-permuted list of candidates which is passed in to predefined_permutation_tiebreaker. This lets you predefine a permuted list of candidates, which makes the election completely deterministic. Example:

tiebreaker=[
    Amy
    Brian
    Carol
    ]

The ballots section defines ballots. In this section, names are candidate names, and values are the score for that candidate. Individual ballots are separated by blank lines and/or comment lines--to start a new ballot, just add a blank line, or a comment line.

The ballots section supports one pragma: ballots. This lets you repeat a ballot multiple times. To use, add a line to the ballots section as follows:

    n ballots:

where n is the number of times you want to repeat a ballot. This will repeat the subsequent ballot n times. For example, to repeat a ballot 5 times, add this line just above the ballot:

    5 ballots:

(You're explicitly permitted to have blank lines between the ballots pragma and the ballot it's repeating.)

The starvote_path and csv_path options allow you to "import" another file into the current election. They can specify relative or absolute paths; if relative, they will be resolved using the directory the starvote format file is in (if the starvote format election was loaded from a file) or the current directory. Both settings will load ballots from the file specified; starvote_path will also load the options from the file specified, however any options set in the current election will override those options.

Note that starvote_path is fully recursive. Starvote file A can import starvote file B, which in turn imports starvote file C, etc. etc.

Example

Here's a sample starvote format election:

[options]

seats=3
method=Bloc
tiebreaker = [
  Chuck
  Brian
  Amy
  ]
verbosity = 1

[ballots]

Amy = 1
Brian = 2
Chuck = 5

Amy = 1
Brian = 5
Chuck = 3

Amy = 1
Brian = 3
Chuck = 3

3 ballots:
Amy = 1
Brian = 3
Chuck = 3

APIs

The starvote module provides two functions that handle starvote format.

The first is parse_starvote:

starvote.parse_starvote(starvote, *, path="<string>")

parse_starvote takes one positional parameter: election, which must be a string in starvote format. It parses the string and returns a kwargs dict. You can run that election by calling election and passing in this dict, using ** to turn the contents of the dict into keyword arguments:

starvote.election(**kwargs)

You may also pass in a keyword-only parameter path, which should represent the filename if the starvote format election was loaded from a file; if specified, it'll be included in exceptions, for context.

The second is load_starvote_file:

starvote.load_starvote_file(path, *, encoding='utf-8')

load_starvote_file takes one positional parameter: path, which must be a str, bytes, or pathlib.Path object. It opens that file, reads and decodes its contents, passes those contents in to parse_starvote, and returns the result. You may specify the text encoding using the encoding keyword-only parameter; the default encoding is 'utf-8'.

Command-line module

The starvote module supports being run as a script (python -m starvote). Run it without arguments to see usage.

To use, specify the path to a single file on the command-line. starvote will read in the file, run the election, and print the result.

The path may be a CSV file. CSV files should end with the file extension .csv, and be in https://star.vote/ format. By default elections in CSV files are run using STAR Voting, for one seat, with verbosity=1 and the default tiebreaker.

Alternatively, the path may be a starvote format file. starvote format files should end with the file extension .starvote and contain a starvote format election encoded in UTF-8. starvote format elections explicitly specify all the parameters of the election.

For example, you can run this from the root of the source-code repository:

% python3 -m starvote test_elections/test_election_breakable_tie_in_automatic_runoff_round_using_max_score_count_round.starvote

to see how starvote handles ties during the automatic runoff round.

Multiple-winner elections

starvote also implements several multiwinner electoral systems. All you need to do is pass in one of the multiwinner methods, such as starvote.bloc, starvote.allocated, or starvote.rrv, when you call election:

poll = election(starvote.bloc, ballots, seats=2)

You can experiment with these with the command-line version of the module too. Simply specify the method with -m, the number of seats with -s, and the maximum score with -x. These will override the settings from inside a starvote format file (and the default settings for CSV files).

Multiwinner vs proportional

Here's a quick explanation about what "proportional voting" means, in the context of multiwinner elections. As with every other topic, you can read more about it at Wikipedia.

A straight multiwinner election simply means you're electing 2 or more candidates instead of one candidate. The candidates that get the most votes--or however the election is tabulated--win.

But electing only the most popular candidates can be a poor representation of the electorate. Let's say you have a city election to fill five seats. In this city, 60% of the voters vote only for party A, and 40% of the voters vote only for party B. Bloc STAR Voting would very likely award all five seats to party A candidates. Is that fair? It seems like the "tyranny of the majority".

There's an alternate approach to apportioning seats, used in political systems across the world, called "Proportional Representation". The idea is, you have N seats, and you assign portions of them based on representing portions of the populace. In the above example, you'd want to give three seats to party A and two seats to party B. What system could do that?

Often proportional representation is done based on voting for parties rather than voting for candidates. This is called Party-list proportional representation, and it's used to elect governmental bodies across the world.

But there are voting methods that permit voting directly for candidates and produces proportional representation. Allocated Score Voting, Reweighted Range Voting, and Sequentially Spent Score are three such methods. They all work something like this:

  • Each vote is a number, with higher numbers indicating a stronger preference.
  • You run N rounds of the election to fill N seats, each round electing one candidate.
  • When a candidate is elected, we may allocate ballots to that candidate, which means we consider them used-up and we remove them from the election. If you vote for candidate A1, and candidate A1 wins the first round, your ballot might get "allocated" to candidate A1 and ignored for the rest of the election.
  • Alternatively, we may re-weigh the ballots of voters who voted for that candidate. That means we reduce the voting power of those ballots. If in the election, you voted for candidate A1, and A1 wins the first round, in the second round your vote will count for less. But if I gave A1 a score of 0, my ballot would still be at full strength.

The differences between the three is how they allocate vs. reweigh ballots. Allocated Score Voting and Sequentially Spent Score both allocate ballots, Reweighted Range Voting never does--RRV counts every vote in every round. Also, the three systems use different formulae to compute the new weight of a ballot after each round. In Allocated Score and Sequentially Spent Score, the new weight is based on how many excess voters were needed to fill a quota (called a Hare quota); in Reweighted Range Voting, the new weight is based on the sum of the score(s) you contribute to the winning candidate(s).

starvote ships a sample election that nicely demonstrates how direct-elected proportional representation elections can work:

test_elections/test_election_reweighted_range_sample_election.starvote

This election is like the example election I used in the description above. It uses Reweighted Range Voting to fill three seats, and each vote has a maximum score of 10. 60 of the voters prefer party A, and give high scores to candidates A1, A2, and A3; 40 of the voters prefer party B, and give high scores to candidates B1 and B2.

You can run the election in the starvote main directory with this command:

% python3 -m starvote test_elections/test_election_reweighted_range_sample_election.starvote

To see how the tabulation works using Allocated Score Voting, run this:

% python3 -m starvote -m allocated test_elections/test_election_reweighted_range_sample_election.starvote

You can also see how it changes with Sequentially Spent Score by running this:

% python3 -m starvote -m allocated test_elections/test_election_reweighted_range_sample_election.starvote

And to see the tyrrany of the majority in action, this command will tabulate the election using Bloc STAR Voting:

% python3 -m starvote -m bloc test_elections/test_election_reweighted_range_sample_election.starvote

Warning

I haven't found a single test corpus for Bloc STAR Voting. I'm following the rules as best I can, and the results I'm getting make sense. But so far I can't confirm my implementation is correct--there's a very real possibility I got something wrong.

I do have one sample poll for Reweighted Range Voting, so I have some confidence that my implementation is okay. And I do test against the reference implementation of Allocated Score Voting a little.

Thanks

Huge thanks to Tim Peters for his continuous input during the development of this library. Although Tim didn't have any input on the library itself--if you don't like the library it's 100% my fault!--he tirelessly answered my questions about voting during its development, and convinced me to change my approach several times. In particular, Tim's feedback pushed me to develop the tiebreaker plug-in interface.

License

starvote is licensed using the MIT license. See the LICENSE file.

It seems particularly relevant to repeat here: there is no warranty for this software. I've done the best job I can implementing this election system tabulator. But this software could have bugs, or my understanding of the rules could be wrong, and either of these could affect the results of elections you run with this software. Use at your own risk.

The source code repository includes sample ballots downloaded from https://star.vote/. The licensing of these sample ballots is unclear, but they're assumed to be public-domain or otherwise freely redistributable.

Changelog

2.1.6 - 2024/12/13

  • Bugfix: previously, hashed_ballots_tiebreaker used marshal.dumps as its binary serializer, because I assumed given identical objects it would always produce an identical bytes string. This is not true! (And thanks to Petr Viktorin for pointing it out!) We also apparently can't rely on pickle.dumps to be deterministic, for similar reasons.

    So, starvote now has its own bespoke--and completely deterministic--simple binary serializer, called starvote_custom_serializer. It's tailor-made for the needs of starvote and isn't useful for anybody else. But it does guarantee that hashed_ballots_tiebreaker will now produce identical results across all supported Python versions, across all architectures, regardless of optimization level.

    (There's also a matching deserializer, naturally called starvote_custom_deserializer. You shouldn't need to use it either.)

2.1.5 - 2024/11/22

  • New tiebreaker: The hashed_ballots_tiebreaker. This uses a fiendish, twisty bit of computation to produce a tiebreaker that's

    • impossible to usefully control externally,
    • completely unpredictable, and yet also
    • completely deterministic.

    This is a big improvement over the previous tiebreaker implementations! Therefore I've made this the new default tiebreaker.

  • The -t | --tiebreaker command-line option for the module ("python3 -m starvote") is now documented. It's been implemented for a while, it just wasn't documented.

  • When specifying the name of the tiebreaker for the -t and --tiebreaker command-line options, and also for the tiebreaker option in a .starvote file, you may now omit the _tiebreaker at the end of the name.

2.1.2 - 2023/10/02

  • New feature: you can specify a seed for the pseudorandom number generator (PRNG) used for random tiebreakers in a starvote format election. The seed is simply an integer number. This required new syntax for the tiebreaker option: you can specify options using a syntax looks something like a Python function call with keyword-only parameters. For example, to use predefined_permutation_tiebreaker with a PRNG seeded with the number 12345, add this to the options section of your starvote format election:
    tiebreaker=predefined_permutation_tiebreaker(seed=12345)
  • New feature: predefined_permutation_tiebreaker and on_demand_random_tiebreaker now both accept a keyword-only parameter named random, which should be either an instance of random.Random or None. This lets you pass in a random.Random instance with a fixed seed which the tiebreaker will use for all randomization--a convenient way to make the tiebreaker completely deterministic. The default value of None means the tiebreaker will use the random module itself for randomization.

p.s. starvote now has over a hundred tests!

2.1.1 - 2023/09/20

  • Bugfix: starvote format used to permit specifying the same candidate more than once on the same ballot; only the last vote would count. It now detects this and raises ValueError on the second instance. Fixes #8.

2.1 - 2023/09/20

  • Change to verbosity. If verbosity is 1, the permuted tiebreaker doesn't print any output at initialization time. Instead, if there's a tiebreaker, it prints the pre-permuted list of candidates immediately before printing the results of the tiebreaker. Note that this doesn't change when the initialization is done; when the the list of candidates is randomly permuted by starvote, that's still done before running the election. If verbosity is 2 or higher, the permuted tiebreaker initialization prints its output at initialization time as before. Fixes #3.
  • Feature request: added blank lines before section headings for verbose output. Fixes #2.
  • Bugfix: When "No Preference" is printed in the output of a preference round, this should be the number of ballots that expressed no preference between the candidates. Previously it was printing the total count of head-to-head matchups where the voter expressed no preference, and there could be more than one of those per ballot. Fixes #7.
  • Bugfix: For multi-winner elections, seats must be less than or equal to the number of candidates. starvote now raises an error when asked to tabulate an election where this is not true. Fixes #6.
  • Slight internal API change: Tiebreaker.print_description accepts two arguments, the second being the description string / list. Since it's always self.description anyway, it now has a default value of None which means it uses self.description.

2.0.6 - 2023/07/22

Extremely minor release. No new features or bug fixes.

  • Added GitHub Actions integration. Tests and coverage are run in the cloud after every checkin. Thanks to Dan Pope for gently walking me through this!
  • Fixed metadata in the pyproject.toml file.
  • Added badges for testing, coverage, and supported Python versions.

2.0.5 - 2023/07/05

  • No code changes, just a tweak to the license to remove the admittedly-confusing phrase "All rights reserved." Also, embedded the license in the module.

2.0.4 - 2023/06/05

  • Added support for Sequentially Spent Score voting.
  • Changed presentation slightly for Allocated Score: the average vote is now computed using the count of all ballots in the election, including allocated ballots. (Previously the average was computing using just the remaining ballots.) This doesn't change the outcome of the election, it's just a presentation change.
  • Removed last traces of "STAR-PR", which I thought was an alternate name for Allocated Score.
  • Doc changes. Standardized on the spelling "multiwinner", instead of "multi-winner".

2.0.3 - 2023/06/01

  • Normalized nomenclature for Method objects. Now, every method has an official correctly-spelled correctly-capitalized name, and exactly one "nickname". The nickname is always lowercase, and would return True for isidentifier. These are used in the starvote.method map; a variant of these names is also bound as module attributes (changed so they're valid identifiers).
  • Improved starvote.reference.monkey_patch. It's now data-driven and thus more reliable.
  • Added starvote.reference.__all__, in case you want to from starvote.reference import *.
  • Doc fixes.

2.0.2 - 2023/06/01

  • Renamed the tiebreaker parameter candidates to tie.
  • Evicted some testing-only tiebreakers from the starvote module. They're now in their own script, which only gets loaded when working with the test suite.

2.0.1 - 2023/06/01

  • Changed the nickname of the reference version of Allocated Score Voting to "Allocated-R".

2.0 - 2023/06/01

A complete rewrite! The 1.x code base was pretty smelly. This codebase is much, much cleaner--and I think I squashed one or two bugs too.

starvote has an entirely new, functional API. election runs the election for you, and takes two required positional parameters: method, which specifies which electoral system you want, and ballots, an iterable of ballot dicts.

You can also now pass in a handler for unbreakable ties. By default, unbreakable ties are now broken using a pre-chosen randomize list of candidates. (election can still raise UnbreakableTieError exceptions if you prefer.)

You can also call the voting method functions directly: star, bloc_star, proportional_star, reweighted_range, are all functions, too. These omit the method parameter but still require the ballots parameter.

There's also a new text format for storing an election, starvote format, with the .starvote file extension. starvote format is a nice alternative to .csv files.

  • Added the election, star_voting, bloc_star_voting, allocated_score_voting, and reweighted_range_voting functions.

    • Removed the Poll class.
  • Now consistently use the official names for all methods:

    • STAR Voting
    • Bloc STAR Voting
    • Allocated Score Voting
    • Reweighted Range Voting

    ("Proportional STAR Voting" is a category of electoral systems. It's not itself a voting system, and it's definitely not restricted to "Allocated Score Voting". "STAR-PR" is another name that category.)

  • Replaced the ElectoralSystem enum with the Method class. Instances (e.g. starvote.STAR_Voting) contain all the metadata needed by election to run the election.

    • methods is a module-level dict mapping strings to Method objects.
  • Implemented the STAR Voting Official Tiebreaker Protocol for STAR Voting and Bloc STAR Voting.

  • All vote tabulation uses strictly int and fractions.Fraction objects. Vote tabulation is now 100% consistent and accurate, from run to run, across all platforms.

  • Added the parse_starvote function, which parses a starvote format string, runs the election, and returns the result.

  • Added the load_starvote_file function, which loads a starvote format file from disk and parses it with parse_starvote.

  • Added the Tiebreaker class, and the on_demand_random_tiebreaker and predefined_permutation_tiebreaker tiebreakers.

    • tiebreakers is a module-level dict mapping strings to tiebreakers.
  • Added the reference implementation of Allocated Score Voting. This requires both NumPy and Pandas, so it's not imported by default. You can import it with import starvote.reference, and you can integrate it into the starvote module by calling starvote.reference.monkey_patch(). The reference implementation doesn't support tiebreakers.

  • When running the module from the command-line using python -m starvote:

    • You may now specify a .starvote file. Its contents should be a starvote format election, encoded using UTF-8.

    • When specifying a .csv file, starvote uses several default values: method is STAR Voting, seats is 1, verbosity is 1.

1.5.1 - 2023/05/24

  • Renamed a bunch of names in the API.
    • Renamed PollVariant enum to ElectoralSystem.
    • Renamed variant parameter to electoral_system.
    • Renamed max_score parameter to maximum_score.
  • Changed command-line module options to match.
    • Changed -v|--variant to -e|--electoral-system.
    • Changed -m|--max_score to -m|--maximum_score.

1.5 - 2023/05/22

  • Added support for Reweighted Range Voting, an attractive alternative to Proportional STAR. Like STAR-PR, RRV is a proportional representation electoral system. But RRV is simpler to understand, simpler to implement, and it never throws away votes. Thanks to Tim Peters for suggesting it!
  • Added the max_score parameter to the Poll constructor. Now you can use whatever range you like. (The minimum score is still always 0.)
  • Changed the spelling of "Bloc STAR". I thought the "Bloc" was always properly capitalized (as "BLOC STAR"), but nope, it's not.

1.4 - 2023/05/21

  • Automated the test suite.
  • Add logging prints for tie-breaker preference round for Proportional STAR.
  • Fixed presentation in __main__ for multiple winner elections that end in a tie.

1.3 - 2023/05/21

  • Added support for Proportional STAR polls. The only visible external change is the new Proportional_STAR enum value.
  • Renamed the winners parameter on the Poll constructor to seats. Sorry to break your code, all zero people planetwide who already started using the parameter! But this new name is a big improvement.

1.2 - 2023/05/20

  • Add support for Bloc STAR polls:

    • Added PollVariant enum containing STAR and BLOC_STAR values.
    • Added variant and winners parameters to Poll.
  • Add the list of tied candidates to the UnbreakableTieError exception as the new candidates attribute.

1.1 - 2023/05/20

  • Bugfix: raise UnbreakableTieError if there's a three-way tie for second place. Previously starvote only noticed if there was a three-way tie for first place.
  • Added sample output for every sample poll in sample_polls/. These outputs have been confirmed correct by inspection, and could in the future be used as part of an automated test suite.

1.0 - 2023/05/20

  • Initial release.