Select the moveset with the best average estimator

This commit is contained in:
Zoé Cassiopée Gauthier 2024-04-11 15:55:29 -04:00
parent 4c8d351e61
commit 8e1952072d
3 changed files with 75 additions and 35 deletions

View File

@ -4,18 +4,26 @@
# license that can be found in the LICENSE file or at # license that can be found in the LICENSE file or at
# https://opensource.org/licenses/MIT. # https://opensource.org/licenses/MIT.
import math
from enum import Flag, auto
from typing import final from typing import final
from rich.progress import Progress from rich.progress import Progress
from pogo_scaled_estimators.pokebattler_proxy import Moveset, PokebattlerProxy, Raid from pogo_scaled_estimators.pokebattler_proxy import MovesetResult, PokebattlerProxy, Raid
from pogo_scaled_estimators.utilities import format_move_name, format_pokemon_name from pogo_scaled_estimators.utilities import format_move_name, format_pokemon_name
class Filter(Flag):
NO_FILTER = 0
DISALLOW_LEGACY_MOVES = auto()
@final @final
class Calculator: class Calculator:
def __init__(self, attacker_types: list[str]) -> None: def __init__(self, attacker_types: list[str], filters: Filter = Filter.NO_FILTER) -> None:
self.attacker_types = attacker_types self.attacker_types = attacker_types
self.filters = filters
self._pokebattler_proxy = PokebattlerProxy() self._pokebattler_proxy = PokebattlerProxy()
self._progress = Progress() self._progress = Progress()
@ -43,11 +51,7 @@ class Calculator:
self._progress.update(task, advance=1) self._progress.update(task, advance=1)
raid = Raid(raid_tier, defender, level, party) raid = Raid(raid_tier, defender, level, party)
results = { results = {
attacker: [ attacker: [moveset for moveset in movesets if self._viable_move(attacker, moveset.charged_move)]
moveset
for moveset in movesets
if self._pokebattler_proxy.move_type(moveset.charged_move) in self.attacker_types
]
for attacker, movesets in self._pokebattler_proxy.simulate(raid).items() for attacker, movesets in self._pokebattler_proxy.simulate(raid).items()
if attacker in attackers if attacker in attackers
} }
@ -64,28 +68,50 @@ class Calculator:
ase_by_attacker: list[tuple[str, str, str, float]] = [] ase_by_attacker: list[tuple[str, str, str, float]] = []
for attacker, raid_tier_results in attackers.items(): for attacker, raid_tier_results in attackers.items():
if ( fast_move, charged_move = self._best_moves(raid_tier_results)
not raid_tier_results["RAID_LEVEL_3"] ase = self._ase(raid_tier_results)
or not raid_tier_results["RAID_LEVEL_5"] if not math.isfinite(ase):
or not raid_tier_results["RAID_LEVEL_MEGA"]
):
continue continue
fast_move = raid_tier_results["RAID_LEVEL_5"][0].fast_move ase_by_attacker.append((attacker, fast_move, charged_move, self._ase(raid_tier_results)))
charged_move = raid_tier_results["RAID_LEVEL_5"][0].charged_move
ase = (
0.15 * self._average_estimator(raid_tier_results["RAID_LEVEL_3"])
+ 0.50 * self._average_estimator(raid_tier_results["RAID_LEVEL_5"])
+ 0.35 * self._average_estimator(raid_tier_results["RAID_LEVEL_MEGA"])
)
ase_by_attacker.append((attacker, fast_move, charged_move, ase))
ase_by_attacker.sort(key=lambda item: item[3]) ase_by_attacker.sort(key=lambda item: item[3])
for attacker, fast_move, charged_move, ase in ase_by_attacker: for attacker, fast_move, charged_move, ase in ase_by_attacker:
attacker_type = self._pokebattler_proxy.pokemon_type(attacker) attacker_type = self._pokebattler_proxy.pokemon_type(attacker)
fast_move_type = self._pokebattler_proxy.move_type(fast_move) fast_move_type = self._pokebattler_proxy.find_move(fast_move).typing
charged_move_type = self._pokebattler_proxy.move_type(charged_move) charged_move_type = self._pokebattler_proxy.find_move(charged_move).typing
self._progress.console.print( self._progress.console.print(
f"[bold]{format_pokemon_name(attacker, attacker_type)}[/bold] ({format_move_name(fast_move, fast_move_type)}/{format_move_name(charged_move, charged_move_type)}): {ase:.2f}" f"[bold]{format_pokemon_name(attacker, attacker_type)}[/bold] ({format_move_name(fast_move, fast_move_type)}/{format_move_name(charged_move, charged_move_type)}): {ase:.2f}"
) )
def _average_estimator(self, movesets: list[Moveset]) -> float: def _viable_move(self, pokemon_id: str, move_id: str) -> bool:
return sum(moveset.estimator for moveset in movesets) / len(movesets) move = self._pokebattler_proxy.find_move(move_id)
if move.typing not in self.attacker_types:
return False
if Filter.DISALLOW_LEGACY_MOVES in self.filters:
pokemon = self._pokebattler_proxy.find_pokemon(pokemon_id)
move_id = move_id.removesuffix("_PLUS_PLUS")
if move_id in pokemon.get("eliteQuickMove", []) or move_id in pokemon.get("eliteCinematicMove", []):
return False
return True
def _best_moves(self, moveset_results_by_tier: dict[str, list[MovesetResult]]) -> tuple[str, str]:
movesets: set[tuple[str, str]] = set()
for moveset_results in moveset_results_by_tier.values():
movesets.update((moveset.fast_move, moveset.charged_move) for moveset in moveset_results)
if not movesets:
return ("MOVE_NONE", "MOVE_NONE")
return min(movesets, key=lambda moveset: self._ase(moveset_results_by_tier, only=moveset))
def _ase(self, moveset_results_by_tier: dict[str, list[MovesetResult]], only=None) -> float:
try:
return (
0.15 * self._average_estimator(moveset_results_by_tier["RAID_LEVEL_3"], only)
+ 0.50 * self._average_estimator(moveset_results_by_tier["RAID_LEVEL_5"], only)
+ 0.35 * self._average_estimator(moveset_results_by_tier["RAID_LEVEL_MEGA"], only)
)
except ZeroDivisionError:
return float("inf")
def _average_estimator(self, moveset_results: list[MovesetResult], only: tuple[str, str] | None = None) -> float:
if only:
moveset_results = [m for m in moveset_results if m.fast_move == only[0] and m.charged_move == only[1]]
return sum(moveset.estimator for moveset in moveset_results) / len(moveset_results)

