2026-02-20 11:01:25 +01:00
|
|
|
from typing import List, Dict, DefaultDict
|
2026-03-06 15:02:53 +01:00
|
|
|
from dataclasses import dataclass, field
|
2026-02-20 11:01:25 +01:00
|
|
|
from collections import defaultdict
|
2026-02-17 16:37:36 +01:00
|
|
|
|
2026-03-06 15:02:53 +01:00
|
|
|
from warchron.constants import ContextType, ScoreKind
|
2026-02-17 16:37:36 +01:00
|
|
|
from warchron.model.exception import ForbiddenOperation
|
2026-02-11 19:22:43 +01:00
|
|
|
from warchron.model.war import War
|
2026-02-17 16:37:36 +01:00
|
|
|
from warchron.model.war_event import InfluenceSpent, TieResolved
|
2026-03-03 11:52:07 +01:00
|
|
|
from warchron.model.score_service import ScoreService, ParticipantScore
|
2026-02-20 11:01:25 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
|
class TieContext:
|
|
|
|
|
context_type: ContextType
|
|
|
|
|
context_id: str
|
2026-03-06 15:02:53 +01:00
|
|
|
participants: List[str] = field(default_factory=list) # war_participant_ids
|
2026-02-25 16:54:21 +01:00
|
|
|
score_value: int | None = None
|
2026-03-06 15:02:53 +01:00
|
|
|
score_kind: ScoreKind | None = None
|
|
|
|
|
objective_id: str | None = None
|
|
|
|
|
|
|
|
|
|
def key(self) -> tuple[str, str, int | None]:
|
|
|
|
|
return (self.context_type, self.context_id, self.score_value)
|
2026-02-11 19:22:43 +01:00
|
|
|
|
|
|
|
|
|
2026-02-17 16:37:36 +01:00
|
|
|
class TieResolver:
|
2026-02-11 19:22:43 +01:00
|
|
|
|
2026-03-11 11:44:57 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def find_choice_ties(
|
|
|
|
|
war: War,
|
|
|
|
|
round_id: str,
|
|
|
|
|
) -> List[TieContext]:
|
|
|
|
|
round = war.get_round(round_id)
|
|
|
|
|
campaign = war.get_campaign_by_round(round_id)
|
|
|
|
|
if campaign is None:
|
|
|
|
|
raise RuntimeError("Round without campaign")
|
|
|
|
|
ties: List[TieContext] = []
|
|
|
|
|
scores = ScoreService.compute_scores(
|
|
|
|
|
war,
|
|
|
|
|
ContextType.CAMPAIGN,
|
|
|
|
|
campaign.id,
|
|
|
|
|
)
|
|
|
|
|
score_groups = ScoreService.group_participants_by_score(
|
|
|
|
|
scores, lambda score: score.victory_points
|
|
|
|
|
)
|
|
|
|
|
sector_to_battle = {b.sector_id: b for b in round.battles.values()}
|
|
|
|
|
for group in score_groups:
|
|
|
|
|
if len(group) <= 1:
|
|
|
|
|
continue
|
|
|
|
|
demand: Dict[str, List[str]] = {}
|
|
|
|
|
for pid in group:
|
|
|
|
|
choice = round.choices.get(pid)
|
|
|
|
|
if not choice:
|
|
|
|
|
continue
|
|
|
|
|
for sec_id in (
|
|
|
|
|
choice.priority_sector_id,
|
|
|
|
|
choice.secondary_sector_id,
|
|
|
|
|
):
|
|
|
|
|
if sec_id:
|
|
|
|
|
demand.setdefault(sec_id, []).append(pid)
|
|
|
|
|
for sector_id, demanders in demand.items():
|
|
|
|
|
battle = sector_to_battle.get(sector_id)
|
|
|
|
|
if battle is None:
|
|
|
|
|
continue
|
|
|
|
|
places = len(battle.get_available_places())
|
|
|
|
|
if len(demanders) <= places:
|
|
|
|
|
continue
|
|
|
|
|
context = TieContext(
|
|
|
|
|
ContextType.CHOICE,
|
|
|
|
|
round_id,
|
|
|
|
|
demanders,
|
|
|
|
|
score_value=None,
|
|
|
|
|
score_kind=ScoreKind.VP,
|
|
|
|
|
)
|
|
|
|
|
if TieResolver.is_tie_resolved(war, context):
|
|
|
|
|
continue
|
|
|
|
|
if not TieResolver.can_tie_be_resolved(
|
|
|
|
|
war,
|
|
|
|
|
context,
|
|
|
|
|
demanders,
|
|
|
|
|
):
|
|
|
|
|
war.events.append(
|
|
|
|
|
TieResolved(
|
|
|
|
|
None,
|
|
|
|
|
ContextType.CHOICE,
|
|
|
|
|
round_id,
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
continue
|
|
|
|
|
ties.append(context)
|
|
|
|
|
return ties
|
|
|
|
|
|
2026-02-17 16:37:36 +01:00
|
|
|
@staticmethod
|
2026-02-20 11:01:25 +01:00
|
|
|
def find_battle_ties(war: War, round_id: str) -> List[TieContext]:
|
|
|
|
|
round = war.get_round(round_id)
|
|
|
|
|
campaign = war.get_campaign_by_round(round_id)
|
2026-02-17 16:37:36 +01:00
|
|
|
ties = []
|
|
|
|
|
for battle in round.battles.values():
|
|
|
|
|
if not battle.is_draw():
|
|
|
|
|
continue
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext = TieContext(
|
|
|
|
|
ContextType.BATTLE,
|
|
|
|
|
battle.sector_id,
|
|
|
|
|
)
|
|
|
|
|
if TieResolver.is_tie_resolved(war, context):
|
2026-02-20 11:01:25 +01:00
|
|
|
continue
|
|
|
|
|
if campaign is None:
|
|
|
|
|
raise RuntimeError("No campaign for this battle tie")
|
|
|
|
|
if battle.player_1_id is None or battle.player_2_id is None:
|
|
|
|
|
raise RuntimeError("Missing player(s) in this battle context.")
|
|
|
|
|
p1 = campaign.participants[battle.player_1_id].war_participant_id
|
|
|
|
|
p2 = campaign.participants[battle.player_2_id].war_participant_id
|
2026-03-06 15:02:53 +01:00
|
|
|
if not TieResolver.can_tie_be_resolved(war, context, [p1, p2]):
|
2026-02-20 23:44:22 +01:00
|
|
|
war.events.append(
|
|
|
|
|
TieResolved(None, ContextType.BATTLE, battle.sector_id)
|
|
|
|
|
)
|
|
|
|
|
continue
|
2026-02-20 11:01:25 +01:00
|
|
|
ties.append(
|
|
|
|
|
TieContext(
|
|
|
|
|
context_type=ContextType.BATTLE,
|
|
|
|
|
context_id=battle.sector_id,
|
|
|
|
|
participants=[p1, p2],
|
2026-02-25 16:54:21 +01:00
|
|
|
score_value=None,
|
2026-03-06 15:02:53 +01:00
|
|
|
score_kind=None,
|
2026-02-20 11:01:25 +01:00
|
|
|
)
|
|
|
|
|
)
|
2026-02-17 16:37:36 +01:00
|
|
|
return ties
|
2026-02-11 19:22:43 +01:00
|
|
|
|
2026-02-20 11:01:25 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def find_campaign_ties(war: War, campaign_id: str) -> List[TieContext]:
|
|
|
|
|
scores = ScoreService.compute_scores(war, ContextType.CAMPAIGN, campaign_id)
|
|
|
|
|
buckets: DefaultDict[int, List[str]] = defaultdict(list)
|
|
|
|
|
for pid, score in scores.items():
|
|
|
|
|
buckets[score.victory_points].append(pid)
|
2026-02-23 11:37:50 +01:00
|
|
|
ties: List[TieContext] = []
|
|
|
|
|
for score_value, participants in buckets.items():
|
|
|
|
|
if len(participants) <= 1:
|
|
|
|
|
continue
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext = TieContext(
|
|
|
|
|
ContextType.CAMPAIGN,
|
|
|
|
|
campaign_id,
|
|
|
|
|
[],
|
|
|
|
|
score_value,
|
|
|
|
|
ScoreKind.VP,
|
|
|
|
|
)
|
|
|
|
|
if TieResolver.is_tie_resolved(war, context):
|
2026-02-23 11:37:50 +01:00
|
|
|
continue
|
2026-03-06 15:02:53 +01:00
|
|
|
if not TieResolver.can_tie_be_resolved(war, context, participants):
|
2026-02-25 16:54:21 +01:00
|
|
|
war.events.append(
|
2026-03-06 15:02:53 +01:00
|
|
|
TieResolved(
|
|
|
|
|
None,
|
|
|
|
|
ContextType.CAMPAIGN,
|
|
|
|
|
campaign_id,
|
|
|
|
|
score_value,
|
|
|
|
|
)
|
2026-02-25 16:54:21 +01:00
|
|
|
)
|
2026-02-23 11:37:50 +01:00
|
|
|
continue
|
|
|
|
|
ties.append(
|
|
|
|
|
TieContext(
|
|
|
|
|
context_type=ContextType.CAMPAIGN,
|
2026-02-25 16:54:21 +01:00
|
|
|
context_id=campaign_id,
|
2026-02-23 11:37:50 +01:00
|
|
|
participants=participants,
|
2026-02-25 16:54:21 +01:00
|
|
|
score_value=score_value,
|
2026-03-06 15:02:53 +01:00
|
|
|
score_kind=ScoreKind.VP,
|
2026-02-20 11:01:25 +01:00
|
|
|
)
|
2026-02-23 11:37:50 +01:00
|
|
|
)
|
2026-02-20 11:01:25 +01:00
|
|
|
return ties
|
|
|
|
|
|
2026-03-03 11:52:07 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def find_campaign_objective_ties(
|
|
|
|
|
war: War,
|
|
|
|
|
campaign_id: str,
|
|
|
|
|
objective_id: str,
|
|
|
|
|
) -> List[TieContext]:
|
2026-03-05 11:37:14 +01:00
|
|
|
scores = ScoreService.compute_scores(
|
2026-03-03 11:52:07 +01:00
|
|
|
war,
|
|
|
|
|
ContextType.CAMPAIGN,
|
|
|
|
|
campaign_id,
|
|
|
|
|
)
|
|
|
|
|
buckets: DefaultDict[int, List[str]] = defaultdict(list)
|
|
|
|
|
for pid, score in scores.items():
|
2026-03-05 11:37:14 +01:00
|
|
|
np_value = score.narrative_points.get(objective_id, 0)
|
|
|
|
|
buckets[np_value].append(pid)
|
2026-03-03 11:52:07 +01:00
|
|
|
ties: List[TieContext] = []
|
2026-03-06 15:02:53 +01:00
|
|
|
context_id = campaign_id
|
2026-03-05 11:37:14 +01:00
|
|
|
for np_value, participants in buckets.items():
|
2026-03-03 11:52:07 +01:00
|
|
|
if len(participants) <= 1:
|
|
|
|
|
continue
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext = TieContext(
|
|
|
|
|
ContextType.CAMPAIGN,
|
|
|
|
|
campaign_id,
|
|
|
|
|
[],
|
2026-03-05 11:37:14 +01:00
|
|
|
np_value,
|
2026-03-06 15:02:53 +01:00
|
|
|
ScoreKind.NP,
|
|
|
|
|
objective_id,
|
|
|
|
|
)
|
|
|
|
|
if TieResolver.is_tie_resolved(war, context):
|
2026-03-03 11:52:07 +01:00
|
|
|
continue
|
|
|
|
|
if not TieResolver.can_tie_be_resolved(
|
|
|
|
|
war,
|
2026-03-06 15:02:53 +01:00
|
|
|
context,
|
2026-03-03 11:52:07 +01:00
|
|
|
participants,
|
|
|
|
|
):
|
|
|
|
|
war.events.append(
|
|
|
|
|
TieResolved(
|
2026-03-06 15:02:53 +01:00
|
|
|
None, ContextType.CAMPAIGN, context_id, np_value, objective_id
|
2026-03-03 11:52:07 +01:00
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
continue
|
|
|
|
|
ties.append(
|
|
|
|
|
TieContext(
|
2026-03-06 15:02:53 +01:00
|
|
|
context_type=ContextType.CAMPAIGN,
|
2026-03-03 11:52:07 +01:00
|
|
|
context_id=context_id,
|
|
|
|
|
participants=participants,
|
2026-03-05 11:37:14 +01:00
|
|
|
score_value=np_value,
|
2026-03-06 15:02:53 +01:00
|
|
|
score_kind=ScoreKind.NP,
|
|
|
|
|
objective_id=objective_id,
|
2026-03-03 11:52:07 +01:00
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
return ties
|
|
|
|
|
|
2026-02-20 11:01:25 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def find_war_ties(war: War) -> List[TieContext]:
|
2026-02-26 15:14:44 +01:00
|
|
|
from warchron.model.result_checker import ResultChecker
|
|
|
|
|
|
2026-02-23 21:18:27 +01:00
|
|
|
scores = ScoreService.compute_scores(war, ContextType.WAR, war.id)
|
2026-02-26 15:14:44 +01:00
|
|
|
ranking = ResultChecker.get_effective_ranking(
|
2026-03-03 15:39:30 +01:00
|
|
|
war,
|
|
|
|
|
ContextType.WAR,
|
|
|
|
|
war.id,
|
2026-03-06 15:02:53 +01:00
|
|
|
ScoreKind.VP,
|
2026-03-03 15:39:30 +01:00
|
|
|
scores,
|
2026-03-06 15:02:53 +01:00
|
|
|
lambda s: s.victory_points,
|
2026-02-26 15:14:44 +01:00
|
|
|
)
|
2026-02-23 21:18:27 +01:00
|
|
|
ties: List[TieContext] = []
|
2026-02-26 15:14:44 +01:00
|
|
|
for _, group, _ in ranking:
|
|
|
|
|
if len(group) <= 1:
|
2026-02-23 21:18:27 +01:00
|
|
|
continue
|
2026-02-26 15:14:44 +01:00
|
|
|
score_value = scores[group[0]].victory_points
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext = TieContext(
|
|
|
|
|
ContextType.WAR,
|
|
|
|
|
war.id,
|
|
|
|
|
[],
|
|
|
|
|
score_value,
|
|
|
|
|
ScoreKind.VP,
|
|
|
|
|
)
|
|
|
|
|
if TieResolver.is_tie_resolved(war, context):
|
2026-02-23 21:18:27 +01:00
|
|
|
continue
|
2026-03-06 15:02:53 +01:00
|
|
|
if not TieResolver.can_tie_be_resolved(war, context, group):
|
2026-02-25 16:54:21 +01:00
|
|
|
war.events.append(
|
|
|
|
|
TieResolved(None, ContextType.WAR, war.id, score_value)
|
|
|
|
|
)
|
2026-02-23 21:18:27 +01:00
|
|
|
continue
|
|
|
|
|
ties.append(
|
|
|
|
|
TieContext(
|
|
|
|
|
context_type=ContextType.WAR,
|
2026-02-25 16:54:21 +01:00
|
|
|
context_id=war.id,
|
2026-02-26 15:14:44 +01:00
|
|
|
participants=group,
|
2026-02-25 16:54:21 +01:00
|
|
|
score_value=score_value,
|
2026-03-06 15:02:53 +01:00
|
|
|
score_kind=ScoreKind.VP,
|
2026-02-23 21:18:27 +01:00
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
return ties
|
2026-02-20 11:01:25 +01:00
|
|
|
|
2026-03-03 11:52:07 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def find_war_objective_ties(
|
|
|
|
|
war: War,
|
|
|
|
|
objective_id: str,
|
|
|
|
|
) -> List[TieContext]:
|
|
|
|
|
from warchron.model.result_checker import ResultChecker
|
|
|
|
|
|
2026-03-05 11:37:14 +01:00
|
|
|
scores = ScoreService.compute_scores(
|
2026-03-03 11:52:07 +01:00
|
|
|
war,
|
|
|
|
|
ContextType.WAR,
|
|
|
|
|
war.id,
|
|
|
|
|
)
|
2026-03-05 11:37:14 +01:00
|
|
|
|
|
|
|
|
def value_getter(score: ParticipantScore) -> int:
|
|
|
|
|
return score.narrative_points.get(objective_id, 0)
|
|
|
|
|
|
2026-03-03 11:52:07 +01:00
|
|
|
ranking = ResultChecker.get_effective_ranking(
|
|
|
|
|
war,
|
2026-03-06 15:02:53 +01:00
|
|
|
ContextType.WAR,
|
|
|
|
|
war.id,
|
|
|
|
|
ScoreKind.NP,
|
2026-03-03 11:52:07 +01:00
|
|
|
scores,
|
2026-03-06 15:02:53 +01:00
|
|
|
value_getter,
|
|
|
|
|
objective_id,
|
2026-03-03 11:52:07 +01:00
|
|
|
)
|
|
|
|
|
ties: List[TieContext] = []
|
|
|
|
|
for _, group, _ in ranking:
|
|
|
|
|
if len(group) <= 1:
|
|
|
|
|
continue
|
2026-03-05 11:37:14 +01:00
|
|
|
np_value = value_getter(scores[group[0]])
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext = TieContext(
|
|
|
|
|
ContextType.WAR, war.id, [], np_value, ScoreKind.NP, objective_id
|
|
|
|
|
)
|
2026-03-03 11:52:07 +01:00
|
|
|
if TieResolver.is_tie_resolved(
|
|
|
|
|
war,
|
2026-03-06 15:02:53 +01:00
|
|
|
context,
|
2026-03-03 11:52:07 +01:00
|
|
|
):
|
|
|
|
|
continue
|
|
|
|
|
if not TieResolver.can_tie_be_resolved(
|
|
|
|
|
war,
|
2026-03-06 15:02:53 +01:00
|
|
|
context,
|
2026-03-03 11:52:07 +01:00
|
|
|
group,
|
|
|
|
|
):
|
|
|
|
|
war.events.append(
|
2026-03-06 15:02:53 +01:00
|
|
|
TieResolved(None, ContextType.WAR, war.id, np_value, objective_id)
|
2026-03-03 11:52:07 +01:00
|
|
|
)
|
|
|
|
|
continue
|
|
|
|
|
ties.append(
|
|
|
|
|
TieContext(
|
2026-03-06 15:02:53 +01:00
|
|
|
context_type=ContextType.WAR,
|
|
|
|
|
context_id=war.id,
|
2026-03-03 11:52:07 +01:00
|
|
|
participants=group,
|
2026-03-05 11:37:14 +01:00
|
|
|
score_value=np_value,
|
2026-03-06 15:02:53 +01:00
|
|
|
score_kind=ScoreKind.NP,
|
|
|
|
|
objective_id=objective_id,
|
2026-03-03 11:52:07 +01:00
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
return ties
|
|
|
|
|
|
2026-02-17 16:37:36 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def apply_bids(
|
|
|
|
|
war: War,
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext,
|
2026-02-17 16:37:36 +01:00
|
|
|
bids: Dict[str, bool], # war_participant_id -> spend?
|
|
|
|
|
) -> None:
|
|
|
|
|
for war_part_id, spend in bids.items():
|
|
|
|
|
if not spend:
|
|
|
|
|
continue
|
|
|
|
|
if war.get_influence_tokens(war_part_id) < 1:
|
|
|
|
|
raise ForbiddenOperation("Not enough tokens")
|
|
|
|
|
war.events.append(
|
|
|
|
|
InfluenceSpent(
|
|
|
|
|
participant_id=war_part_id,
|
|
|
|
|
amount=1,
|
2026-03-06 15:02:53 +01:00
|
|
|
context_type=context.context_type,
|
|
|
|
|
context_id=context.context_id,
|
|
|
|
|
objective_id=context.objective_id,
|
2026-02-17 16:37:36 +01:00
|
|
|
)
|
|
|
|
|
)
|
2026-02-11 19:22:43 +01:00
|
|
|
|
2026-02-24 10:04:16 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def cancel_tie_break(
|
|
|
|
|
war: War,
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext,
|
2026-02-24 10:04:16 +01:00
|
|
|
) -> None:
|
|
|
|
|
war.events = [
|
|
|
|
|
ev
|
|
|
|
|
for ev in war.events
|
|
|
|
|
if not (
|
|
|
|
|
(
|
|
|
|
|
isinstance(ev, InfluenceSpent)
|
2026-03-06 15:02:53 +01:00
|
|
|
and ev.context_type == context.context_type
|
|
|
|
|
and ev.context_id == context.context_id
|
2026-02-24 10:04:16 +01:00
|
|
|
)
|
|
|
|
|
or (
|
|
|
|
|
isinstance(ev, TieResolved)
|
2026-03-06 15:02:53 +01:00
|
|
|
and ev.context_type == context.context_type
|
|
|
|
|
and ev.context_id == context.context_id
|
2026-02-24 10:04:16 +01:00
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
]
|
|
|
|
|
|
2026-02-17 16:37:36 +01:00
|
|
|
@staticmethod
|
2026-02-20 23:44:22 +01:00
|
|
|
def rank_by_tokens(
|
2026-02-17 16:37:36 +01:00
|
|
|
war: War,
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext,
|
2026-02-20 23:44:22 +01:00
|
|
|
participants: List[str],
|
|
|
|
|
) -> List[List[str]]:
|
|
|
|
|
spent = {pid: 0 for pid in participants}
|
|
|
|
|
for ev in war.events:
|
|
|
|
|
if (
|
|
|
|
|
isinstance(ev, InfluenceSpent)
|
2026-03-06 15:02:53 +01:00
|
|
|
and ev.context_type == context.context_type
|
|
|
|
|
and ev.context_id == context.context_id
|
|
|
|
|
and ev.objective_id == context.objective_id
|
2026-02-20 23:44:22 +01:00
|
|
|
and ev.participant_id in spent
|
|
|
|
|
):
|
|
|
|
|
spent[ev.participant_id] += ev.amount
|
|
|
|
|
sorted_items = sorted(spent.items(), key=lambda x: x[1], reverse=True)
|
|
|
|
|
groups: List[List[str]] = []
|
|
|
|
|
current_score = None
|
|
|
|
|
for pid, score in sorted_items:
|
|
|
|
|
if score != current_score:
|
|
|
|
|
groups.append([])
|
|
|
|
|
current_score = score
|
|
|
|
|
groups[-1].append(pid)
|
|
|
|
|
return groups
|
|
|
|
|
|
2026-02-23 19:28:13 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def tokens_spent_map(
|
|
|
|
|
war: War,
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext,
|
2026-02-23 19:28:13 +01:00
|
|
|
participants: List[str],
|
|
|
|
|
) -> Dict[str, int]:
|
|
|
|
|
spent = {pid: 0 for pid in participants}
|
|
|
|
|
for ev in war.events:
|
|
|
|
|
if (
|
|
|
|
|
isinstance(ev, InfluenceSpent)
|
2026-03-06 15:02:53 +01:00
|
|
|
and ev.context_type == context.context_type
|
|
|
|
|
and ev.context_id == context.context_id
|
|
|
|
|
and ev.objective_id == context.objective_id
|
2026-02-23 19:28:13 +01:00
|
|
|
and ev.participant_id in spent
|
|
|
|
|
):
|
|
|
|
|
spent[ev.participant_id] += ev.amount
|
|
|
|
|
return spent
|
|
|
|
|
|
2026-02-23 13:16:45 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def get_active_participants(
|
|
|
|
|
war: War,
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext,
|
2026-02-23 13:16:45 +01:00
|
|
|
participants: List[str],
|
|
|
|
|
) -> List[str]:
|
2026-03-06 15:02:53 +01:00
|
|
|
groups = TieResolver.rank_by_tokens(war, context, participants)
|
2026-02-23 13:16:45 +01:00
|
|
|
return groups[0]
|
|
|
|
|
|
2026-02-20 23:44:22 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def resolve_tie_state(
|
|
|
|
|
war: War,
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext,
|
2026-02-20 23:44:22 +01:00
|
|
|
bids: dict[str, bool] | None = None,
|
|
|
|
|
) -> None:
|
2026-02-23 13:16:45 +01:00
|
|
|
active = TieResolver.get_active_participants(
|
2026-02-25 16:54:21 +01:00
|
|
|
war,
|
2026-03-06 15:02:53 +01:00
|
|
|
context,
|
|
|
|
|
context.participants,
|
2026-02-23 13:16:45 +01:00
|
|
|
)
|
|
|
|
|
# confirmed draw if non had bid
|
|
|
|
|
if not active:
|
2026-02-25 16:54:21 +01:00
|
|
|
war.events.append(
|
2026-03-06 15:02:53 +01:00
|
|
|
TieResolved(
|
|
|
|
|
None,
|
|
|
|
|
context.context_type,
|
|
|
|
|
context.context_id,
|
|
|
|
|
context.score_value,
|
|
|
|
|
context.objective_id,
|
|
|
|
|
)
|
2026-02-25 16:54:21 +01:00
|
|
|
)
|
2026-02-23 13:16:45 +01:00
|
|
|
return
|
|
|
|
|
# confirmed draw if current bids are 0
|
2026-02-20 23:44:22 +01:00
|
|
|
if bids is not None and not any(bids.values()):
|
2026-02-25 16:54:21 +01:00
|
|
|
war.events.append(
|
2026-03-06 15:02:53 +01:00
|
|
|
TieResolved(
|
|
|
|
|
None,
|
|
|
|
|
context.context_type,
|
|
|
|
|
context.context_id,
|
|
|
|
|
context.score_value,
|
|
|
|
|
context.objective_id,
|
|
|
|
|
)
|
2026-02-25 16:54:21 +01:00
|
|
|
)
|
2026-02-20 23:44:22 +01:00
|
|
|
return
|
|
|
|
|
# else rank_by_tokens
|
2026-03-06 15:02:53 +01:00
|
|
|
groups = TieResolver.rank_by_tokens(war, context, context.participants)
|
2026-02-20 23:44:22 +01:00
|
|
|
if len(groups[0]) == 1:
|
2026-02-25 16:54:21 +01:00
|
|
|
war.events.append(
|
|
|
|
|
TieResolved(
|
2026-03-06 15:02:53 +01:00
|
|
|
groups[0][0],
|
|
|
|
|
context.context_type,
|
|
|
|
|
context.context_id,
|
|
|
|
|
context.score_value,
|
|
|
|
|
context.objective_id,
|
2026-02-25 16:54:21 +01:00
|
|
|
)
|
|
|
|
|
)
|
2026-02-20 23:44:22 +01:00
|
|
|
return
|
|
|
|
|
# if tie persists, do nothing, workflow will call again
|
2026-02-11 19:22:43 +01:00
|
|
|
|
|
|
|
|
@staticmethod
|
2026-02-23 13:16:45 +01:00
|
|
|
def can_tie_be_resolved(
|
2026-03-06 15:02:53 +01:00
|
|
|
war: War, context: TieContext, participants: List[str]
|
2026-02-23 13:16:45 +01:00
|
|
|
) -> bool:
|
2026-03-06 15:02:53 +01:00
|
|
|
active = TieResolver.get_active_participants(war, context, participants)
|
2026-02-23 13:16:45 +01:00
|
|
|
return any(war.get_influence_tokens(pid) > 0 for pid in active)
|
2026-02-17 16:37:36 +01:00
|
|
|
|
2026-02-18 11:15:53 +01:00
|
|
|
@staticmethod
|
|
|
|
|
def was_tie_broken_by_tokens(
|
|
|
|
|
war: War,
|
2026-03-06 15:02:53 +01:00
|
|
|
context: TieContext,
|
2026-02-18 11:15:53 +01:00
|
|
|
) -> bool:
|
|
|
|
|
for ev in reversed(war.events):
|
|
|
|
|
if (
|
|
|
|
|
isinstance(ev, TieResolved)
|
2026-03-06 15:02:53 +01:00
|
|
|
and ev.context_type == context.context_type
|
|
|
|
|
and ev.context_id == context.context_id
|
|
|
|
|
and ev.score_value == context.score_value
|
|
|
|
|
and ev.objective_id == context.objective_id
|
2026-02-18 11:15:53 +01:00
|
|
|
):
|
|
|
|
|
return ev.participant_id is not None
|
|
|
|
|
return False
|
2026-02-20 23:44:22 +01:00
|
|
|
|
|
|
|
|
@staticmethod
|
2026-03-06 15:02:53 +01:00
|
|
|
def is_tie_resolved(war: War, context: TieContext) -> bool:
|
2026-02-20 23:44:22 +01:00
|
|
|
return any(
|
|
|
|
|
isinstance(ev, TieResolved)
|
2026-03-06 15:02:53 +01:00
|
|
|
and ev.context_type == context.context_type
|
|
|
|
|
and ev.context_id == context.context_id
|
|
|
|
|
and ev.score_value == context.score_value
|
|
|
|
|
and ev.objective_id == context.objective_id
|
2026-02-20 23:44:22 +01:00
|
|
|
for ev in war.events
|
|
|
|
|
)
|