Check for ties when electing candidates with surpluses
Refactor constraint-related code into constraints module
This commit is contained in:
		
							parent
							
								
									7e3d015be3
								
							
						
					
					
						commit
						34545ad179
					
				| @ -15,9 +15,11 @@ | ||||
|  *  along with this program.  If not, see <https://www.gnu.org/licenses/>.
 | ||||
|  */ | ||||
| 
 | ||||
| use crate::election::{Candidate, CandidateState, CountCard, Election}; | ||||
| use crate::election::{Candidate, CandidateState, CountCard, CountState, Election}; | ||||
| use crate::numbers::Number; | ||||
| use crate::stv::{ConstraintMode, STVOptions}; | ||||
| 
 | ||||
| use itertools::Itertools; | ||||
| use ndarray::{Array, Dimension, IxDyn}; | ||||
| 
 | ||||
| use std::collections::HashMap; | ||||
| @ -462,6 +464,106 @@ impl ops::IndexMut<&[usize]> for ConstraintMatrix { | ||||
| 	fn index_mut(&mut self, index: &[usize]) -> &mut Self::Output { &mut self.0[index] } | ||||
| } | ||||
| 
 | ||||
| /// Return the [Candidate]s referred to in the given [ConstraintMatrixCell] at location `idx`
 | ||||
| fn candidates_in_constraint_cell<'a, N: Number>(election: &'a Election<N>, candidates: &HashMap<&Candidate, CountCard<N>>, idx: &[usize]) -> Vec<&'a Candidate> { | ||||
| 	let mut result: Vec<&Candidate> = Vec::new(); | ||||
| 	for (i, candidate) in election.candidates.iter().enumerate() { | ||||
| 		let cc = candidates.get(candidate).unwrap(); | ||||
| 		if cc.state != CandidateState::Hopeful { | ||||
| 			continue; | ||||
| 		} | ||||
| 		
 | ||||
| 		// Is this candidate within this constraint cell?
 | ||||
| 		let mut matches = true; | ||||
| 		for (coord, constraint) in idx.iter().zip(election.constraints.as_ref().unwrap().0.iter()) { | ||||
| 			let group = &constraint.groups[coord - 1]; // The group referred to by this constraint cell
 | ||||
| 			if !group.candidates.contains(&i) { | ||||
| 				matches = false; | ||||
| 				break; | ||||
| 			} | ||||
| 		} | ||||
| 		
 | ||||
| 		if matches { | ||||
| 			result.push(candidate); | ||||
| 		} | ||||
| 	} | ||||
| 	
 | ||||
| 	return result; | ||||
| } | ||||
| 
 | ||||
| /// Update the constraints matrix, and perform the necessary actions given by [STVOptions::constraint_mode]
 | ||||
