Module botroyale.api.time_test
Tool for measuring calculation time of bots.
The timing_test()
function is a convenient way to quickly measure mean and
maximum calculation time of bots. Results are logged (printed to console,
currently) as they are available.
Expand source code Browse git
"""Tool for measuring calculation time of bots.
The `timing_test` function is a convenient way to quickly measure mean and
maximum calculation time of bots. Results are logged (printed to console,
currently) as they are available.
"""
from typing import Optional, Sequence, NamedTuple
import numpy as np
from botroyale.api.logging import logger as glogger
from botroyale.logic.maps import get_map_state
from botroyale.logic.battle_manager import BattleManager
from botroyale.api.bots import BotSelection, BOTS
from botroyale.bots.idle_bot import DummyBot
class TimeResult(NamedTuple):
"""Nametuple with fields (mean, max)."""
mean: float
"""Mean calculation time"""
max: float
"""Max calculation time"""
def timing_test(
bots: Sequence[str],
battle_count: int,
map_name: Optional[str] = None,
shuffle_bots: bool = True,
disable_logging: bool = True,
verbose_results: bool = True,
) -> dict[str, TimeResult]:
"""A timing test for bots.
Plays a number of battles and logs bot calculation times. Asserts each bot
specified in *bots* will play in every game. If more bots than specified are
required, dummy bots will be supplied.
Args:
bots: List of bot names.
battle_count: Number of battles to play.
map_name: Name of map to play on.
shuffle_bots: Automatically shuffle the order of the bots for each battle.
disable_logging: Disable logging during battles.
verbose_results: Show time table of each battle.
Returns:
Dictionary of bot names mapped to a `TimeResult`.
"""
bots = [b for b in bots if BOTS[b].NAME != "dummy"]
battle_index = 0
all_results = {b: TimeResult([], []) for b in bots}
glogger("\n\n========== Timing Test ==========")
glogger("Selected:\n" + "\n".join(f"{i:>2} {b}" for i, b in enumerate(bots)))
for battle_index in range(battle_count):
botselect = BotSelection(bots, all_play=True, max_repeat=1)
battle = BattleManager(
bots=botselect,
initial_state=get_map_state(map_name),
description=f"time test {battle_index+1} / {battle_count} @ {map_name}",
enable_logging=not disable_logging,
)
glogger(f"\nPlaying battle : {battle.description}")
battle.play_all(print_progress=disable_logging)
if verbose_results:
glogger(f"Battle time results:\n{battle.get_timer_str()}")
for uid, bot in enumerate(battle.bots):
if type(bot) is DummyBot:
continue
all_results[bot.NAME].mean.append(battle.bot_timer.mean(uid))
all_results[bot.NAME].max.append(battle.bot_timer.max(uid))
_print_final_results(
_get_final_results(all_results),
battle_index,
battle_count,
)
final_results = _get_final_results(all_results)
return final_results
def _get_final_results(results):
final_results = {}
for bot_name, result in results.items():
final_results[bot_name] = TimeResult(
mean=np.mean(np.asarray(result.mean)),
max=np.max(np.asarray(result.max)),
)
return final_results
def _print_final_results(results, battle_index, battle_count):
glogger("\n\n========== Timing Test Results ==========")
glogger(
"\n".join(
[
f"Played {battle_index+1:>3} / {battle_count:>3} battles.",
"One of each bot played in each game.",
"Mean represents the mean of means given by each game"
"(itself a mean of turns).",
"Max represents the maximum time the bot timed in a single"
"turn over all games.",
"\n",
]
)
)
for bot_name, result in results.items():
glogger(
f'{bot_name:<20} Mean: {f"{result.mean:.3f}":>12} ms/t '
f'Max: {f"{result.max:.3f}":>14} ms'
)
glogger("\n_________________________________________")
Functions
def timing_test(bots: Sequence[str], battle_count: int, map_name: Optional[str] = None, shuffle_bots: bool = True, disable_logging: bool = True, verbose_results: bool = True) ‑> dict[str, TimeResult]
-
A timing test for bots.
Plays a number of battles and logs bot calculation times. Asserts each bot specified in bots will play in every game. If more bots than specified are required, dummy bots will be supplied.
Args
bots
- List of bot names.
battle_count
- Number of battles to play.
map_name
- Name of map to play on.
shuffle_bots
- Automatically shuffle the order of the bots for each battle.
disable_logging
- Disable logging during battles.
verbose_results
- Show time table of each battle.
Returns
Dictionary of bot names mapped to a
TimeResult
.Expand source code Browse git
def timing_test( bots: Sequence[str], battle_count: int, map_name: Optional[str] = None, shuffle_bots: bool = True, disable_logging: bool = True, verbose_results: bool = True, ) -> dict[str, TimeResult]: """A timing test for bots. Plays a number of battles and logs bot calculation times. Asserts each bot specified in *bots* will play in every game. If more bots than specified are required, dummy bots will be supplied. Args: bots: List of bot names. battle_count: Number of battles to play. map_name: Name of map to play on. shuffle_bots: Automatically shuffle the order of the bots for each battle. disable_logging: Disable logging during battles. verbose_results: Show time table of each battle. Returns: Dictionary of bot names mapped to a `TimeResult`. """ bots = [b for b in bots if BOTS[b].NAME != "dummy"] battle_index = 0 all_results = {b: TimeResult([], []) for b in bots} glogger("\n\n========== Timing Test ==========") glogger("Selected:\n" + "\n".join(f"{i:>2} {b}" for i, b in enumerate(bots))) for battle_index in range(battle_count): botselect = BotSelection(bots, all_play=True, max_repeat=1) battle = BattleManager( bots=botselect, initial_state=get_map_state(map_name), description=f"time test {battle_index+1} / {battle_count} @ {map_name}", enable_logging=not disable_logging, ) glogger(f"\nPlaying battle : {battle.description}") battle.play_all(print_progress=disable_logging) if verbose_results: glogger(f"Battle time results:\n{battle.get_timer_str()}") for uid, bot in enumerate(battle.bots): if type(bot) is DummyBot: continue all_results[bot.NAME].mean.append(battle.bot_timer.mean(uid)) all_results[bot.NAME].max.append(battle.bot_timer.max(uid)) _print_final_results( _get_final_results(all_results), battle_index, battle_count, ) final_results = _get_final_results(all_results) return final_results
Classes
class TimeResult (mean: float, max: float)
-
Nametuple with fields (mean, max).
Expand source code Browse git
class TimeResult(NamedTuple): """Nametuple with fields (mean, max).""" mean: float """Mean calculation time""" max: float """Max calculation time"""
Ancestors
- builtins.tuple
Instance variables
var max : float
-
Max calculation time
var mean : float
-
Mean calculation time