View File

@ -7,7 +7,7 @@
import argparse import argparse
import sys import sys
import pogo_scaled_estimators.calculator from pogo_scaled_estimators.calculator import Calculator, Filter
def main_cli(): def main_cli():
@ -15,9 +15,13 @@ def main_cli():
_ = parser.add_argument("type", nargs="+", help="an attacker type") _ = parser.add_argument("type", nargs="+", help="an attacker type")
_ = parser.add_argument("--level", type=int, default=40) _ = parser.add_argument("--level", type=int, default=40)
_ = parser.add_argument("--party", type=int, default=1) _ = parser.add_argument("--party", type=int, default=1)
_ = parser.add_argument("--no-elite", action="store_true")
args = parser.parse_args() args = parser.parse_args()
calculator = pogo_scaled_estimators.calculator.Calculator(args.type) filters = Filter.NO_FILTER
if args.no_elite:
filters |= Filter.DISALLOW_LEGACY_MOVES
calculator = Calculator(args.type, filters)
calculator.calculate(level=args.level, party=args.party) calculator.calculate(level=args.level, party=args.party)

View File

@ -25,13 +25,19 @@ class Raid:
@dataclass @dataclass
class Moveset: class Move:
move_id: str
typing: str
@dataclass
class MovesetResult:
fast_move: str fast_move: str
charged_move: str charged_move: str
estimator: float estimator: float
def scale(self, factor: float): def scale(self, factor: float):
return Moveset(self.fast_move, self.charged_move, self.estimator / factor) return MovesetResult(self.fast_move, self.charged_move, self.estimator / factor)
@final @final
@ -62,7 +68,7 @@ class PokebattlerProxy:
def resists(self) -> dict: def resists(self) -> dict:
return self.cached_session.get(f"{BASE_URL}/resists").json() return self.cached_session.get(f"{BASE_URL}/resists").json()
def simulate(self, raid: Raid) -> dict[str, list[Moveset]]: def simulate(self, raid: Raid) -> dict[str, list[MovesetResult]]:
query_string = { query_string = {
"sort": "ESTIMATOR", "sort": "ESTIMATOR",
"weatherCondition": "NO_WEATHER", "weatherCondition": "NO_WEATHER",
@ -78,10 +84,10 @@ class PokebattlerProxy:
query_string["friendshipLevel"] = "FRIENDSHIP_LEVEL_4" query_string["friendshipLevel"] = "FRIENDSHIP_LEVEL_4"
url = f"{BASE_URL}/raids/defenders/{raid.defender}/levels/{raid.tier}/attackers/levels/{raid.level}/strategies/CINEMATIC_ATTACK_WHEN_POSSIBLE/DEFENSE_RANDOM_MC?{urllib.parse.urlencode(query_string, doseq=True)}" url = f"{BASE_URL}/raids/defenders/{raid.defender}/levels/{raid.tier}/attackers/levels/{raid.level}/strategies/CINEMATIC_ATTACK_WHEN_POSSIBLE/DEFENSE_RANDOM_MC?{urllib.parse.urlencode(query_string, doseq=True)}"
response = self._cached_session.get(url) response = self._cached_session.get(url)
results: dict[str, list[Moveset]] = {} results: dict[str, list[MovesetResult]] = {}
for attacker in response.json()["attackers"][0]["randomMove"]["defenders"]: for attacker in response.json()["attackers"][0]["randomMove"]["defenders"]:
results[attacker["pokemonId"]] = [ results[attacker["pokemonId"]] = [
Moveset( MovesetResult(
attacker_moves["move1"], attacker_moves["move2"], cast(float, attacker_moves["result"]["estimator"]) attacker_moves["move1"], attacker_moves["move2"], cast(float, attacker_moves["result"]["estimator"])
) )
for attacker_moves in attacker["byMove"] for attacker_moves in attacker["byMove"]
@ -162,8 +168,12 @@ class PokebattlerProxy:
if any(moveset["cinematicMove"] in charged_moves for moveset in mon["movesets"]) if any(moveset["cinematicMove"] in charged_moves for moveset in mon["movesets"])
] ]
def pokemon_type(self, name: str) -> str: def find_pokemon(self, name: str) -> dict:
return next(filter(lambda mon: mon["pokemonId"] == name, self.pokemon))["type"] return next(filter(lambda mon: mon["pokemonId"] == name, self.pokemon))
def move_type(self, name: str) -> str: def pokemon_type(self, name: str) -> str:
return next(filter(lambda move: "moveId" in move and move["moveId"] == name, self.moves))["type"] return self.find_pokemon(name)["type"]
def find_move(self, move_id: str) -> Move:
move = next(filter(lambda move: "moveId" in move and move["moveId"] == move_id, self.moves))
return Move(move_id, move["type"])