| pub fn update_constraints<N: Number>(state: &mut CountState<N>, opts: &STVOptions) -> bool { | ||||
| 	if state.constraint_matrix.is_none() { | ||||
| 		return false; | ||||
| 	} | ||||
| 	let cm = state.constraint_matrix.as_mut().unwrap(); | ||||
| 	
 | ||||
| 	// Update cands/elected
 | ||||
| 	cm.update_from_state(&state.election, &state.candidates); | ||||
| 	cm.recount_cands(); | ||||
| 	
 | ||||
| 	// Iterate for stable state
 | ||||
| 	//println!("{}", cm);
 | ||||
| 	while !cm.step().expect("No conformant result is possible") { | ||||
| 		//println!("{}", cm);
 | ||||
| 	} | ||||
| 	//println!("{}", cm);
 | ||||
| 	
 | ||||
| 	match opts.constraint_mode { | ||||
| 		ConstraintMode::GuardDoom => { | ||||
| 			// Check for guarded or doomed candidates
 | ||||
| 			let mut guarded_or_doomed = false; | ||||
| 			
 | ||||
| 			for idx in ndarray::indices(cm.0.shape()) { | ||||
| 				if (0..idx.ndim()).fold(0, |acc, d| if idx[d] == 0 { acc + 1 } else { acc }) != 0 { | ||||
| 					continue; | ||||
| 				} | ||||
| 				let cell = &cm.0[&idx]; | ||||
| 				
 | ||||
| 				if cell.elected == cell.max { | ||||
| 					// Doom remaining candidates in this cell
 | ||||
| 					let doomed = candidates_in_constraint_cell(state.election, &state.candidates, idx.slice()); | ||||
| 					if !doomed.is_empty() { | ||||
| 						for candidate in doomed.iter() { | ||||
| 							state.candidates.get_mut(candidate).unwrap().state = CandidateState::Doomed; | ||||
| 						} | ||||
| 						
 | ||||
| 						state.logger.log_smart( | ||||
| 							"{} must be doomed to comply with constraints.", | ||||
| 							"{} must be doomed to comply with constraints.", | ||||
| 							doomed.iter().map(|c| c.name.as_str()).sorted().collect() | ||||
| 						); | ||||
| 						
 | ||||
| 						guarded_or_doomed = true; | ||||
| 					} | ||||
| 				} | ||||
| 				if cell.cands == cell.min { | ||||
| 					// Guard remaining candidates in this cell
 | ||||
| 					let guarded = candidates_in_constraint_cell(state.election, &state.candidates, idx.slice()); | ||||
| 					if !guarded.is_empty() { | ||||
| 						for candidate in guarded.iter() { | ||||
| 							state.candidates.get_mut(candidate).unwrap().state = CandidateState::Guarded; | ||||
| 						} | ||||
| 						
 | ||||
| 						state.logger.log_smart( | ||||
| 							"{} must be guarded to comply with constraints.", | ||||
| 							"{} must be guarded to comply with constraints.", | ||||
| 							guarded.iter().map(|c| c.name.as_str()).sorted().collect() | ||||
| 						); | ||||
| 						
 | ||||
| 						guarded_or_doomed = true; | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			
 | ||||
| 			return guarded_or_doomed; | ||||
| 		} | ||||
| 		_ => { todo!() } | ||||
| 	} | ||||
| 	
 | ||||
| 	//return false;
 | ||||
| } | ||||
| 
 | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
| 	use super::*; | ||||
|  | ||||
| @ -284,7 +284,7 @@ where | ||||
| 	let mut state = CountState::new(&election); | ||||
| 	
 | ||||
| 	// Distribute first preferences
 | ||||
| 	stv::count_init(&mut state, &stv_opts); | ||||
| 	stv::count_init(&mut state, &stv_opts).unwrap(); | ||||
| 	let mut stage_num = 1; | ||||
| 	make_and_print_result(stage_num, &state, &cmd_opts); | ||||
| 	
 | ||||
|  | ||||
| @ -17,6 +17,7 @@ | ||||
| 
 | ||||
| use super::{ExclusionMethod, NextPreferencesEntry, NextPreferencesResult, STVError, STVOptions, SumSurplusTransfersMode, SurplusMethod, SurplusOrder}; | ||||
| 
 | ||||
| use crate::constraints; | ||||
| use crate::election::{Candidate, CandidateState, CountCard, CountState, Parcel, Vote}; | ||||
| use crate::numbers::Number; | ||||
| 
 | ||||
| @ -361,7 +362,7 @@ where | ||||
| 			state.num_excluded += 1; | ||||
| 			count_card.order_elected = -(order_excluded as isize); | ||||
| 			
 | ||||
| 			super::update_constraints(state, opts); | ||||
| 			constraints::update_constraints(state, opts); | ||||
| 		} | ||||
| 	} | ||||
| 	
 | ||||
| @ -521,7 +522,7 @@ where | ||||
| 			count_card.order_elected = -(order_excluded as isize); | ||||
| 		} | ||||
| 		
 | ||||
| 		super::update_constraints(state, opts); | ||||
| 		constraints::update_constraints(state, opts); | ||||
| 	} | ||||
| 	
 | ||||
| 	// Reset count
 | ||||
|  | ||||
| @ -281,7 +281,7 @@ where | ||||
| 			
 | ||||
