-
Notifications
You must be signed in to change notification settings - Fork 3
/
audit_type.rs
153 lines (128 loc) · 7.11 KB
/
audit_type.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
// Copyright 2023 Andrew Conway.
// Based on software (c) Michelle Blom in C++ https://github.com/michelleblom/audit-irv-cp/tree/raire-branch
// documented in https://arxiv.org/pdf/1903.08804.pdf
//
// This file is part of raire-rs.
// raire-rs is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
// raire-rs is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details.
// You should have received a copy of the GNU Affero General Public License along with ConcreteSTV. If not, see <https://www.gnu.org/licenses/>.
//! The types of assertions about the election, generally relative standings of various candidates
use crate::irv::{BallotPaperCount};
use serde::Deserialize;
use serde::Serialize;
pub type AssertionDifficulty = f64;
/// An audit type is a method for determining a difficulty (higher means more difficult, infinite means impossible) for
/// a comparison of two claimed tallys.
///
/// Often the actual number computed is more complex than some simpler number that is a monotonic proxy for the
/// difficulty, and the raire algorithm could use this at some improvement in execution time and possibly memory use.
/// This is not done as:
/// * The gains would be very small and the extra complexity would be significant
/// * It makes pre-specifying the difficulty difficult.
pub trait AuditType {
fn difficulty(&self, lowest_tally_winner:BallotPaperCount, highest_tally_loser:BallotPaperCount) -> AssertionDifficulty;
}
/// A BRAVO ballot polling audit as described in the original paper.
#[derive(Clone,Copy,Debug,Serialize,Deserialize)]
pub struct BallotPollingBRAVO {
/// The desired confidence α. A number between 0 and 1 bounding the probability of not rejecting a false result.
pub confidence : f64,
pub total_auditable_ballots : BallotPaperCount,
}
impl BallotPollingBRAVO {
/// compute ASN using the BRAVO method described in the original paper.
pub fn average_sample_number_original_paper_using_total_auditable_ballots(&self,lowest_tally_winner:BallotPaperCount,highest_tally_loser:BallotPaperCount) -> AssertionDifficulty {
// println!("Doing BRAVO with winner {lowest_tally_winner} loser {highest_tally_loser} active_paper_count={active_paper_count} α={}",self.confidence);
self.bravo_function(lowest_tally_winner,highest_tally_loser,self.total_auditable_ballots)
}
/// This function is only public for testing some historical data. You probably don't want to use this directly.
pub fn bravo_function(&self,winner_tally:BallotPaperCount,loser_tally:BallotPaperCount,paper_count:BallotPaperCount) -> AssertionDifficulty {
if winner_tally.0<=loser_tally.0 { f64::INFINITY } else {
let w = winner_tally.0 as f64;
let l = loser_tally.0 as f64;
let s = w/(w+l);
let twos = 2.0*s;
let ln2s = twos.ln();
let numerator = 0.5*ln2s-self.confidence.ln();
let denominator = (w*ln2s+l*(2.0-twos).ln())/(paper_count.0 as f64);
numerator/denominator
}
}
}
impl AuditType for BallotPollingBRAVO {
fn difficulty(&self, lowest_tally_winner: BallotPaperCount, highest_tally_loser: BallotPaperCount) -> AssertionDifficulty {
self.average_sample_number_original_paper_using_total_auditable_ballots(lowest_tally_winner,highest_tally_loser)
}
}
/// A MACRO ballot level comparison audit.
#[derive(Clone,Copy,Debug,Serialize,Deserialize)]
pub struct BallotComparisonMACRO {
/// The desired confidence α. A number between 0 and 1 bounding the probability of not rejecting a false result.
pub confidence : f64,
/// γ ≥ 1
pub error_inflation_factor : f64,
pub total_auditable_ballots : BallotPaperCount,
}
impl BallotComparisonMACRO {
/// Compute ASN using the
pub fn average_sample_number_original_paper(&self,lowest_tally_winner:BallotPaperCount,highest_tally_loser:BallotPaperCount) -> AssertionDifficulty {
if lowest_tally_winner<=highest_tally_loser { f64::INFINITY } else {
let v = lowest_tally_winner-highest_tally_loser;
let u = 2.0*self.error_inflation_factor*self.total_auditable_ballots.0 as f64/v.0 as f64;
-self.confidence.ln()*u
}
}
}
impl AuditType for BallotComparisonMACRO {
fn difficulty(&self, lowest_tally_winner: BallotPaperCount, highest_tally_loser: BallotPaperCount) -> AssertionDifficulty {
self.average_sample_number_original_paper(lowest_tally_winner,highest_tally_loser)
}
}
/// A comparison where the difficulty = 1/diluted margin.
/// Useful for BallotComparison audits
#[derive(Clone,Copy,Debug,Serialize,Deserialize)]
pub struct BallotComparisonOneOnDilutedMargin {
pub total_auditable_ballots : BallotPaperCount,
}
impl AuditType for BallotComparisonOneOnDilutedMargin {
fn difficulty(&self, lowest_tally_winner: BallotPaperCount, highest_tally_loser: BallotPaperCount) -> AssertionDifficulty {
if lowest_tally_winner<=highest_tally_loser { f64::INFINITY } else {
let reciprocal_diluted_margin = self.total_auditable_ballots.0 as f64/(lowest_tally_winner-highest_tally_loser).0 as f64;
reciprocal_diluted_margin
}
}
}
/// A comparison where the difficulty = 1/diluted margin^2.
/// Useful for Ballot Polling audits.
#[derive(Clone,Copy,Debug,Serialize,Deserialize)]
pub struct BallotPollingOneOnDilutedMarginSquared {
pub total_auditable_ballots : BallotPaperCount,
}
impl AuditType for BallotPollingOneOnDilutedMarginSquared {
fn difficulty(&self, lowest_tally_winner: BallotPaperCount, highest_tally_loser: BallotPaperCount) -> AssertionDifficulty {
if lowest_tally_winner<=highest_tally_loser { f64::INFINITY } else {
let reciprocal_diluted_margin = self.total_auditable_ballots.0 as f64/(lowest_tally_winner-highest_tally_loser).0 as f64;
reciprocal_diluted_margin*reciprocal_diluted_margin
}
}
}
#[derive(Clone,Debug,Serialize,Deserialize)]
#[serde(tag = "type")]
pub enum Audit {
BRAVO(BallotPollingBRAVO),
MACRO(BallotComparisonMACRO),
#[serde(alias = "Margin")] // for backwards compatibility
OneOnMargin(BallotComparisonOneOnDilutedMargin),
#[serde(alias = "MarginSq")] // for backwards compatibility
OneOnMarginSq(BallotPollingOneOnDilutedMarginSquared),
}
impl AuditType for Audit {
fn difficulty(&self, lowest_tally_winner: BallotPaperCount, highest_tally_loser: BallotPaperCount) -> AssertionDifficulty {
match self {
Audit::BRAVO(audit) => audit.difficulty(lowest_tally_winner,highest_tally_loser),
Audit::MACRO(audit) => audit.difficulty(lowest_tally_winner,highest_tally_loser),
Audit::OneOnMargin(audit) => audit.difficulty(lowest_tally_winner, highest_tally_loser),
Audit::OneOnMarginSq(audit) => audit.difficulty(lowest_tally_winner, highest_tally_loser),
}
}
}