warchron_app/src/warchron/model/war.py

384 lines
13 KiB
Python
Raw Normal View History

from __future__ import annotations
2026-02-02 14:33:31 +01:00
from uuid import uuid4
2026-01-19 18:55:07 +01:00
from datetime import datetime
2026-02-04 16:10:53 +01:00
from typing import Any, Dict, List
2026-01-19 18:55:07 +01:00
from warchron.model.exception import DeletionForbidden
from warchron.model.war_participant import WarParticipant
from warchron.model.objective import Objective
from warchron.model.campaign_participant import CampaignParticipant
from warchron.model.sector import Sector
from warchron.model.campaign import Campaign
from warchron.model.round import Round
from warchron.model.choice import Choice
from warchron.model.battle import Battle
2026-01-19 18:55:07 +01:00
2026-01-19 18:55:07 +01:00
class War:
2026-02-04 16:10:53 +01:00
def __init__(self, name: str, year: int) -> None:
self.id: str = str(uuid4())
self.name: str = name
self.year: int = year
2026-02-10 16:26:49 +01:00
self.major_value: int = 2
self.minor_value: int = 1
self.influence_token: bool = True
2026-02-04 16:10:53 +01:00
self.participants: Dict[str, WarParticipant] = {}
self.objectives: Dict[str, Objective] = {}
self.campaigns: List[Campaign] = []
self.is_over: bool = False
2026-01-19 18:55:07 +01:00
2026-02-04 16:10:53 +01:00
def set_id(self, new_id: str) -> None:
2026-01-19 18:55:07 +01:00
self.id = new_id
2026-02-04 16:10:53 +01:00
def set_name(self, new_name: str) -> None:
2026-01-19 18:55:07 +01:00
self.name = new_name
2026-02-04 16:10:53 +01:00
def set_year(self, new_year: int) -> None:
2026-01-19 18:55:07 +01:00
self.year = new_year
2026-02-02 14:33:31 +01:00
2026-02-10 16:26:49 +01:00
def set_major(self, new_value: int) -> None:
if new_value < self.minor_value:
raise ValueError("major_value cannot be < minor_value")
self.major_value = new_value
def set_minor(self, new_value: int) -> None:
if new_value > self.major_value:
raise ValueError("minor_value cannot be > major_value")
self.minor_value = new_value
def set_influence(self, new_state: bool) -> None:
self.influence_token = new_state
2026-02-04 16:10:53 +01:00
def set_state(self, new_state: bool) -> None:
2026-01-19 18:55:07 +01:00
self.is_over = new_state
2026-02-04 16:10:53 +01:00
def toDict(self) -> Dict[str, Any]:
2026-01-19 18:55:07 +01:00
return {
2026-02-02 14:33:31 +01:00
"id": self.id,
"name": self.name,
"year": self.year,
2026-02-10 16:26:49 +01:00
"major_value": self.major_value,
"minor_value": self.minor_value,
"influence_token": self.influence_token,
"participants": [part.toDict() for part in self.participants.values()],
"objectives": [obj.toDict() for obj in self.objectives.values()],
2026-01-21 08:31:48 +01:00
"campaigns": [camp.toDict() for camp in self.campaigns],
2026-02-02 14:33:31 +01:00
"is_over": self.is_over,
2026-01-19 18:55:07 +01:00
}
2026-02-02 14:33:31 +01:00
2026-01-19 18:55:07 +01:00
@staticmethod
2026-02-04 16:10:53 +01:00
def fromDict(data: Dict[str, Any]) -> War:
2026-01-22 23:42:47 +01:00
war = War(name=data["name"], year=data["year"])
2026-01-21 08:31:48 +01:00
war.set_id(data["id"])
2026-02-10 16:26:49 +01:00
war.set_major(data["major_value"])
war.set_minor(data["minor_value"])
war.set_influence(data["influence_token"])
for part_data in data.get("participants", []):
part = WarParticipant.fromDict(part_data)
war.participants[part.id] = part
for obj_data in data.get("objectives", []):
obj = Objective.fromDict(obj_data)
war.objectives[obj.id] = obj
2026-01-21 08:31:48 +01:00
for camp_data in data.get("campaigns", []):
war.campaigns.append(Campaign.fromDict(camp_data))
war.set_state(data.get("is_over", False))
return war
2026-02-02 14:33:31 +01:00
# Objective methods
def add_objective(self, name: str, description: str | None) -> Objective:
obj = Objective(name, description)
self.objectives[obj.id] = obj
return obj
def get_objective(self, id: str) -> Objective:
return self.objectives[id]
2026-02-04 16:10:53 +01:00
def get_all_objectives(self) -> List[Objective]:
return list(self.objectives.values())
def get_objective_name(self, objective_id: str | None) -> str:
if objective_id is None:
return ""
obj = self.objectives.get(objective_id)
return obj.name if obj else ""
2026-02-02 14:33:31 +01:00
2026-02-04 16:10:53 +01:00
def update_objective(
self, objective_id: str, *, name: str, description: str | None
2026-02-04 16:10:53 +01:00
) -> None:
obj = self.get_objective(objective_id)
obj.set_name(name)
obj.set_description(description)
2026-02-02 14:33:31 +01:00
2026-02-04 16:10:53 +01:00
def remove_objective(self, objective_id: str) -> None:
camp_using_obj: List[str] = []
for camp in self.campaigns:
if camp.has_sector_with_objective(objective_id):
camp_using_obj.append(camp.name)
if camp_using_obj:
camps_str = ", ".join(camp_using_obj)
raise DeletionForbidden(
f"This objective is used in campaign(s) sector(s): {camps_str}.\n"
"Remove it from campaign(s) first."
)
del self.objectives[objective_id]
2026-02-02 14:33:31 +01:00
# War participant methods
def get_all_war_participants_ids(self) -> set[str]:
return set(self.participants.keys())
def has_participant(self, participant_id: str) -> bool:
return participant_id in self.participants
def has_player(self, player_id: str) -> bool:
return any(part.player_id == player_id for part in self.participants.values())
def add_war_participant(self, player_id: str, faction: str) -> WarParticipant:
if self.has_player(player_id):
raise ValueError("Player already registered in this war")
participant = WarParticipant(player_id=player_id, faction=faction)
self.participants[participant.id] = participant
return participant
def get_war_participant(self, id: str) -> WarParticipant:
return self.participants[id]
2026-02-04 16:10:53 +01:00
def get_all_war_participants(self) -> List[WarParticipant]:
return list(self.participants.values())
def update_war_participant(self, participant_id: str, *, faction: str) -> None:
part = self.get_war_participant(participant_id)
# Can't change referred Model.players
part.set_faction(faction)
def remove_war_participant(self, participant_id: str) -> None:
camp_using_part: List[str] = []
for camp in self.campaigns:
if camp.has_war_participant(participant_id):
camp_using_part.append(camp.name)
if camp_using_part:
camps_str = ", ".join(camp_using_part)
raise DeletionForbidden(
f"This war participant is used in campaign(s): {camps_str}.\n"
"Remove it from campaign(s) first."
)
del self.participants[participant_id]
2026-02-02 14:33:31 +01:00
# Campaign methods
def has_campaign(self, campaign_id: str) -> bool:
return any(c.id == campaign_id for c in self.campaigns)
2026-02-04 16:10:53 +01:00
def get_default_campaign_values(self) -> Dict[str, Any]:
2026-02-02 14:33:31 +01:00
return {"month": datetime.now().month}
def all_campaigns_finished(self) -> bool:
return all(c.is_over for c in self.campaigns)
2026-01-22 23:42:47 +01:00
def add_campaign(self, name: str, month: int | None = None) -> Campaign:
if month is None:
month = self.get_default_campaign_values()["month"]
campaign = Campaign(name, month)
2026-01-21 07:43:04 +01:00
self.campaigns.append(campaign)
return campaign
def get_campaign(self, campaign_id: str) -> Campaign:
2026-01-22 23:42:47 +01:00
for camp in self.campaigns:
if camp.id == campaign_id:
return camp
raise KeyError(f"Campaign {campaign_id} not found in War {self.id}")
2026-01-21 07:43:04 +01:00
# TODO replace multiloops by internal has_* method
2026-02-05 10:19:14 +01:00
def get_campaign_by_round(self, round_id: str) -> Campaign | None:
2026-01-22 23:42:47 +01:00
for camp in self.campaigns:
for rnd in camp.rounds:
if rnd.id == round_id:
return camp
2026-02-05 10:19:14 +01:00
return None
2026-01-22 23:42:47 +01:00
# TODO replace multiloops by internal has_* method
def get_campaign_by_sector(self, sector_id: str) -> Campaign:
for camp in self.campaigns:
for sect in camp.sectors.values():
if sect.id == sector_id:
return camp
raise KeyError(f"Sector {sector_id} not found in any Campaign")
def get_campaign_by_campaign_participant(self, participant_id: str) -> Campaign:
for camp in self.campaigns:
2026-02-02 14:33:31 +01:00
if camp.has_participant(participant_id):
return camp
raise KeyError(f"Participant {participant_id} not found in any Campaign")
2026-02-04 16:10:53 +01:00
def update_campaign(self, campaign_id: str, *, name: str, month: int) -> None:
2026-01-22 23:42:47 +01:00
camp = self.get_campaign(campaign_id)
camp.set_name(name)
camp.set_month(month)
2026-02-02 14:33:31 +01:00
2026-02-04 16:10:53 +01:00
def get_all_campaigns(self) -> List[Campaign]:
2026-01-21 07:43:04 +01:00
return list(self.campaigns)
2026-02-02 14:33:31 +01:00
2026-02-04 16:10:53 +01:00
def remove_campaign(self, campaign_id: str) -> None:
2026-01-22 23:42:47 +01:00
camp = self.get_campaign(campaign_id)
self.campaigns.remove(camp)
2026-02-02 14:33:31 +01:00
# Sector methods
def add_sector(
self,
2026-02-04 16:10:53 +01:00
campaign_id: str,
2026-02-02 14:33:31 +01:00
name: str,
round_id: str,
major_id: str,
minor_id: str,
influence_id: str,
) -> Sector:
camp = self.get_campaign(campaign_id)
return camp.add_sector(name, round_id, major_id, minor_id, influence_id)
# TODO replace multiloops by internal has_* method
2026-02-04 16:10:53 +01:00
def get_sector(self, sector_id: str) -> Sector:
for camp in self.campaigns:
for sect in camp.sectors.values():
if sect.id == sector_id:
return sect
raise KeyError("Sector not found")
2026-02-02 14:33:31 +01:00
def update_sector(
self,
sector_id: str,
*,
name: str,
round_id: str | None,
major_id: str | None,
minor_id: str | None,
influence_id: str | None,
2026-02-04 16:10:53 +01:00
) -> None:
2026-01-30 15:32:44 +01:00
camp = self.get_campaign_by_sector(sector_id)
2026-02-02 14:33:31 +01:00
camp.update_sector(
sector_id,
name=name,
round_id=round_id,
major_id=major_id,
minor_id=minor_id,
influence_id=influence_id,
)
2026-01-30 15:32:44 +01:00
2026-02-04 16:10:53 +01:00
def remove_sector(self, sector_id: str) -> None:
camp = self.get_campaign_by_sector(sector_id)
camp.remove_sector(sector_id)
2026-02-02 14:33:31 +01:00
# Campaign participant methods
2026-02-04 16:10:53 +01:00
def get_available_war_participants(self, campaign_id: str) -> List[WarParticipant]:
camp = self.get_campaign(campaign_id)
return [
part
for part in self.participants.values()
if not camp.has_war_participant(part.id)
]
2026-02-02 14:33:31 +01:00
def add_campaign_participant(
self, campaign_id: str, participant_id: str, leader: str, theme: str
) -> CampaignParticipant:
camp = self.get_campaign(campaign_id)
2026-02-02 14:33:31 +01:00
return camp.add_campaign_participant(participant_id, leader, theme)
# TODO replace multiloops by internal has_* method
2026-02-04 16:10:53 +01:00
def get_campaign_participant(self, participant_id: str) -> CampaignParticipant:
for camp in self.campaigns:
for part in camp.participants.values():
if part.id == participant_id:
return part
raise KeyError("Participant not found")
2026-02-02 14:33:31 +01:00
def update_campaign_participant(
self, participant_id: str, *, leader: str, theme: str
2026-02-04 16:10:53 +01:00
) -> None:
camp = self.get_campaign_by_campaign_participant(participant_id)
2026-02-02 14:33:31 +01:00
camp.update_campaign_participant(participant_id, leader=leader, theme=theme)
2026-02-04 16:10:53 +01:00
def remove_campaign_participant(self, participant_id: str) -> None:
camp = self.get_campaign_by_campaign_participant(participant_id)
camp.remove_campaign_participant(participant_id)
2026-02-02 14:33:31 +01:00
# Round methods
2026-01-30 15:32:44 +01:00
def add_round(self, campaign_id: str) -> Round:
camp = self.get_campaign(campaign_id)
return camp.add_round()
2026-02-04 16:10:53 +01:00
def remove_round(self, round_id: str) -> None:
2026-01-30 15:32:44 +01:00
camp = self.get_campaign_by_round(round_id)
2026-02-05 10:19:14 +01:00
if camp is not None:
camp.remove_round(round_id)
2026-01-30 15:32:44 +01:00
2026-02-02 14:33:31 +01:00
# Choice methods
2026-01-30 15:32:44 +01:00
def create_choice(self, round_id: str, participant_id: str) -> Choice:
camp = self.get_campaign_by_round(round_id)
2026-02-05 10:19:14 +01:00
if camp is not None:
return camp.create_choice(round_id, participant_id)
raise KeyError("Campaign with round {round_id} doesn't exist")
2026-02-02 14:33:31 +01:00
def update_choice(
self,
round_id: str,
participant_id: str,
priority_sector_id: str | None,
secondary_sector_id: str | None,
comment: str | None,
2026-02-04 16:10:53 +01:00
) -> None:
2026-02-02 10:41:16 +01:00
camp = self.get_campaign_by_round(round_id)
2026-02-05 10:19:14 +01:00
if camp is not None:
camp.update_choice(
round_id,
participant_id,
priority_sector_id,
secondary_sector_id,
comment,
)
2026-01-30 15:32:44 +01:00
2026-02-04 16:10:53 +01:00
def remove_choice(self, round_id: str, participant_id: str) -> None:
2026-01-30 15:32:44 +01:00
camp = self.get_campaign_by_round(round_id)
2026-02-05 10:19:14 +01:00
if camp is not None:
camp.remove_choice(round_id, participant_id)
2026-01-30 15:32:44 +01:00
2026-02-02 14:33:31 +01:00
# Battle methods
2026-01-30 18:55:39 +01:00
def create_battle(self, round_id: str, sector_id: str) -> Battle:
camp = self.get_campaign_by_round(round_id)
2026-02-05 10:19:14 +01:00
if camp is not None:
return camp.create_battle(round_id, sector_id)
raise KeyError("Campaign with round {round_id} doesn't exist")
2026-01-30 18:55:39 +01:00
2026-02-02 14:33:31 +01:00
def update_battle(
self,
round_id: str,
sector_id: str,
player_1_id: str | None,
player_2_id: str | None,
winner_id: str | None,
score: str | None,
victory_condition: str | None,
comment: str | None,
2026-02-04 16:10:53 +01:00
) -> None:
2026-02-02 10:41:16 +01:00
camp = self.get_campaign_by_round(round_id)
2026-02-05 10:19:14 +01:00
if camp is not None:
camp.update_battle(
round_id,
sector_id,
player_1_id,
player_2_id,
winner_id,
score,
victory_condition,
comment,
)
2026-02-02 10:41:16 +01:00
2026-02-04 16:10:53 +01:00
def remove_battle(self, round_id: str, sector_id: str) -> None:
2026-01-30 18:55:39 +01:00
camp = self.get_campaign_by_round(round_id)
2026-02-05 10:19:14 +01:00
if camp is not None:
camp.remove_battle(round_id, sector_id)