| 			if opts.meek_immediate_elect { | ||||
| 				// Try to elect candidates
 | ||||
| 				if super::elect_meeting_quota(state, opts) { | ||||
| 				if super::elect_meeting_quota(state, opts)? { | ||||
| 					candidates_elected = Some(state.logger.entries.pop().unwrap()); | ||||
| 					break; | ||||
| 				} | ||||
|  | ||||
							
								
								
									
										216
									
								
								src/stv/mod.rs
									
									
									
									
									
								
							
							
						
						
									
										216
									
								
								src/stv/mod.rs
									
									
									
									
									
								
							| @ -26,13 +26,13 @@ pub mod meek; | ||||
| //#[cfg(target_arch = "wasm32")]
 | ||||
| pub mod wasm; | ||||
| 
 | ||||
| use crate::constraints; | ||||
| use crate::numbers::Number; | ||||
| use crate::election::{Candidate, CandidateState, CountCard, CountState, Election, Vote}; | ||||
| use crate::election::{Candidate, CandidateState, CountCard, CountState, Vote}; | ||||
| use crate::sharandom::SHARandom; | ||||
| use crate::ties::TieStrategy; | ||||
| 
 | ||||
| use itertools::Itertools; | ||||
| use ndarray::Dimension; | ||||
| use wasm_bindgen::prelude::wasm_bindgen; | ||||
| 
 | ||||
| use std::collections::HashMap; | ||||
| @ -432,8 +432,18 @@ pub enum STVError { | ||||
| 	UnresolvedTie, | ||||
| } | ||||
| 
 | ||||
| impl STVError { | ||||
| 	/// Return the name of the error as a string
 | ||||
| 	pub fn name(&self) -> &'static str { | ||||
| 		match self { | ||||
| 			STVError::RequireInput => "RequireInput", | ||||
| 			STVError::UnresolvedTie => "UnresolvedTie", | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /// Distribute first preferences, and initialise other states such as the random number generator and tie-breaking rules
 | ||||
| pub fn count_init<'a, N: Number>(mut state: &mut CountState<'a, N>, opts: &'a STVOptions) | ||||
| pub fn count_init<'a, N: Number>(state: &mut CountState<'a, N>, opts: &'a STVOptions) -> Result<bool, STVError> | ||||
| where | ||||
| 	for<'r> &'r N: ops::Sub<&'r N, Output=N>, | ||||
| 	for<'r> &'r N: ops::Mul<&'r N, Output=N>, | ||||
| @ -445,14 +455,18 @@ where | ||||
| 		} | ||||
| 	} | ||||
| 	
 | ||||
| 	distribute_first_preferences(&mut state, opts); | ||||
| 	calculate_quota(&mut state, opts); | ||||
| 	elect_meeting_quota(&mut state, opts); | ||||
| 	init_tiebreaks(&mut state, opts); | ||||
| 	constraints::update_constraints(state, opts); | ||||
| 	
 | ||||
| 	distribute_first_preferences(state, opts); | ||||
| 	calculate_quota(state, opts); | ||||
| 	elect_meeting_quota(state, opts)?; | ||||
| 	init_tiebreaks(state, opts); | ||||
| 	
 | ||||
| 	return Ok(true); | ||||
| } | ||||
| 
 | ||||
| /// Perform a single stage of the STV count
 | ||||
| pub fn count_one_stage<'a, N: Number>(mut state: &mut CountState<'a, N>, opts: &STVOptions) -> Result<bool, STVError> | ||||
| pub fn count_one_stage<'a, N: Number>(state: &mut CountState<'a, N>, opts: &STVOptions) -> Result<bool, STVError> | ||||
| where | ||||
| 	for<'r> &'r N: ops::Sub<&'r N, Output=N>, | ||||
| 	for<'r> &'r N: ops::Mul<&'r N, Output=N>, | ||||
| @ -469,45 +483,45 @@ where | ||||
| 	
 | ||||
| 	// Attempt early bulk election
 | ||||
| 	if opts.early_bulk_elect { | ||||
| 		if bulk_elect(&mut state, &opts)? { | ||||
| 		if bulk_elect(state, &opts)? { | ||||
| 			return Ok(false); | ||||
| 		} | ||||
| 	} | ||||
| 	
 | ||||
| 	// Continue exclusions
 | ||||
| 	if continue_exclusion(&mut state, &opts) { | ||||
| 		calculate_quota(&mut state, opts); | ||||
| 		elect_meeting_quota(&mut state, opts); | ||||
| 		update_tiebreaks(&mut state, opts); | ||||
| 	if continue_exclusion(state, &opts) { | ||||
| 		calculate_quota(state, opts); | ||||
| 		elect_meeting_quota(state, opts)?; | ||||
| 		update_tiebreaks(state, opts); | ||||
| 		return Ok(false); | ||||
| 	} | ||||
| 	
 | ||||
| 	// Exclude doomed candidates
 | ||||
| 	if exclude_doomed(&mut state, &opts)? { | ||||
| 		calculate_quota(&mut state, opts); | ||||
| 		elect_meeting_quota(&mut state, opts); | ||||
| 		update_tiebreaks(&mut state, opts); | ||||
| 	if exclude_doomed(state, &opts)? { | ||||
| 		calculate_quota(state, opts); | ||||
| 		elect_meeting_quota(state, opts)?; | ||||
| 		update_tiebreaks(state, opts); | ||||
| 		return Ok(false); | ||||
| 	} | ||||
| 	
 | ||||
| 	// Distribute surpluses
 | ||||
| 	if distribute_surpluses(&mut state, &opts)? { | ||||
| 		calculate_quota(&mut state, opts); | ||||
| 		elect_meeting_quota(&mut state, opts); | ||||
| 		update_tiebreaks(&mut state, opts); | ||||
| 	if distribute_surpluses(state, &opts)? { | ||||
| 		calculate_quota(state, opts); | ||||
| 		elect_meeting_quota(state, opts)?; | ||||
| 		update_tiebreaks(state, opts); | ||||
| 		return Ok(false); | ||||
| 	} | ||||
| 	
 | ||||
| 	// Attempt late bulk election
 | ||||
| 	if bulk_elect(&mut state, &opts)? { | ||||
| 	if bulk_elect(state, &opts)? { | ||||
| 		return Ok(false); | ||||
| 	} | ||||
| 	
 | ||||
| 	// Exclude lowest hopeful
 | ||||
| 	if exclude_hopefuls(&mut state, &opts)? { | ||||
| 		calculate_quota(&mut state, opts); | ||||
| 		elect_meeting_quota(&mut state, opts); | ||||
| 		update_tiebreaks(&mut state, opts); | ||||
| 	if exclude_hopefuls(state, &opts)? { | ||||
| 		calculate_quota(state, opts); | ||||
| 		elect_meeting_quota(state, opts)?; | ||||
| 		update_tiebreaks(state, opts); | ||||
| 		return Ok(false); | ||||
| 	} | ||||
| 	
 | ||||
| @ -728,10 +742,10 @@ fn meets_quota<N: Number>(quota: &N, count_card: &CountCard<N>, opts: &STVOption | ||||
| } | ||||
| 
 | ||||
| /// Declare elected all candidates meeting the quota
 | ||||
| fn elect_meeting_quota<N: Number>(state: &mut CountState<N>, opts: &STVOptions) -> bool { | ||||
| fn elect_meeting_quota<N: Number>(state: &mut CountState<N>, opts: &STVOptions) -> Result<bool, STVError> { | ||||
| 	let vote_req = state.vote_required_election.as_ref().unwrap().clone(); // Have to do this or else the borrow checker gets confused
 | ||||
| 	
 | ||||
| 	let mut cands_meeting_quota: Vec<&Candidate> = state.election.candidates.iter() | ||||
| 	let mut cands_meeting_quota: Vec<&Candidate> = state.election.candidates.iter() // Present in order in case of tie
 | ||||
| 		.filter(|c| { | ||||
| 			let cc = state.candidates.get(c).unwrap(); | ||||
| 			return (cc.state == CandidateState::Hopeful || cc.state == CandidateState::Guarded) && meets_quota(&vote_req, cc, opts); | ||||
| @ -745,7 +759,22 @@ fn elect_meeting_quota<N: Number>(state: &mut CountState<N>, opts: &STVOptions) | ||||
| 	
 | ||||
| 	while !cands_meeting_quota.is_empty() { | ||||
| 		// Declare elected in descending order of votes
 | ||||
| 		let candidate = cands_meeting_quota.pop().unwrap(); | ||||
| 		let max_votes = cands_meeting_quota.iter() | ||||
| 			.max_by(|a, b| state.candidates.get(**a).unwrap().votes.cmp(&state.candidates.get(**b).unwrap().votes)) | ||||
| 			.unwrap(); | ||||
| 		let max_votes = &state.candidates.get(max_votes).unwrap().votes; | ||||
| 		let max_cands_meeting_quota: Vec<&Candidate> = cands_meeting_quota.iter() | ||||
| 			.filter(|c| &state.candidates.get(**c).unwrap().votes == max_votes) | ||||
| 			.map(|c| *c) | ||||
| 			.collect(); | ||||
| 		
 | ||||
| 		let candidate; | ||||
| 		if max_cands_meeting_quota.len() > 1 { | ||||
| 			// Handle ties
 | ||||
| 			candidate = choose_highest(state, opts, max_cands_meeting_quota)?; | ||||
| 		} else { | ||||
| 			candidate = max_cands_meeting_quota[0]; | ||||
| 		} | ||||
| 		
 | ||||
| 		let count_card = state.candidates.get_mut(candidate).unwrap(); | ||||
| 		count_card.state = CandidateState::Elected; | ||||
| @ -757,12 +786,14 @@ fn elect_meeting_quota<N: Number>(state: &mut CountState<N>, opts: &STVOptions) | ||||
| 			vec![&candidate.name] | ||||
| 		); | ||||
| 		
 | ||||
| 		if update_constraints(state, opts) { | ||||
| 		if constraints::update_constraints(state, opts) { | ||||
| 			// Recheck as some candidates may have been doomed
 | ||||
| 			cands_meeting_quota = state.election.candidates.iter() | ||||
| 				.filter(|c| { let cc = state.candidates.get(c).unwrap(); cc.state == CandidateState::Hopeful && meets_quota(&vote_req, cc, opts) }) | ||||
| 				.collect(); | ||||
| 			cands_meeting_quota.sort_unstable_by(|a, b| state.candidates.get(a).unwrap().votes.cmp(&state.candidates.get(b).unwrap().votes)); | ||||
| 		} else { | ||||
| 			cands_meeting_quota.remove(cands_meeting_quota.iter().position(|c| *c == candidate).unwrap()); | ||||
| 		} | ||||
| 		
 | ||||
| 		if opts.quota_mode == QuotaMode::ERS97 { | ||||
| @ -772,110 +803,14 @@ fn elect_meeting_quota<N: Number>(state: &mut CountState<N>, opts: &STVOptions) | ||||
| 		
 | ||||
| 		if opts.quota_mode == QuotaMode::ERS97 { | ||||
| 			// Repeat in case vote required for election has changed
 | ||||
| 			elect_meeting_quota(state, opts); | ||||
| 		} | ||||
| 	} | ||||
| 	
 | ||||
| 	return elected; | ||||
| } | ||||
| 
 | ||||
| fn candidates_in_constraint_cell<'a, N: Number>(election: &'a Election<N>, candidates: &HashMap<&Candidate, CountCard<N>>, idx: &[usize]) -> Vec<&'a Candidate> { | ||||
| 	let mut result: Vec<&Candidate> = Vec::new(); | ||||
| 	for (i, candidate) in election.candidates.iter().enumerate() { | ||||
| 		let cc = candidates.get(candidate).unwrap(); | ||||
| 		if cc.state != CandidateState::Hopeful { | ||||
| 			continue; | ||||
| 		} | ||||
| 		
 | ||||
| 		// Is this candidate within this constraint cell?
 | ||||
| 		let mut matches = true; | ||||
| 		for (coord, constraint) in idx.iter().zip(election.constraints.as_ref().unwrap().0.iter()) { | ||||
| 			let group = &constraint.groups[coord - 1]; // The group referred to by this constraint cell
 | ||||
| 			if !group.candidates.contains(&i) { | ||||
| 				matches = false; | ||||
| 				break; | ||||
| 			match elect_meeting_quota(state, opts) { | ||||
| 				Ok(_) => {} | ||||
| 				Err(e) => { return Err(e); } | ||||
| 			} | ||||
| 		} | ||||
| 		
 | ||||
| 		if matches { | ||||
| 			result.push(candidate); | ||||
| 		} | ||||
| 	} | ||||
| 	
 | ||||
| 	return result; | ||||
| } | ||||
| 
 | ||||
| fn update_constraints<N: Number>(state: &mut CountState<N>, opts: &STVOptions) -> bool { | ||||
| 	if state.constraint_matrix.is_none() { | ||||
| 		return false; | ||||
| 	} | ||||
| 	let cm = state.constraint_matrix.as_mut().unwrap(); | ||||
| 	
 | ||||
| 	// Update cands/elected
 | ||||
| 	cm.update_from_state(&state.election, &state.candidates); | ||||
| 	cm.recount_cands(); | ||||
| 	
 | ||||
| 	// Iterate for stable state
 | ||||
| 	//println!("{}", cm);
 | ||||
| 	while !cm.step().expect("No conformant result") { | ||||
| 		//println!("{}", cm);
 | ||||
| 	} | ||||
| 	//println!("{}", cm);
 | ||||
| 	
 | ||||
| 	// TODO: Refactor and move this to constraints module?
 | ||||
| 	match opts.constraint_mode { | ||||
| 		ConstraintMode::GuardDoom => { | ||||
| 			// Check for guarded or doomed candidates
 | ||||
| 			let mut guarded_or_doomed = false; | ||||
| 			
 | ||||
| 			for idx in ndarray::indices(cm.0.shape()) { | ||||
| 				if (0..idx.ndim()).fold(0, |acc, d| if idx[d] == 0 { acc + 1 } else { acc }) != 0 { | ||||
| 					continue; | ||||
| 				} | ||||
| 				let cell = &cm.0[&idx]; | ||||
| 				
 | ||||
| 				if cell.elected == cell.max { | ||||
| 					// Doom remaining candidates in this cell
 | ||||
| 					let doomed = candidates_in_constraint_cell(state.election, &state.candidates, idx.slice()); | ||||
| 					if !doomed.is_empty() { | ||||
| 						for candidate in doomed.iter() { | ||||
| 							state.candidates.get_mut(candidate).unwrap().state = CandidateState::Doomed; | ||||
| 						} | ||||
| 						
 | ||||
| 						state.logger.log_smart( | ||||
| 							"{} must be doomed to comply with constraints.", | ||||
| 							"{} must be doomed to comply with constraints.", | ||||
| 							doomed.iter().map(|c| c.name.as_str()).collect() | ||||
| 						); | ||||
| 						
 | ||||
| 						guarded_or_doomed = true; | ||||
| 					} | ||||
| 				} | ||||
| 				if cell.cands == cell.min { | ||||
| 					// Guard remaining candidates in this cell
 | ||||
| 					let guarded = candidates_in_constraint_cell(state.election, &state.candidates, idx.slice()); | ||||
| 					if !guarded.is_empty() { | ||||
| 						for candidate in guarded.iter() { | ||||
| 							state.candidates.get_mut(candidate).unwrap().state = CandidateState::Guarded; | ||||
| 						} | ||||
| 						
 | ||||
| 						state.logger.log_smart( | ||||
| 							"{} must be guarded to comply with constraints.", | ||||
| 							"{} must be guarded to comply with constraints.", | ||||
| 							guarded.iter().map(|c| c.name.as_str()).collect() | ||||
| 						); | ||||
| 						
 | ||||
| 						guarded_or_doomed = true; | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			
 | ||||
| 			return guarded_or_doomed; | ||||
| 		} | ||||
| 		_ => { todo!() } | ||||
| 	} | ||||
| 	
 | ||||
| 	//return false;
 | ||||
| 	return Ok(elected); | ||||
| } | ||||
| 
 | ||||
| /// Determine whether the transfer of all surpluses can be deferred
 | ||||
| @ -970,9 +905,17 @@ fn bulk_elect<N: Number>(state: &mut CountState<N>, opts: &STVOptions) -> Result | ||||
| 				vec![&candidate.name] | ||||
| 			); | ||||
| 			
 | ||||
| 			hopefuls.remove(hopefuls.iter().position(|c| *c == candidate).unwrap()); | ||||
| 			
 | ||||
| 			update_constraints(state, opts); | ||||
| 			if constraints::update_constraints(state, opts) { | ||||
| 				// Recheck as some candidates may have been doomed
 | ||||
| 				hopefuls = state.election.candidates.iter() | ||||
| 					.filter(|c| { | ||||
| 						let cc = state.candidates.get(c).unwrap(); | ||||
| 						return cc.state == CandidateState::Hopeful || cc.state == CandidateState::Guarded; | ||||
| 					}) | ||||
| 					.collect(); | ||||
| 			} else { | ||||
| 				hopefuls.remove(hopefuls.iter().position(|c| *c == candidate).unwrap()); | ||||
| 			} | ||||
| 		} | ||||
| 		
 | ||||
| 		return Ok(true); | ||||
| @ -1011,7 +954,7 @@ where | ||||
| 			} | ||||
| 		} | ||||
| 		
 | ||||
| 		let names: Vec<&str> = excluded_candidates.iter().map(|c| c.name.as_str()).collect(); | ||||
| 		let names: Vec<&str> = excluded_candidates.iter().map(|c| c.name.as_str()).sorted().collect(); | ||||
| 		state.kind = Some("Exclusion of"); | ||||
| 		state.title = names.join(", "); | ||||
| 		state.logger.log_smart( | ||||
| @ -1103,7 +1046,7 @@ where | ||||
| 		} | ||||
| 	} | ||||
| 	
 | ||||
| 	let names: Vec<&str> = excluded_candidates.iter().map(|c| c.name.as_str()).collect(); | ||||
| 	let names: Vec<&str> = excluded_candidates.iter().map(|c| c.name.as_str()).sorted().collect(); | ||||
| 	state.kind = Some("Exclusion of"); | ||||
| 	state.title = names.join(", "); | ||||
| 	state.logger.log_smart( | ||||
| @ -1138,8 +1081,7 @@ where | ||||
| 			.map(|(c, _)| *c) | ||||
| 			.collect(); | ||||
| 		
 | ||||
| 		let mut names: Vec<&str> = excluded_candidates.iter().map(|c| c.name.as_str()).collect(); | ||||
| 		names.sort(); | ||||
| 		let names: Vec<&str> = excluded_candidates.iter().map(|c| c.name.as_str()).sorted().collect(); | ||||
| 		state.kind = Some("Exclusion of"); | ||||
| 		state.title = names.join(", "); | ||||
| 		state.logger.log_smart( | ||||
|  | ||||
| @ -77,8 +77,11 @@ macro_rules! impl_type { | ||||
| 		/// Wrapper for [stv::count_init]
 | ||||
| 		#[wasm_bindgen] | ||||
| 		#[allow(non_snake_case)] | ||||
| 		pub fn [<count_init_$type>](state: &mut [<CountState$type>], opts: &STVOptions) { | ||||
| 			stv::count_init(&mut state.0, opts.as_static()); | ||||
| 		pub fn [<count_init_$type>](state: &mut [<CountState$type>], opts: &STVOptions) -> Result<bool, JsValue> { | ||||
| 			match stv::count_init(&mut state.0, opts.as_static()) { | ||||
| 				Ok(v) => Ok(v), | ||||
| 				Err(e) => Err(e.name().into()), | ||||
| 			} | ||||
| 		} | ||||
| 		
 | ||||
| 		/// Wrapper for [stv::count_one_stage]
 | ||||
| @ -87,8 +90,7 @@ macro_rules! impl_type { | ||||
| 		pub fn [<count_one_stage_$type>](state: &mut [<CountState$type>], opts: &STVOptions) -> Result<bool, JsValue> { | ||||
| 			match stv::count_one_stage::<[<$type>]>(&mut state.0, &opts.0) { | ||||
| 				Ok(v) => Ok(v), | ||||
| 				Err(stv::STVError::RequireInput) => Err("RequireInput".into()), | ||||
| 				Err(stv::STVError::UnresolvedTie) => Err("UnresolvedTie".into()), | ||||
| 				Err(e) => Err(e.name().into()), | ||||
| 			} | ||||
| 		} | ||||
| 		
 | ||||
|  | ||||
| @ -71,7 +71,7 @@ fn prsa1_constr1_rational() { | ||||
| 	let mut state = CountState::new(&election); | ||||
| 	
 | ||||
| 	// Count election
 | ||||
| 	stv::count_init(&mut state, &stv_opts); | ||||
| 	stv::count_init(&mut state, &stv_opts).unwrap(); | ||||
| 	while stv::count_one_stage::<Rational>(&mut state, &stv_opts).unwrap() == false {} | ||||
| 	
 | ||||
| 	// Validate winners
 | ||||
| @ -135,7 +135,7 @@ fn prsa1_constr2_rational() { | ||||
| 	let mut state = CountState::new(&election); | ||||
| 	
 | ||||
| 	// Count election
 | ||||
| 	stv::count_init(&mut state, &stv_opts); | ||||
| 	stv::count_init(&mut state, &stv_opts).unwrap(); | ||||
| 	while stv::count_one_stage::<Rational>(&mut state, &stv_opts).unwrap() == false {} | ||||
| 	
 | ||||
| 	// Validate winners
 | ||||
| @ -199,7 +199,7 @@ fn prsa1_constr3_rational() { | ||||
| 	let mut state = CountState::new(&election); | ||||
| 	
 | ||||
| 	// Count election
 | ||||
| 	stv::count_init(&mut state, &stv_opts); | ||||
| 	stv::count_init(&mut state, &stv_opts).unwrap(); | ||||
| 	while stv::count_one_stage::<Rational>(&mut state, &stv_opts).unwrap() == false {} | ||||
| 	
 | ||||
| 	// Validate winners
 | ||||
|  | ||||
| @ -96,7 +96,7 @@ fn meek06_ers97_fixed12() { | ||||
| 	let mut state = CountState::new(&election); | ||||
| 	
 | ||||
| 	// Count to completion
 | ||||
| 	stv::count_init(&mut state, &stv_opts); | ||||
| 	stv::count_init(&mut state, &stv_opts).unwrap(); | ||||
| 	while !stv::count_one_stage::<Fixed>(&mut state, &stv_opts).unwrap() {} | ||||
| 	
 | ||||
| 	// Check states and keep values
 | ||||
| @ -172,7 +172,7 @@ fn meeknz_ers97_fixed12() { | ||||
| 	let mut state = CountState::new(&election); | ||||
| 	
 | ||||
| 	// Count to completion
 | ||||
| 	stv::count_init(&mut state, &stv_opts); | ||||
| 	stv::count_init(&mut state, &stv_opts).unwrap(); | ||||
| 	while !stv::count_one_stage::<Fixed>(&mut state, &stv_opts).unwrap() {} | ||||
| 	
 | ||||
| 	// Check states and keep values
 | ||||
|  | ||||
| @ -127,7 +127,7 @@ where | ||||
| 	let mut state = CountState::new(&election); | ||||
| 	
 | ||||
| 	// Distribute first preferences
 | ||||
| 	stv::count_init(&mut state, &stv_opts); | ||||
| 	stv::count_init(&mut state, &stv_opts).unwrap(); | ||||
| 	let mut stage_num = 1; | ||||
| 	
 | ||||
| 	for i in 0..num_stages { | ||||
|  | ||||
| @ -78,7 +78,7 @@ where | ||||
| 	let mut state = CountState::new(&election); | ||||
| 	
 | ||||
| 	// Distribute first preferences
 | ||||
| 	stv::count_init(&mut state, &stv_opts); | ||||
| 	stv::count_init(&mut state, &stv_opts).unwrap(); | ||||
| 	let mut stage_num = 1; | ||||
| 	
 | ||||
| 	for (idx, stage) in stages.into_iter().enumerate() { | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user