From ff8fb5320b1d731e379e8b59fd557bf6a220cef4 Mon Sep 17 00:00:00 2001 From: Peter Kruse Date: Sun, 21 Apr 2024 12:37:02 +0200 Subject: [PATCH] season 4 update --- bloodytools/simulations/simulator.py | 2 +- bloodytools/simulations/tier_set_simulator.py | 10 +- bloodytools/utils/config.py | 6 +- bloodytools/utils/profile_extraction.py | 210 +++++++++++++++++- requirements.txt | 2 +- 5 files changed, 217 insertions(+), 13 deletions(-) diff --git a/bloodytools/simulations/simulator.py b/bloodytools/simulations/simulator.py index 2b37cef..6a357d5 100644 --- a/bloodytools/simulations/simulator.py +++ b/bloodytools/simulations/simulator.py @@ -399,7 +399,7 @@ def get_additional_talent_paths( except EmptyFileError: custom_profile = False - profile_name = "T" + self.settings.tier + profile_name = self.settings.tier if custom_profile: profile_name = "custom profile" diff --git a/bloodytools/simulations/tier_set_simulator.py b/bloodytools/simulations/tier_set_simulator.py index ff06088..5e0c894 100644 --- a/bloodytools/simulations/tier_set_simulator.py +++ b/bloodytools/simulations/tier_set_simulator.py @@ -30,14 +30,16 @@ def add_simulation_data( "set_bonus=tier29_4pc=0", "set_bonus=tier30_2pc=0", "set_bonus=tier30_4pc=0", + "set_bonus=tier31_2pc=0", + "set_bonus=tier31_4pc=0", ], "2p": [ - "set_bonus=tier30_2pc=1", - "set_bonus=tier30_4pc=0", + "set_bonus=tier32_2pc=1", + "set_bonus=tier32_4pc=0", ], "4p": [ - "set_bonus=tier30_2pc=1", - "set_bonus=tier30_4pc=1", + "set_bonus=tier32_2pc=1", + "set_bonus=tier32_4pc=1", ], } diff --git a/bloodytools/utils/config.py b/bloodytools/utils/config.py index a840ffe..520b12f 100644 --- a/bloodytools/utils/config.py +++ b/bloodytools/utils/config.py @@ -30,9 +30,9 @@ class Config: iterations: str = "60000" keep_files: bool = False # affects trinkets - max_ilevel: int = 496 + max_ilevel: int = 528 # affects trinkets - min_ilevel: int = 447 + min_ilevel: int = 480 pretty: bool = False profileset_work_threads: str = "2" ptr: str = "0" @@ -55,7 +55,7 @@ class Config: talent_permutations: bool = False target_error: typing.Dict[str, str] = dataclasses.field(default_factory=dict) threads: str = "" - tier: str = "31" + tier: str = "DF4" use_raidbots: bool = False write_humanreadable_secondary_distribution_file: bool = False apikey: str = "" diff --git a/bloodytools/utils/profile_extraction.py b/bloodytools/utils/profile_extraction.py index 23392a9..632599f 100644 --- a/bloodytools/utils/profile_extraction.py +++ b/bloodytools/utils/profile_extraction.py @@ -1,3 +1,5 @@ +import dataclasses +import enum import logging import os import re @@ -10,6 +12,198 @@ logger = logging.getLogger(__name__) +class CharacterSource(enum.Enum): + SIMULATIONCRAFT = enum.auto() + CUSTOM_PROFILE = enum.auto() + FALLBACK_PROFILE = enum.auto() + + +class ItemSlot(enum.Enum): + HEAD = "head" + NECK = "neck" + SHOULDERS = "shoulders" + BACK = "back" + CHEST = "chest" + WRISTS = "wrists" + HANDS = "hands" + WAIST = "waist" + LEGS = "legs" + FEET = "feet" + FINGER_1 = "finger1" + FINGER_2 = "finger2" + TRINKET_1 = "trinket1" + TRINKET_2 = "trinket2" + MAIN_HAND = "main_hand" + OFF_HAND = "off_hand" + NONE = "none" + + +class NotAnItemLineError(Exception): + pass + + +@dataclasses.dataclass +class Item: + slot: ItemSlot + slot_alternative_names: typing.List[str] + + item_id: int + bonus_id: typing.List[int] + enchant: str + ilevel: int + gem_id: typing.List[int] + enchant_id: int + crafted_stats: typing.List[int] + drop_level: int + + @staticmethod + def from_simc_string(simc_string: str) -> "Item": + simc_string = simc_string.strip() + + # drop comments + simc_string = simc_string.split("#")[0] + + if not simc_string: + raise NotAnItemLineError( + "Empty line found (comments were dropped beforehand)." + ) + + slot: typing.Union[None, ItemSlot] = None + alternative_slot_names: typing.Dict[ItemSlot, str] = { + ItemSlot.SHOULDERS: "shoulder", + ItemSlot.WRISTS: "wrist", + } + for slot_option in ItemSlot: + if ( + simc_string.startswith(slot_option.value) + or slot_option in alternative_slot_names + and simc_string.startswith(alternative_slot_names[slot_option]) + ): + slot = slot_option + + if not slot: + raise NotAnItemLineError("ItemSlot not found in line") + + # dropping slot information + simc_parts = simc_string.split(",")[1:] + + empty_item = Item( + slot=slot, + slot_alternative_names=[], + item_id=-1, + bonus_id=[], + enchant="", + ilevel=-1, + gem_id=[], + enchant_id=-1, + crafted_stats=[], + drop_level=-1, + ) + for part in simc_parts: + # ! stop the implementation! the actual goal is to ensure only one profile is printed in profilesets. head over there + pass + + return empty_item + + +@dataclasses.dataclass +class HeadItem(Item): + slot = ItemSlot.HEAD + + +@dataclasses.dataclass +class NeckItem(Item): + slot = ItemSlot.NECK + + +@dataclasses.dataclass +class ShouldersItem(Item): + slot = ItemSlot.SHOULDERS + + +@dataclasses.dataclass +class BackItem(Item): + slot = ItemSlot.BACK + + +@dataclasses.dataclass +class ChestItem(Item): + slot = ItemSlot.CHEST + + +@dataclasses.dataclass +class WristsItem(Item): + slot = ItemSlot.WRISTS + + +@dataclasses.dataclass +class HandsItem(Item): + slot = ItemSlot.HANDS + + +@dataclasses.dataclass +class WaistItem(Item): + slot = ItemSlot.WAIST + + +@dataclasses.dataclass +class LegsItem(Item): + slot = ItemSlot.LEGS + + +@dataclasses.dataclass +class FeetItem(Item): + slot = ItemSlot.FEET + + +@dataclasses.dataclass +class Finger1Item(Item): + slot = ItemSlot.FINGER_1 + + +@dataclasses.dataclass +class Finger2Item(Item): + slot = ItemSlot.FINGER_2 + + +@dataclasses.dataclass +class Trinket1Item(Item): + slot = ItemSlot.TRINKET_1 + + +@dataclasses.dataclass +class Trinket2Item(Item): + slot = ItemSlot.TRINKET_2 + + +@dataclasses.dataclass +class MainHandItem(Item): + slot = ItemSlot.MAIN_HAND + + +@dataclasses.dataclass +class OffHandItem(Item): + slot = ItemSlot.OFF_HAND + + +@dataclasses.dataclass +class CharacterProfile: + source: str + path: str + + # character + class_str: str + level: str + race: str + role: str + spec: str + + # items + back: BackItem + chest: ChestItem + feet: FeetItem + + class EmptyFileError(Exception): pass @@ -23,13 +217,21 @@ class SpecMismatchError(Exception): def _get_tier_directory_name(tier: str) -> str: - """PreRaids vs TierXX""" - return "PreRaids" if tier == "PR" else f"Tier{tier}" + """ + SimulationCraft switched to expansion + season count as names + + old:PreRaids vs TierXX + """ + return tier def _get_tier_file_name_part(tier: str) -> str: - """PR vs TXX""" - return "PR" if "PR" in str(tier) else f"T{tier}" + """ + SimulationCraft switched to expansion + season count as names + + PR vs TXX + """ + return tier def _get_simc_profile_file_name(tier: str, wow_spec: WowSpec) -> str: diff --git a/requirements.txt b/requirements.txt index d7131a7..3a3fcf2 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,5 +1,5 @@ # Lib with World of Warcraft game data for simulations and some input checks for SimulationCraft. -simc-support~=10.2.0.0 +simc-support~=10.2.6.0 requests # load special cases pyyaml