mirror of
https://github.com/mwisnowski/mtg_python_deckbuilder.git
synced 2025-12-16 23:50:12 +01:00
fix(lint): improved type checking and code quality (77% error reduction)
This commit is contained in:
parent
3c45a31aa3
commit
83fe527979
37 changed files with 423 additions and 303 deletions
|
|
@ -25,11 +25,11 @@ No behavior change intended.
|
|||
|
||||
# Attempt to use a fast fuzzy library; fall back gracefully
|
||||
try:
|
||||
from rapidfuzz import process as rf_process, fuzz as rf_fuzz # type: ignore
|
||||
from rapidfuzz import process as rf_process, fuzz as rf_fuzz
|
||||
_FUZZ_BACKEND = "rapidfuzz"
|
||||
except ImportError: # pragma: no cover - environment dependent
|
||||
try:
|
||||
from fuzzywuzzy import process as fw_process, fuzz as fw_fuzz # type: ignore
|
||||
from fuzzywuzzy import process as fw_process, fuzz as fw_fuzz
|
||||
_FUZZ_BACKEND = "fuzzywuzzy"
|
||||
except ImportError: # pragma: no cover
|
||||
_FUZZ_BACKEND = "difflib"
|
||||
|
|
|
|||
|
|
@ -68,7 +68,7 @@ class CommanderSelectionMixin:
|
|||
out_words[0] = out_words[0][:1].upper() + out_words[0][1:]
|
||||
return ' '.join(out_words)
|
||||
|
||||
def choose_commander(self) -> str: # type: ignore[override]
|
||||
def choose_commander(self) -> str:
|
||||
df = self.load_commander_data()
|
||||
names = df["name"].tolist()
|
||||
while True:
|
||||
|
|
@ -113,7 +113,7 @@ class CommanderSelectionMixin:
|
|||
continue
|
||||
query = self._normalize_commander_query(choice) # treat as new (normalized) query
|
||||
|
||||
def _present_commander_and_confirm(self, df: pd.DataFrame, name: str) -> bool: # type: ignore[override]
|
||||
def _present_commander_and_confirm(self, df: pd.DataFrame, name: str) -> bool:
|
||||
row = df[df["name"] == name].iloc[0]
|
||||
pretty = self._format_commander_pretty(row)
|
||||
self.output_func("\n" + pretty)
|
||||
|
|
@ -126,7 +126,7 @@ class CommanderSelectionMixin:
|
|||
return False
|
||||
self.output_func("Please enter y or n.")
|
||||
|
||||
def _apply_commander_selection(self, row: pd.Series): # type: ignore[override]
|
||||
def _apply_commander_selection(self, row: pd.Series):
|
||||
self.commander_name = row["name"]
|
||||
self.commander_row = row
|
||||
tags_value = row.get("themeTags", [])
|
||||
|
|
@ -136,7 +136,7 @@ class CommanderSelectionMixin:
|
|||
# ---------------------------
|
||||
# Tag Prioritization
|
||||
# ---------------------------
|
||||
def select_commander_tags(self) -> List[str]: # type: ignore[override]
|
||||
def select_commander_tags(self) -> List[str]:
|
||||
if not self.commander_name:
|
||||
self.output_func("No commander chosen yet. Selecting commander first...")
|
||||
self.choose_commander()
|
||||
|
|
@ -173,7 +173,7 @@ class CommanderSelectionMixin:
|
|||
self._update_commander_dict_with_selected_tags()
|
||||
return self.selected_tags
|
||||
|
||||
def _prompt_tag_choice(self, available: List[str], prompt_text: str, allow_stop: bool) -> Optional[str]: # type: ignore[override]
|
||||
def _prompt_tag_choice(self, available: List[str], prompt_text: str, allow_stop: bool) -> Optional[str]:
|
||||
while True:
|
||||
self.output_func("\nCurrent options:")
|
||||
for i, t in enumerate(available, 1):
|
||||
|
|
@ -192,7 +192,7 @@ class CommanderSelectionMixin:
|
|||
return matches[0]
|
||||
self.output_func("Invalid selection. Try again.")
|
||||
|
||||
def _update_commander_dict_with_selected_tags(self): # type: ignore[override]
|
||||
def _update_commander_dict_with_selected_tags(self):
|
||||
if not self.commander_dict and self.commander_row is not None:
|
||||
self._initialize_commander_dict(self.commander_row)
|
||||
if not self.commander_dict:
|
||||
|
|
@ -205,7 +205,7 @@ class CommanderSelectionMixin:
|
|||
# ---------------------------
|
||||
# Power Bracket Selection
|
||||
# ---------------------------
|
||||
def select_power_bracket(self) -> BracketDefinition: # type: ignore[override]
|
||||
def select_power_bracket(self) -> BracketDefinition:
|
||||
if self.bracket_definition:
|
||||
return self.bracket_definition
|
||||
self.output_func("\nChoose Deck Power Bracket:")
|
||||
|
|
@ -229,14 +229,14 @@ class CommanderSelectionMixin:
|
|||
return match
|
||||
self.output_func("Invalid input. Type 1-5 or 'info'.")
|
||||
|
||||
def _print_bracket_details(self): # type: ignore[override]
|
||||
def _print_bracket_details(self):
|
||||
self.output_func("\nBracket Details:")
|
||||
for bd in BRACKET_DEFINITIONS:
|
||||
self.output_func(f"\n[{bd.level}] {bd.name}")
|
||||
self.output_func(bd.long_desc)
|
||||
self.output_func(self._format_limits(bd.limits))
|
||||
|
||||
def _print_selected_bracket_summary(self): # type: ignore[override]
|
||||
def _print_selected_bracket_summary(self):
|
||||
self.output_func("\nBracket Constraints:")
|
||||
if self.bracket_limits:
|
||||
self.output_func(self._format_limits(self.bracket_limits))
|
||||
|
|
|
|||
|
|
@ -22,7 +22,7 @@ Expected attributes / methods on the host DeckBuilder:
|
|||
|
||||
|
||||
class LandBasicsMixin:
|
||||
def add_basic_lands(self): # type: ignore[override]
|
||||
def add_basic_lands(self):
|
||||
"""Add basic (or snow basic) lands based on color identity.
|
||||
|
||||
Logic:
|
||||
|
|
@ -71,8 +71,8 @@ class LandBasicsMixin:
|
|||
basic_min: Optional[int] = None
|
||||
land_total: Optional[int] = None
|
||||
if hasattr(self, 'ideal_counts') and getattr(self, 'ideal_counts'):
|
||||
basic_min = self.ideal_counts.get('basic_lands') # type: ignore[attr-defined]
|
||||
land_total = self.ideal_counts.get('lands') # type: ignore[attr-defined]
|
||||
basic_min = self.ideal_counts.get('basic_lands')
|
||||
land_total = self.ideal_counts.get('lands')
|
||||
if basic_min is None:
|
||||
basic_min = getattr(bc, 'DEFAULT_BASIC_LAND_COUNT', 20)
|
||||
if land_total is None:
|
||||
|
|
@ -136,7 +136,7 @@ class LandBasicsMixin:
|
|||
self.output_func(f" {name.ljust(width)} : {cnt}")
|
||||
self.output_func(f" Total Basics : {sum(allocation.values())} (Target {target_basics}, Min {basic_min})")
|
||||
|
||||
def run_land_step1(self): # type: ignore[override]
|
||||
def run_land_step1(self):
|
||||
"""Public wrapper to execute land building step 1 (basics)."""
|
||||
self.add_basic_lands()
|
||||
try:
|
||||
|
|
|
|||
|
|
@ -21,7 +21,7 @@ Host DeckBuilder must provide:
|
|||
"""
|
||||
|
||||
class LandDualsMixin:
|
||||
def add_dual_lands(self, requested_count: int | None = None): # type: ignore[override]
|
||||
def add_dual_lands(self, requested_count: int | None = None):
|
||||
"""Add two-color 'typed' dual lands based on color identity."""
|
||||
if not getattr(self, 'files_to_load', []):
|
||||
try:
|
||||
|
|
@ -117,10 +117,10 @@ class LandDualsMixin:
|
|||
pair_buckets[key] = names
|
||||
min_basic_cfg = getattr(bc, 'DEFAULT_BASIC_LAND_COUNT', 20)
|
||||
if getattr(self, 'ideal_counts', None):
|
||||
min_basic_cfg = self.ideal_counts.get('basic_lands', min_basic_cfg) # type: ignore[attr-defined]
|
||||
basic_floor = self._basic_floor(min_basic_cfg) # type: ignore[attr-defined]
|
||||
min_basic_cfg = self.ideal_counts.get('basic_lands', min_basic_cfg)
|
||||
basic_floor = self._basic_floor(min_basic_cfg)
|
||||
default_dual_target = getattr(bc, 'DUAL_LAND_DEFAULT_COUNT', 6)
|
||||
remaining_capacity = max(0, land_target - self._current_land_count()) # type: ignore[attr-defined]
|
||||
remaining_capacity = max(0, land_target - self._current_land_count())
|
||||
effective_default = min(default_dual_target, remaining_capacity if remaining_capacity>0 else len(pool), len(pool))
|
||||
desired = effective_default if requested_count is None else max(0, int(requested_count))
|
||||
if desired == 0:
|
||||
|
|
@ -129,14 +129,14 @@ class LandDualsMixin:
|
|||
if remaining_capacity == 0 and desired > 0:
|
||||
slots_needed = desired
|
||||
freed_slots = 0
|
||||
while freed_slots < slots_needed and self._count_basic_lands() > basic_floor: # type: ignore[attr-defined]
|
||||
target_basic = self._choose_basic_to_trim() # type: ignore[attr-defined]
|
||||
if not target_basic or not self._decrement_card(target_basic): # type: ignore[attr-defined]
|
||||
while freed_slots < slots_needed and self._count_basic_lands() > basic_floor:
|
||||
target_basic = self._choose_basic_to_trim()
|
||||
if not target_basic or not self._decrement_card(target_basic):
|
||||
break
|
||||
freed_slots += 1
|
||||
if freed_slots == 0:
|
||||
desired = 0
|
||||
remaining_capacity = max(0, land_target - self._current_land_count()) # type: ignore[attr-defined]
|
||||
remaining_capacity = max(0, land_target - self._current_land_count())
|
||||
desired = min(desired, remaining_capacity, len(pool))
|
||||
if desired <= 0:
|
||||
self.output_func("Dual Lands: No capacity after trimming; skipping.")
|
||||
|
|
@ -146,7 +146,7 @@ class LandDualsMixin:
|
|||
rng = getattr(self, 'rng', None)
|
||||
try:
|
||||
if rng:
|
||||
rng.shuffle(bucket_keys) # type: ignore
|
||||
rng.shuffle(bucket_keys)
|
||||
else:
|
||||
random.shuffle(bucket_keys)
|
||||
except Exception:
|
||||
|
|
@ -171,7 +171,7 @@ class LandDualsMixin:
|
|||
break
|
||||
added: List[str] = []
|
||||
for name in chosen:
|
||||
if self._current_land_count() >= land_target: # type: ignore[attr-defined]
|
||||
if self._current_land_count() >= land_target:
|
||||
break
|
||||
# Determine sub_role as concatenated color pair for traceability
|
||||
try:
|
||||
|
|
@ -198,7 +198,7 @@ class LandDualsMixin:
|
|||
role='dual',
|
||||
sub_role=sub_role,
|
||||
added_by='lands_step5'
|
||||
) # type: ignore[attr-defined]
|
||||
)
|
||||
added.append(name)
|
||||
self.output_func("\nDual Lands Added (Step 5):")
|
||||
if not added:
|
||||
|
|
@ -207,11 +207,11 @@ class LandDualsMixin:
|
|||
width = max(len(n) for n in added)
|
||||
for n in added:
|
||||
self.output_func(f" {n.ljust(width)} : 1")
|
||||
self.output_func(f" Land Count Now : {self._current_land_count()} / {land_target}") # type: ignore[attr-defined]
|
||||
self.output_func(f" Land Count Now : {self._current_land_count()} / {land_target}")
|
||||
|
||||
def run_land_step5(self, requested_count: int | None = None): # type: ignore[override]
|
||||
def run_land_step5(self, requested_count: int | None = None):
|
||||
self.add_dual_lands(requested_count=requested_count)
|
||||
self._enforce_land_cap(step_label="Duals (Step 5)") # type: ignore[attr-defined]
|
||||
self._enforce_land_cap(step_label="Duals (Step 5)")
|
||||
try:
|
||||
from .. import builder_utils as _bu
|
||||
_bu.export_current_land_pool(self, '5')
|
||||
|
|
|
|||
|
|
@ -19,7 +19,7 @@ Host DeckBuilder must supply:
|
|||
"""
|
||||
|
||||
class LandFetchMixin:
|
||||
def add_fetch_lands(self, requested_count: int | None = None): # type: ignore[override]
|
||||
def add_fetch_lands(self, requested_count: int | None = None):
|
||||
"""Add fetch lands (color-specific + generic) respecting land target."""
|
||||
if not getattr(self, 'files_to_load', []):
|
||||
try:
|
||||
|
|
@ -28,8 +28,8 @@ class LandFetchMixin:
|
|||
except Exception as e: # pragma: no cover - defensive
|
||||
self.output_func(f"Cannot add fetch lands until color identity resolved: {e}")
|
||||
return
|
||||
land_target = (getattr(self, 'ideal_counts', {}).get('lands') if getattr(self, 'ideal_counts', None) else None) or getattr(bc, 'DEFAULT_LAND_COUNT', 35) # type: ignore[attr-defined]
|
||||
current = self._current_land_count() # type: ignore[attr-defined]
|
||||
land_target = (getattr(self, 'ideal_counts', {}).get('lands') if getattr(self, 'ideal_counts', None) else None) or getattr(bc, 'DEFAULT_LAND_COUNT', 35)
|
||||
current = self._current_land_count()
|
||||
color_order = [c for c in getattr(self, 'color_identity', []) if c in ['W','U','B','R','G']]
|
||||
color_map = getattr(bc, 'COLOR_TO_FETCH_LANDS', {})
|
||||
candidates: List[str] = []
|
||||
|
|
@ -56,7 +56,7 @@ class LandFetchMixin:
|
|||
self.output_func("\nAdd Fetch Lands (Step 4):")
|
||||
self.output_func("Fetch lands help fix colors & enable landfall / graveyard synergies.")
|
||||
prompt = f"Enter desired number of fetch lands (default: {effective_default}):"
|
||||
desired = self._prompt_int_with_default(prompt + ' ', effective_default, minimum=0, maximum=20) # type: ignore[attr-defined]
|
||||
desired = self._prompt_int_with_default(prompt + ' ', effective_default, minimum=0, maximum=20)
|
||||
else:
|
||||
desired = max(0, int(requested_count))
|
||||
if desired > remaining_fetch_slots:
|
||||
|
|
@ -70,20 +70,20 @@ class LandFetchMixin:
|
|||
if remaining_capacity == 0 and desired > 0:
|
||||
min_basic_cfg = getattr(bc, 'DEFAULT_BASIC_LAND_COUNT', 20)
|
||||
if getattr(self, 'ideal_counts', None):
|
||||
min_basic_cfg = self.ideal_counts.get('basic_lands', min_basic_cfg) # type: ignore[attr-defined]
|
||||
floor_basics = self._basic_floor(min_basic_cfg) # type: ignore[attr-defined]
|
||||
min_basic_cfg = self.ideal_counts.get('basic_lands', min_basic_cfg)
|
||||
floor_basics = self._basic_floor(min_basic_cfg)
|
||||
slots_needed = desired
|
||||
while slots_needed > 0 and self._count_basic_lands() > floor_basics: # type: ignore[attr-defined]
|
||||
target_basic = self._choose_basic_to_trim() # type: ignore[attr-defined]
|
||||
if not target_basic or not self._decrement_card(target_basic): # type: ignore[attr-defined]
|
||||
while slots_needed > 0 and self._count_basic_lands() > floor_basics:
|
||||
target_basic = self._choose_basic_to_trim()
|
||||
if not target_basic or not self._decrement_card(target_basic):
|
||||
break
|
||||
slots_needed -= 1
|
||||
remaining_capacity = max(0, land_target - self._current_land_count()) # type: ignore[attr-defined]
|
||||
remaining_capacity = max(0, land_target - self._current_land_count())
|
||||
if remaining_capacity > 0 and slots_needed == 0:
|
||||
break
|
||||
if slots_needed > 0 and remaining_capacity == 0:
|
||||
desired -= slots_needed
|
||||
remaining_capacity = max(0, land_target - self._current_land_count()) # type: ignore[attr-defined]
|
||||
remaining_capacity = max(0, land_target - self._current_land_count())
|
||||
desired = min(desired, remaining_capacity, len(candidates), remaining_fetch_slots)
|
||||
if desired <= 0:
|
||||
self.output_func("Fetch Lands: No capacity (after trimming) or desired reduced to 0; skipping.")
|
||||
|
|
@ -101,7 +101,7 @@ class LandFetchMixin:
|
|||
if k >= len(pool):
|
||||
return pool.copy()
|
||||
try:
|
||||
return (rng.sample if rng else random.sample)(pool, k) # type: ignore
|
||||
return (rng.sample if rng else random.sample)(pool, k)
|
||||
except Exception:
|
||||
return pool[:k]
|
||||
need = desired
|
||||
|
|
@ -117,7 +117,7 @@ class LandFetchMixin:
|
|||
|
||||
added: List[str] = []
|
||||
for nm in chosen:
|
||||
if self._current_land_count() >= land_target: # type: ignore[attr-defined]
|
||||
if self._current_land_count() >= land_target:
|
||||
break
|
||||
note = 'generic' if nm in generic_list else 'color-specific'
|
||||
self.add_card(
|
||||
|
|
@ -126,11 +126,11 @@ class LandFetchMixin:
|
|||
role='fetch',
|
||||
sub_role=note,
|
||||
added_by='lands_step4'
|
||||
) # type: ignore[attr-defined]
|
||||
)
|
||||
added.append(nm)
|
||||
# Record actual number of fetch lands added for export/replay context
|
||||
try:
|
||||
setattr(self, 'fetch_count', len(added)) # type: ignore[attr-defined]
|
||||
setattr(self, 'fetch_count', len(added))
|
||||
except Exception:
|
||||
pass
|
||||
self.output_func("\nFetch Lands Added (Step 4):")
|
||||
|
|
@ -141,9 +141,9 @@ class LandFetchMixin:
|
|||
for n in added:
|
||||
note = 'generic' if n in generic_list else 'color-specific'
|
||||
self.output_func(f" {n.ljust(width)} : 1 ({note})")
|
||||
self.output_func(f" Land Count Now : {self._current_land_count()} / {land_target}") # type: ignore[attr-defined]
|
||||
self.output_func(f" Land Count Now : {self._current_land_count()} / {land_target}")
|
||||
|
||||
def run_land_step4(self, requested_count: int | None = None): # type: ignore[override]
|
||||
def run_land_step4(self, requested_count: int | None = None):
|
||||
"""Public wrapper to add fetch lands.
|
||||
|
||||
If ideal_counts['fetch_lands'] is set, it will be used to bypass the prompt in both CLI and web builds.
|
||||
|
|
@ -155,7 +155,7 @@ class LandFetchMixin:
|
|||
except Exception:
|
||||
desired = requested_count
|
||||
self.add_fetch_lands(requested_count=desired)
|
||||
self._enforce_land_cap(step_label="Fetch (Step 4)") # type: ignore[attr-defined]
|
||||
self._enforce_land_cap(step_label="Fetch (Step 4)")
|
||||
try:
|
||||
from .. import builder_utils as _bu
|
||||
_bu.export_current_land_pool(self, '4')
|
||||
|
|
|
|||
|
|
@ -20,7 +20,7 @@ Host DeckBuilder must provide:
|
|||
"""
|
||||
|
||||
class LandKindredMixin:
|
||||
def add_kindred_lands(self): # type: ignore[override]
|
||||
def add_kindred_lands(self):
|
||||
"""Add kindred-oriented lands ONLY if a selected tag includes 'Kindred' or 'Tribal'.
|
||||
|
||||
Baseline inclusions on kindred focus:
|
||||
|
|
@ -41,32 +41,32 @@ class LandKindredMixin:
|
|||
self.output_func("Kindred Lands: No selected kindred/tribal tag; skipping.")
|
||||
return
|
||||
if hasattr(self, 'ideal_counts') and getattr(self, 'ideal_counts'):
|
||||
land_target = self.ideal_counts.get('lands', getattr(bc, 'DEFAULT_LAND_COUNT', 35)) # type: ignore[attr-defined]
|
||||
land_target = self.ideal_counts.get('lands', getattr(bc, 'DEFAULT_LAND_COUNT', 35))
|
||||
else:
|
||||
land_target = getattr(bc, 'DEFAULT_LAND_COUNT', 35)
|
||||
min_basic_cfg = getattr(bc, 'DEFAULT_BASIC_LAND_COUNT', 20)
|
||||
if hasattr(self, 'ideal_counts') and getattr(self, 'ideal_counts'):
|
||||
min_basic_cfg = self.ideal_counts.get('basic_lands', min_basic_cfg) # type: ignore[attr-defined]
|
||||
basic_floor = self._basic_floor(min_basic_cfg) # type: ignore[attr-defined]
|
||||
min_basic_cfg = self.ideal_counts.get('basic_lands', min_basic_cfg)
|
||||
basic_floor = self._basic_floor(min_basic_cfg)
|
||||
|
||||
def ensure_capacity() -> bool:
|
||||
if self._current_land_count() < land_target: # type: ignore[attr-defined]
|
||||
if self._current_land_count() < land_target:
|
||||
return True
|
||||
if self._count_basic_lands() <= basic_floor: # type: ignore[attr-defined]
|
||||
if self._count_basic_lands() <= basic_floor:
|
||||
return False
|
||||
target_basic = self._choose_basic_to_trim() # type: ignore[attr-defined]
|
||||
target_basic = self._choose_basic_to_trim()
|
||||
if not target_basic:
|
||||
return False
|
||||
if not self._decrement_card(target_basic): # type: ignore[attr-defined]
|
||||
if not self._decrement_card(target_basic):
|
||||
return False
|
||||
return self._current_land_count() < land_target # type: ignore[attr-defined]
|
||||
return self._current_land_count() < land_target
|
||||
|
||||
colors = getattr(self, 'color_identity', []) or []
|
||||
added: List[str] = []
|
||||
reasons: Dict[str, str] = {}
|
||||
|
||||
def try_add(name: str, reason: str):
|
||||
if name in self.card_library: # type: ignore[attr-defined]
|
||||
if name in self.card_library:
|
||||
return
|
||||
if not ensure_capacity():
|
||||
return
|
||||
|
|
@ -77,7 +77,7 @@ class LandKindredMixin:
|
|||
sub_role='baseline' if reason.startswith('kindred focus') else 'tribe-specific',
|
||||
added_by='lands_step3',
|
||||
trigger_tag='Kindred/Tribal'
|
||||
) # type: ignore[attr-defined]
|
||||
)
|
||||
added.append(name)
|
||||
reasons[name] = reason
|
||||
|
||||
|
|
@ -105,14 +105,14 @@ class LandKindredMixin:
|
|||
if snapshot is not None and not snapshot.empty and tribe_terms:
|
||||
dynamic_limit = 5
|
||||
for tribe in sorted(tribe_terms):
|
||||
if self._current_land_count() >= land_target or dynamic_limit <= 0: # type: ignore[attr-defined]
|
||||
if self._current_land_count() >= land_target or dynamic_limit <= 0:
|
||||
break
|
||||
tribe_lower = tribe.lower()
|
||||
matches: List[str] = []
|
||||
for _, row in snapshot.iterrows():
|
||||
try:
|
||||
nm = str(row.get('name', ''))
|
||||
if not nm or nm in self.card_library: # type: ignore[attr-defined]
|
||||
if not nm or nm in self.card_library:
|
||||
continue
|
||||
tline = str(row.get('type', row.get('type_line', ''))).lower()
|
||||
if 'land' not in tline:
|
||||
|
|
@ -125,7 +125,7 @@ class LandKindredMixin:
|
|||
except Exception:
|
||||
continue
|
||||
for nm in matches[:2]:
|
||||
if self._current_land_count() >= land_target or dynamic_limit <= 0: # type: ignore[attr-defined]
|
||||
if self._current_land_count() >= land_target or dynamic_limit <= 0:
|
||||
break
|
||||
if nm in added or nm in getattr(bc, 'BASIC_LANDS', []):
|
||||
continue
|
||||
|
|
@ -139,12 +139,12 @@ class LandKindredMixin:
|
|||
width = max(len(n) for n in added)
|
||||
for n in added:
|
||||
self.output_func(f" {n.ljust(width)} : 1 ({reasons.get(n,'')})")
|
||||
self.output_func(f" Land Count Now : {self._current_land_count()} / {land_target}") # type: ignore[attr-defined]
|
||||
self.output_func(f" Land Count Now : {self._current_land_count()} / {land_target}")
|
||||
|
||||
def run_land_step3(self): # type: ignore[override]
|
||||
def run_land_step3(self):
|
||||
"""Public wrapper to add kindred-focused lands."""
|
||||
self.add_kindred_lands()
|
||||
self._enforce_land_cap(step_label="Kindred (Step 3)") # type: ignore[attr-defined]
|
||||
self._enforce_land_cap(step_label="Kindred (Step 3)")
|
||||
try:
|
||||
from .. import builder_utils as _bu
|
||||
_bu.export_current_land_pool(self, '3')
|
||||
|
|
|
|||
|
|
@ -19,7 +19,7 @@ class LandMiscUtilityMixin:
|
|||
- Diagnostics & CSV exports
|
||||
"""
|
||||
|
||||
def add_misc_utility_lands(self, requested_count: Optional[int] = None): # type: ignore[override]
|
||||
def add_misc_utility_lands(self, requested_count: Optional[int] = None):
|
||||
# --- Initialization & candidate collection ---
|
||||
if not getattr(self, 'files_to_load', None):
|
||||
try:
|
||||
|
|
@ -293,7 +293,7 @@ class LandMiscUtilityMixin:
|
|||
if getattr(self, 'show_diagnostics', False) and filtered_out:
|
||||
self.output_func(f" (Mono-color excluded candidates: {', '.join(filtered_out)})")
|
||||
|
||||
def run_land_step7(self, requested_count: Optional[int] = None): # type: ignore[override]
|
||||
def run_land_step7(self, requested_count: Optional[int] = None):
|
||||
self.add_misc_utility_lands(requested_count=requested_count)
|
||||
self._enforce_land_cap(step_label="Utility (Step 7)")
|
||||
self._build_tag_driven_land_suggestions()
|
||||
|
|
@ -305,12 +305,12 @@ class LandMiscUtilityMixin:
|
|||
pass
|
||||
|
||||
# ---- Tag-driven suggestion helpers (used after Step 7) ----
|
||||
def _build_tag_driven_land_suggestions(self): # type: ignore[override]
|
||||
def _build_tag_driven_land_suggestions(self):
|
||||
suggestions = bu.build_tag_driven_suggestions(self)
|
||||
if suggestions:
|
||||
self.suggested_lands_queue.extend(suggestions)
|
||||
|
||||
def _apply_land_suggestions_if_room(self): # type: ignore[override]
|
||||
def _apply_land_suggestions_if_room(self):
|
||||
if not self.suggested_lands_queue:
|
||||
return
|
||||
land_target = getattr(self, 'ideal_counts', {}).get('lands', getattr(bc, 'DEFAULT_LAND_COUNT', 35)) if getattr(self, 'ideal_counts', None) else getattr(bc, 'DEFAULT_LAND_COUNT', 35)
|
||||
|
|
|
|||
|
|
@ -12,7 +12,7 @@ class LandOptimizationMixin:
|
|||
Provides optimize_tapped_lands and run_land_step8 (moved from monolithic builder).
|
||||
"""
|
||||
|
||||
def optimize_tapped_lands(self): # type: ignore[override]
|
||||
def optimize_tapped_lands(self):
|
||||
df = getattr(self, '_combined_cards_df', None)
|
||||
if df is None or df.empty:
|
||||
return
|
||||
|
|
@ -146,7 +146,7 @@ class LandOptimizationMixin:
|
|||
new_tapped += 1
|
||||
self.output_func(f" Tapped Lands After : {new_tapped} (threshold {threshold})")
|
||||
|
||||
def run_land_step8(self): # type: ignore[override]
|
||||
def run_land_step8(self):
|
||||
self.optimize_tapped_lands()
|
||||
self._enforce_land_cap(step_label="Tapped Opt (Step 8)")
|
||||
if self.color_source_matrix_baseline is None:
|
||||
|
|
|
|||
|
|
@ -27,10 +27,10 @@ class LandStaplesMixin:
|
|||
# ---------------------------
|
||||
# Land Building Step 2: Staple Nonbasic Lands (NO Kindred yet)
|
||||
# ---------------------------
|
||||
def _current_land_count(self) -> int: # type: ignore[override]
|
||||
def _current_land_count(self) -> int:
|
||||
"""Return total number of land cards currently in the library (counts duplicates)."""
|
||||
total = 0
|
||||
for name, entry in self.card_library.items(): # type: ignore[attr-defined]
|
||||
for name, entry in self.card_library.items():
|
||||
ctype = entry.get('Card Type', '')
|
||||
if ctype and 'land' in ctype.lower():
|
||||
total += entry.get('Count', 1)
|
||||
|
|
@ -47,7 +47,7 @@ class LandStaplesMixin:
|
|||
continue
|
||||
return total
|
||||
|
||||
def add_staple_lands(self): # type: ignore[override]
|
||||
def add_staple_lands(self):
|
||||
"""Add generic staple lands defined in STAPLE_LAND_CONDITIONS (excluding kindred lands).
|
||||
|
||||
Respects total land target (ideal_counts['lands']). Skips additions once target reached.
|
||||
|
|
@ -62,25 +62,25 @@ class LandStaplesMixin:
|
|||
return
|
||||
land_target = None
|
||||
if hasattr(self, 'ideal_counts') and getattr(self, 'ideal_counts'):
|
||||
land_target = self.ideal_counts.get('lands') # type: ignore[attr-defined]
|
||||
land_target = self.ideal_counts.get('lands')
|
||||
if land_target is None:
|
||||
land_target = getattr(bc, 'DEFAULT_LAND_COUNT', 35)
|
||||
min_basic_cfg = getattr(bc, 'DEFAULT_BASIC_LAND_COUNT', 20)
|
||||
if hasattr(self, 'ideal_counts') and getattr(self, 'ideal_counts'):
|
||||
min_basic_cfg = self.ideal_counts.get('basic_lands', min_basic_cfg) # type: ignore[attr-defined]
|
||||
basic_floor = self._basic_floor(min_basic_cfg) # type: ignore[attr-defined]
|
||||
min_basic_cfg = self.ideal_counts.get('basic_lands', min_basic_cfg)
|
||||
basic_floor = self._basic_floor(min_basic_cfg)
|
||||
|
||||
def ensure_capacity() -> bool:
|
||||
if self._current_land_count() < land_target: # type: ignore[attr-defined]
|
||||
if self._current_land_count() < land_target:
|
||||
return True
|
||||
if self._count_basic_lands() <= basic_floor: # type: ignore[attr-defined]
|
||||
if self._count_basic_lands() <= basic_floor:
|
||||
return False
|
||||
target_basic = self._choose_basic_to_trim() # type: ignore[attr-defined]
|
||||
target_basic = self._choose_basic_to_trim()
|
||||
if not target_basic:
|
||||
return False
|
||||
if not self._decrement_card(target_basic): # type: ignore[attr-defined]
|
||||
if not self._decrement_card(target_basic):
|
||||
return False
|
||||
return self._current_land_count() < land_target # type: ignore[attr-defined]
|
||||
return self._current_land_count() < land_target
|
||||
|
||||
commander_tags_all = set(getattr(self, 'commander_tags', []) or []) | set(getattr(self, 'selected_tags', []) or [])
|
||||
colors = getattr(self, 'color_identity', []) or []
|
||||
|
|
@ -102,7 +102,7 @@ class LandStaplesMixin:
|
|||
if not ensure_capacity():
|
||||
self.output_func("Staple Lands: Cannot free capacity without violating basic floor; stopping additions.")
|
||||
break
|
||||
if land_name in self.card_library: # type: ignore[attr-defined]
|
||||
if land_name in self.card_library:
|
||||
continue
|
||||
try:
|
||||
include = cond(list(commander_tags_all), colors, commander_power)
|
||||
|
|
@ -115,7 +115,7 @@ class LandStaplesMixin:
|
|||
role='staple',
|
||||
sub_role='generic-staple',
|
||||
added_by='lands_step2'
|
||||
) # type: ignore[attr-defined]
|
||||
)
|
||||
added.append(land_name)
|
||||
if land_name == 'Command Tower':
|
||||
reasons[land_name] = f"multi-color ({len(colors)} colors)"
|
||||
|
|
@ -137,12 +137,12 @@ class LandStaplesMixin:
|
|||
for n in added:
|
||||
reason = reasons.get(n, '')
|
||||
self.output_func(f" {n.ljust(width)} : 1 {('(' + reason + ')') if reason else ''}")
|
||||
self.output_func(f" Land Count Now : {self._current_land_count()} / {land_target}") # type: ignore[attr-defined]
|
||||
self.output_func(f" Land Count Now : {self._current_land_count()} / {land_target}")
|
||||
|
||||
def run_land_step2(self): # type: ignore[override]
|
||||
def run_land_step2(self):
|
||||
"""Public wrapper for adding generic staple nonbasic lands (excluding kindred)."""
|
||||
self.add_staple_lands()
|
||||
self._enforce_land_cap(step_label="Staples (Step 2)") # type: ignore[attr-defined]
|
||||
self._enforce_land_cap(step_label="Staples (Step 2)")
|
||||
try:
|
||||
from .. import builder_utils as _bu
|
||||
_bu.export_current_land_pool(self, '2')
|
||||
|
|
|
|||
|
|
@ -59,7 +59,7 @@ class LandTripleMixin:
|
|||
'forest': 'G',
|
||||
}
|
||||
|
||||
for _, row in df.iterrows(): # type: ignore
|
||||
for _, row in df.iterrows():
|
||||
try:
|
||||
name = str(row.get('name',''))
|
||||
if not name or name in self.card_library:
|
||||
|
|
|
|||
|
|
@ -33,7 +33,7 @@ class CreatureAdditionMixin:
|
|||
self.output_func("Card pool missing 'type' column; cannot add creatures.")
|
||||
return
|
||||
try:
|
||||
context = self.get_theme_context() # type: ignore[attr-defined]
|
||||
context = self.get_theme_context()
|
||||
except Exception:
|
||||
context = None
|
||||
if context is None or not getattr(context, 'ordered_targets', []):
|
||||
|
|
@ -480,7 +480,7 @@ class CreatureAdditionMixin:
|
|||
drop_idx = tags_series.apply(lambda lst, nd=needles: any(any(n in t for n in nd) for t in lst))
|
||||
mask_keep = [mk and (not di) for mk, di in zip(mask_keep, drop_idx.tolist())]
|
||||
try:
|
||||
import pandas as _pd # type: ignore
|
||||
import pandas as _pd
|
||||
mask_keep = _pd.Series(mask_keep, index=df.index)
|
||||
except Exception:
|
||||
pass
|
||||
|
|
|
|||
|
|
@ -78,7 +78,7 @@ class SpellAdditionMixin:
|
|||
# Combine into keep mask
|
||||
mask_keep = [mk and (not di) for mk, di in zip(mask_keep, drop_idx.tolist())]
|
||||
try:
|
||||
import pandas as _pd # type: ignore
|
||||
import pandas as _pd
|
||||
mask_keep = _pd.Series(mask_keep, index=df.index)
|
||||
except Exception:
|
||||
pass
|
||||
|
|
@ -742,7 +742,7 @@ class SpellAdditionMixin:
|
|||
if df is None or df.empty or 'type' not in df.columns:
|
||||
return
|
||||
try:
|
||||
context = self.get_theme_context() # type: ignore[attr-defined]
|
||||
context = self.get_theme_context()
|
||||
except Exception:
|
||||
context = None
|
||||
if context is None or not getattr(context, 'ordered_targets', []):
|
||||
|
|
|
|||
|
|
@ -14,7 +14,7 @@ from ..shared_copy import build_land_headline, dfc_card_note
|
|||
logger = logging_util.logging.getLogger(__name__)
|
||||
|
||||
try:
|
||||
from prettytable import PrettyTable # type: ignore
|
||||
from prettytable import PrettyTable
|
||||
except Exception: # pragma: no cover
|
||||
PrettyTable = None # type: ignore
|
||||
|
||||
|
|
@ -176,7 +176,7 @@ class ReportingMixin:
|
|||
"""
|
||||
try:
|
||||
# Lazy import to avoid cycles
|
||||
from deck_builder.enforcement import enforce_bracket_compliance # type: ignore
|
||||
from deck_builder.enforcement import enforce_bracket_compliance
|
||||
except Exception:
|
||||
self.output_func("Enforcement module unavailable.")
|
||||
return {}
|
||||
|
|
@ -194,7 +194,7 @@ class ReportingMixin:
|
|||
if int(total_cards) < 100 and hasattr(self, 'fill_remaining_theme_spells'):
|
||||
before = int(total_cards)
|
||||
try:
|
||||
self.fill_remaining_theme_spells() # type: ignore[attr-defined]
|
||||
self.fill_remaining_theme_spells()
|
||||
except Exception:
|
||||
pass
|
||||
# Recompute after filler
|
||||
|
|
@ -239,13 +239,13 @@ class ReportingMixin:
|
|||
csv_name = base_stem + ".csv"
|
||||
txt_name = base_stem + ".txt"
|
||||
# Overwrite exports with updated library
|
||||
self.export_decklist_csv(directory='deck_files', filename=csv_name, suppress_output=True) # type: ignore[attr-defined]
|
||||
self.export_decklist_text(directory='deck_files', filename=txt_name, suppress_output=True) # type: ignore[attr-defined]
|
||||
self.export_decklist_csv(directory='deck_files', filename=csv_name, suppress_output=True)
|
||||
self.export_decklist_text(directory='deck_files', filename=txt_name, suppress_output=True)
|
||||
# Re-export the JSON config to reflect any changes from enforcement
|
||||
json_name = base_stem + ".json"
|
||||
self.export_run_config_json(directory='config', filename=json_name, suppress_output=True) # type: ignore[attr-defined]
|
||||
self.export_run_config_json(directory='config', filename=json_name, suppress_output=True)
|
||||
# Recompute and write compliance next to them
|
||||
self.compute_and_print_compliance(base_stem=base_stem) # type: ignore[attr-defined]
|
||||
self.compute_and_print_compliance(base_stem=base_stem)
|
||||
# Inject enforcement details into the saved compliance JSON for UI transparency
|
||||
comp_path = _os.path.join('deck_files', f"{base_stem}_compliance.json")
|
||||
try:
|
||||
|
|
@ -259,18 +259,18 @@ class ReportingMixin:
|
|||
pass
|
||||
else:
|
||||
# Fall back to default export flow
|
||||
csv_path = self.export_decklist_csv() # type: ignore[attr-defined]
|
||||
csv_path = self.export_decklist_csv()
|
||||
try:
|
||||
base, _ = _os.path.splitext(csv_path)
|
||||
base_only = _os.path.basename(base)
|
||||
except Exception:
|
||||
base_only = None
|
||||
self.export_decklist_text(filename=(base_only + '.txt') if base_only else None) # type: ignore[attr-defined]
|
||||
self.export_decklist_text(filename=(base_only + '.txt') if base_only else None)
|
||||
# Re-export JSON config after enforcement changes
|
||||
if base_only:
|
||||
self.export_run_config_json(directory='config', filename=base_only + '.json', suppress_output=True) # type: ignore[attr-defined]
|
||||
self.export_run_config_json(directory='config', filename=base_only + '.json', suppress_output=True)
|
||||
if base_only:
|
||||
self.compute_and_print_compliance(base_stem=base_only) # type: ignore[attr-defined]
|
||||
self.compute_and_print_compliance(base_stem=base_only)
|
||||
# Inject enforcement into written JSON as above
|
||||
try:
|
||||
comp_path = _os.path.join('deck_files', f"{base_only}_compliance.json")
|
||||
|
|
@ -294,7 +294,7 @@ class ReportingMixin:
|
|||
"""
|
||||
try:
|
||||
# Late import to avoid circulars in some environments
|
||||
from deck_builder.brackets_compliance import evaluate_deck # type: ignore
|
||||
from deck_builder.brackets_compliance import evaluate_deck
|
||||
except Exception:
|
||||
self.output_func("Bracket compliance module unavailable.")
|
||||
return {}
|
||||
|
|
@ -373,7 +373,7 @@ class ReportingMixin:
|
|||
full_df = getattr(self, '_full_cards_df', None)
|
||||
combined_df = getattr(self, '_combined_cards_df', None)
|
||||
snapshot = full_df if full_df is not None else combined_df
|
||||
row_lookup: Dict[str, any] = {}
|
||||
row_lookup: Dict[str, Any] = {}
|
||||
if snapshot is not None and hasattr(snapshot, 'empty') and not snapshot.empty and 'name' in snapshot.columns:
|
||||
for _, r in snapshot.iterrows():
|
||||
nm = str(r.get('name'))
|
||||
|
|
@ -429,7 +429,7 @@ class ReportingMixin:
|
|||
|
||||
# Surface land vs. MDFC counts for CLI users to mirror web summary copy
|
||||
try:
|
||||
summary = self.build_deck_summary() # type: ignore[attr-defined]
|
||||
summary = self.build_deck_summary()
|
||||
except Exception:
|
||||
summary = None
|
||||
if isinstance(summary, dict):
|
||||
|
|
@ -483,9 +483,9 @@ class ReportingMixin:
|
|||
full_df = getattr(self, '_full_cards_df', None)
|
||||
combined_df = getattr(self, '_combined_cards_df', None)
|
||||
snapshot = full_df if full_df is not None else combined_df
|
||||
row_lookup: Dict[str, any] = {}
|
||||
row_lookup: Dict[str, Any] = {}
|
||||
if snapshot is not None and not getattr(snapshot, 'empty', True) and 'name' in snapshot.columns:
|
||||
for _, r in snapshot.iterrows(): # type: ignore[attr-defined]
|
||||
for _, r in snapshot.iterrows():
|
||||
nm = str(r.get('name'))
|
||||
if nm and nm not in row_lookup:
|
||||
row_lookup[nm] = r
|
||||
|
|
@ -521,7 +521,7 @@ class ReportingMixin:
|
|||
|
||||
builder_utils_module = None
|
||||
try:
|
||||
from deck_builder import builder_utils as _builder_utils # type: ignore
|
||||
from deck_builder import builder_utils as _builder_utils
|
||||
builder_utils_module = _builder_utils
|
||||
color_matrix = builder_utils_module.compute_color_source_matrix(self.card_library, full_df)
|
||||
except Exception:
|
||||
|
|
@ -856,7 +856,7 @@ class ReportingMixin:
|
|||
full_df = getattr(self, '_full_cards_df', None)
|
||||
combined_df = getattr(self, '_combined_cards_df', None)
|
||||
snapshot = full_df if full_df is not None else combined_df
|
||||
row_lookup: Dict[str, any] = {}
|
||||
row_lookup: Dict[str, Any] = {}
|
||||
if snapshot is not None and not snapshot.empty and 'name' in snapshot.columns:
|
||||
for _, r in snapshot.iterrows():
|
||||
nm = str(r.get('name'))
|
||||
|
|
@ -1128,7 +1128,7 @@ class ReportingMixin:
|
|||
full_df = getattr(self, '_full_cards_df', None)
|
||||
combined_df = getattr(self, '_combined_cards_df', None)
|
||||
snapshot = full_df if full_df is not None else combined_df
|
||||
row_lookup: Dict[str, any] = {}
|
||||
row_lookup: Dict[str, Any] = {}
|
||||
if snapshot is not None and not snapshot.empty and 'name' in snapshot.columns:
|
||||
for _, r in snapshot.iterrows():
|
||||
nm = str(r.get('name'))
|
||||
|
|
@ -1136,7 +1136,7 @@ class ReportingMixin:
|
|||
row_lookup[nm] = r
|
||||
|
||||
try:
|
||||
from deck_builder import builder_utils as _builder_utils # type: ignore
|
||||
from deck_builder import builder_utils as _builder_utils
|
||||
color_matrix = _builder_utils.compute_color_source_matrix(self.card_library, full_df)
|
||||
except Exception:
|
||||
color_matrix = {}
|
||||
|
|
@ -1387,3 +1387,4 @@ class ReportingMixin:
|
|||
"""
|
||||
# Card library printout suppressed; use CSV and text export for card list.
|
||||
pass
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue