import random
import traceback
from typing import Iterable, List, Optional, Tuple, Dict, Any

# Must import adapters via the provided thin wrapper. Do NOT import catanatron directly.
from .adapters import (
    Game,
    Player,
    Color,
    ActionType,
    copy_game,
    execute_deterministic,
    execute_spectrum,
    expand_spectrum,
    base_fn,
    list_prunned_actions,
)


class FooPlayer(Player):
    """A player that uses a selective 2-ply adversarial lookahead built on adapters.

    Key features:
    - Phase-aware 1-ply sampling/pruning to keep runtime bounded.
    - Chance-aware expected values using execute_spectrum/expand_spectrum when available.
    - Selective 2-ply adversarial (min over opponent responses) evaluation for top-K 1-ply
      candidates to improve robustness against counters.
    - Deterministic sampling/tie-breaking via seeded RNG.

    All interactions with the engine use only the adapters surface.
    Set self.debug = True on the instance to enable diagnostic printing.
    """

    # Tunable class defaults (STRATEGIZER recommendations)
    MAX_ACTIONS_TO_EVAL: int = 80  # increased from 60
    SAMPLE_PER_ACTION_TYPE: int = 4  # increased from 3
    SPECTRUM_MAX_OUTCOMES: int = 8
    EARLY_TURN_THRESHOLD: int = 30

    # Reintroduce selective 2-ply with conservative parameters
    TOP_K_1PLY: int = 6
    OP_MAX_ACTIONS: int = 10
    OP_SAMPLE_PER_ACTION_TYPE: int = 2

    # Simulation caps and reliability thresholds (updated)
    MAX_SIMULATION_NODES: int = 4000
    MIN_EVAL_SUCCESS_RATE_FOR_2PLY: float = 0.80
    MIN_SPECTRUM_SUCCESS_RATE: float = 0.60
    SCORE_AMBIGUITY_THRESHOLD: float = 0.05

    # Multi-turn expansion planning
    SELF_LOOKAHEAD_DEPTH: int = 3
    SELF_LOOKAHEAD_BUDGET: int = 200

    # reserved/compat
    TOP_K_DEEP: int = 0  # disabled by default
    RNG_SEED: int = 0

    def __init__(self, name: Optional[str] = None):
        # Initialize as BLUE by default (preserve original behavior)
        super().__init__(Color.BLUE, name)
        # Toggle to True to get per-turn diagnostic prints
        self.debug: bool = False
        # Pre-create the value function from adapters.base_fn factory if possible.
        # base_fn returns a callable: (game, color) -> float.
        try:
            self._value_fn = base_fn()
        except Exception:
            # If the factory has a different signature, lazily resolve in evaluation.
            self._value_fn = None

        # Diagnostic counters to help debug evaluation failures and fallbacks
        self._diag = {
            "n_candidates": 0,
            "n_eval_attempts": 0,
            "n_eval_success": 0,
            "n_spectrum_calls": 0,
            "n_spectrum_success": 0,
            "n_det_calls": 0,
            "n_det_success": 0,
            "n_skipped": 0,
            "n_fallbacks_to_first_action": 0,
            "n_2ply_runs": 0,
            "n_2ply_skipped": 0,
            # Additional counters for diagnostics
            "n_road_candidates_included": 0,
            "simulated_nodes_total": 0,
            "n_road_candidates_included": 0,
        }

    # ------------------ Helper methods ------------------
    def _stable_color_hash(self, color: Color) -> int:
        """Stable small hash for a Color used to seed RNG deterministically.

        We keep this deterministic across runs by summing character ordinals of the color's
        string representation. This avoids relying on Python's randomized hash().
        """
        try:
            return sum(ord(c) for c in str(color)) & 0xFFFFFFFF
        except Exception:
            return 0

    def _action_type_key(self, action) -> str:
        """Return a stable grouping key for an action.

        Prefer action.action_type, then other attributes, then class name or string.
        """
        k = getattr(action, "action_type", None)
        if k is not None:
            return str(k)
        for attr in ("type", "name"):
            k = getattr(action, attr, None)
            if k is not None:
                return str(k)
        try:
            return action.__class__.__name__
        except Exception:
            return str(action)

    def _is_build_or_upgrade(self, action) -> bool:
        """Detect actions that build or upgrade (settlement, city, road, upgrade).

        This function is defensive: it checks action_type when available and falls back
        to class name matching so grouping remains robust.
        """
        at = getattr(action, "action_type", None)
        try:
            return at in {
                ActionType.BUILD_SETTLEMENT,
                ActionType.BUILD_CITY,
                ActionType.BUILD_ROAD,
            }
        except Exception:
            name = getattr(action, "name", None) or getattr(action, "type", None) or action.__class__.__name__
            name_str = str(name).lower()
            return any(k in name_str for k in ("build", "settle", "city", "road", "upgrade"))

    def _is_robber_or_chance(self, action) -> bool:
        """Detect robber placement or development-card (chance) actions.

        Uses action_type when available; otherwise checks common name tokens.
        """
        at = getattr(action, "action_type", None)
        try:
            return at in {
                ActionType.PLAY_DEV_CARD,
                ActionType.PLACE_ROBBER,
                ActionType.DRAW_DEV_CARD,
            }
        except Exception:
            name = getattr(action, "name", None) or getattr(action, "type", None) or action.__class__.__name__
            name_str = str(name).lower()
            return any(k in name_str for k in ("robber", "dev", "development", "draw"))

    def _get_visible_vp(self, game: Game, my_color: Color) -> int:
        """Try to extract a visible/observable victory point count for my_color.

        This is intentionally defensive: if no visible metric exists, return 0.
        """
        try:
            vp_map = getattr(game, "visible_vp", None)
            if isinstance(vp_map, dict):
                return int(vp_map.get(my_color, 0))
        except Exception:
            pass
        try:
            vp_map = getattr(game, "visible_victory_points", None)
            if isinstance(vp_map, dict):
                return int(vp_map.get(my_color, 0))
        except Exception:
            pass
        return 0

    def _is_road_action(self, action) -> bool:
        """Detect road-building actions."""
        at = getattr(action, "action_type", None)
        try:
            return at == ActionType.BUILD_ROAD
        except Exception:
            name = getattr(action, "name", None) or getattr(action, "type", None) or action.__class__.__name__
            return "road" in str(name).lower()

    def _sample_actions(self, playable_actions: Iterable, game: Game) -> List:
        """Phase-aware sampling: prioritize builds early, roads mid-game, VP actions late.

        Returns a deterministic, pruned list of candidate actions up to MAX_ACTIONS_TO_EVAL.
        """
        actions = list(playable_actions)
        n = len(actions)
        if n <= self.MAX_ACTIONS_TO_EVAL:
            return actions

        # Determine phase using available heuristics on game. Use tick or current_turn if present.
        current_turn = getattr(game, "current_turn", None)
        if current_turn is None:
            current_turn = getattr(game, "tick", 0)
        early_game = (current_turn <= self.EARLY_TURN_THRESHOLD)
        mid_game = (self.EARLY_TURN_THRESHOLD < current_turn <= 2 * self.EARLY_TURN_THRESHOLD)

        # Group actions by stable key
        groups: Dict[str, List] = {}
        for a in actions:
            key = self._action_type_key(a)
            groups.setdefault(key, []).append(a)

        # Deterministic RNG seeded with a combination of RNG_SEED and player's color
        color_seed = self._stable_color_hash(self.color)
        rng = random.Random(self.RNG_SEED + color_seed)

        sampled: List = []
        # Iterate through groups in a stable order to keep behavior deterministic
        for key in sorted(groups.keys()):
            group = list(groups[key])
            # Determine how many to sample from this group, with phase-aware bias
            sample_count = self.SAMPLE_PER_ACTION_TYPE
            try:
                if early_game and any(self._is_build_or_upgrade(a) for a in group):
                    sample_count += 1
                elif mid_game and any(self._is_road_action(a) for a in group):
                    sample_count += 1
                elif not early_game and any(
                    getattr(a, "action_type", None) in {ActionType.BUILD_CITY, ActionType.BUILD_SETTLEMENT}
                    for a in group
                ):
                    sample_count += 1
            except Exception:
                pass

            # Deterministic shuffle and pick
            rng.shuffle(group)
            take = min(sample_count, len(group))
            sampled.extend(group[:take])
            if len(sampled) >= self.MAX_ACTIONS_TO_EVAL:
                break

        # If under budget, fill deterministically from remaining actions
        if len(sampled) < self.MAX_ACTIONS_TO_EVAL:
            for a in actions:
                if a not in sampled:
                    sampled.append(a)
                    if len(sampled) >= self.MAX_ACTIONS_TO_EVAL:
                        break

        if self.debug:
            phase = "early" if early_game else ("mid" if mid_game else "late")
            print(f"_sample_actions: phase={phase}, pruned {n} -> {len(sampled)} actions (cap={self.MAX_ACTIONS_TO_EVAL})")
        return sampled

    def _sample_opponent_actions(self, playable_actions: Iterable, game: Game, opponent_color: Color) -> List:
        """Opponent-specific sampling that respects OP_SAMPLE_PER_ACTION_TYPE and OP_MAX_ACTIONS.

        Uses a deterministic RNG seeded with opponent color so opponent sampling is reproducible.
        """
        actions = list(playable_actions)
        n = len(actions)
        if n <= self.OP_MAX_ACTIONS:
            return actions

        # Phase detection reused from our own sampling
        current_turn = getattr(game, "current_turn", None)
        if current_turn is None:
            current_turn = getattr(game, "tick", 0)
        early_game = (current_turn <= self.EARLY_TURN_THRESHOLD)

        groups: Dict[str, List] = {}
        for a in actions:
            key = self._action_type_key(a)
            groups.setdefault(key, []).append(a)

        color_seed = self._stable_color_hash(opponent_color)
        rng = random.Random(self.RNG_SEED + color_seed)

        sampled: List = []
        for key in sorted(groups.keys()):
            group = list(groups[key])
            # opponent sampling budget
            sample_count = self.OP_SAMPLE_PER_ACTION_TYPE
            try:
                if early_game and any(self._is_build_or_upgrade(a) for a in group):
                    sample_count += 1
            except Exception:
                pass
            rng.shuffle(group)
            take = min(sample_count, len(group))
            sampled.extend(group[:take])
            if len(sampled) >= self.OP_MAX_ACTIONS:
                break

        if len(sampled) < self.OP_MAX_ACTIONS:
            for a in actions:
                if a not in sampled:
                    sampled.append(a)
                    if len(sampled) >= self.OP_MAX_ACTIONS:
                        break

        if self.debug:
            print(f"_sample_opponent_actions: pruned {n} -> {len(sampled)} actions (cap={self.OP_MAX_ACTIONS})")
        return sampled

    def _normalize_and_cap_spectrum(self, spectrum: Iterable, cap: int) -> List[Tuple[Game, float]]:
        """Normalize spectrum outcomes and cap to `cap` entries.

        Accepts iterables like those returned by execute_spectrum or expand_spectrum entry lists.
        Returns a list of (game, prob) with probabilities summing to 1.
        """
        try:
            lst = list(spectrum)
            if not lst:
                return []
            # Sort by probability descending when possible, then cap
            try:
                sorted_lst = sorted(lst, key=lambda x: float(x[1]) if len(x) > 1 else 0.0, reverse=True)
            except Exception:
                sorted_lst = lst
            capped = sorted_lst[:cap]
            probs = []
            games = []
            for entry in capped:
                try:
                    g, p = entry
                except Exception:
                    # Unexpected shape: skip
                    continue
                games.append(g)
                probs.append(float(p))
            if not games:
                return []
            total = sum(probs)
            if total > 0.0:
                normalized = [(g, p / total) for g, p in zip(games, probs)]
            else:
                n = len(games)
                normalized = [(g, 1.0 / n) for g in games]
            return normalized
        except Exception:
            if self.debug:
                print("_normalize_and_cap_spectrum: failed to normalize spectrum")
                traceback.print_exc()
            return []

    def _determine_opponent_color(self, game: Game, my_color: Color) -> Color:
        """Try to determine the opponent's color from the game state.

        This is defensive: it checks common attributes and falls back to a two-player assumption.
        """
        try:
            cur = getattr(game, "current_player", None)
            if cur is not None:
                # If cur is a Player instance, extract its color attribute when possible
                try:
                    if cur != my_color:
                        return cur
                except Exception:
                    pass
        except Exception:
            pass

        # As a simple fallback, assume a two-player game and pick a different color deterministically
        try:
            colors = [c for c in list(Color)]
            if len(colors) >= 2:
                for c in colors:
                    if c != my_color:
                        return c
        except Exception:
            pass
        # Last resort: return my_color (harmless, though less correct)
        return my_color

    def _derive_opponent_actions(self, game: Game, opponent_color: Color) -> List:
        """Obtain a list of opponent actions with several fallbacks.

        Order:
        1) adapters.list_prunned_actions(game)
        2) game.playable_actions() if present
        3) empty list (conservative)
        """
        try:
            # Preferred: adapters-provided pruned action list (designed for search)
            pruned = list_prunned_actions(game)
            if pruned:
                return pruned
        except Exception:
            if self.debug:
                print("_derive_opponent_actions: list_prunned_actions failed")
                traceback.print_exc()

        try:
            pa = getattr(game, "playable_actions", None)
            if callable(pa):
                res = pa()
                if res:
                    return list(res)
        except Exception:
            if self.debug:
                print("_derive_opponent_actions: game.playable_actions() failed")
                traceback.print_exc()

        # As a conservative fallback, return empty list so we evaluate the post-action state directly
        return []

    def _safe_eval_base_fn(self, g: Game, color: Color) -> Optional[float]:
        """Safely call the adapters' base value function in its possible forms.

        Tries self._value_fn(g,color) if available; otherwise attempts base_fn()(g,color) and
        finally base_fn(g,color). Returns None on failure and logs when debug=True.
        """
        try:
            if self._value_fn is not None:
                return float(self._value_fn(g, color))
        except Exception:
            if self.debug:
                print("_safe_eval_base_fn: self._value_fn failed")
                traceback.print_exc()
        # Try factory form
        try:
            vf = base_fn()
            try:
                return float(vf(g, color))
            except Exception:
                if self.debug:
                    print("_safe_eval_base_fn: vf(g,color) failed")
                    traceback.print_exc()
        except Exception:
            # Maybe base_fn itself accepts (g,color)
            pass
        try:
            return float(base_fn(g, color))
        except Exception:
            if self.debug:
                print("_safe_eval_base_fn: all attempts to call base_fn failed")
                traceback.print_exc()
            return None

    def _simulate_and_evaluate(self, game: Game, action, my_color: Color) -> Optional[float]:
        """Simulate `action` from `game` and return a numeric expected score for my_color.

        If action is None, simply evaluate the provided game state.
        This function handles spectrum (chance) outcomes when available and falls back to
        deterministic execution. Returns None on failure for the given simulation.
        """
        # Copy the game to avoid mutating caller's state
        try:
            game_copy = copy_game(game)
        except Exception as e:
            if self.debug:
                print("_simulate_and_evaluate: copy_game failed:", e)
                traceback.print_exc()
            return None

        # If action is None, just evaluate the provided state
        if action is None:
            return self._safe_eval_base_fn(game_copy, my_color)

        # Chance-aware path
        if self._is_robber_or_chance(action):
            try:
                spec = None
                try:
                    spec = execute_spectrum(game_copy, action)
                except Exception:
                    # Try expand_spectrum single-action expansion
                    try:
                        spec_map = expand_spectrum(game_copy, [action])
                        if isinstance(spec_map, dict):
                            spec = spec_map.get(action, None)
                    except Exception:
                        spec = None

                if spec:
                    outcomes = self._normalize_and_cap_spectrum(spec, self.SPECTRUM_MAX_OUTCOMES)
                    if not outcomes:
                        # Fall through to deterministic
                        pass
                    else:
                        total_score = 0.0
                        for og, prob in outcomes:
                            sc = self._safe_eval_base_fn(og, my_color)
                            if sc is None:
                                # If any outcome can't be evaluated reliably, abort spectrum path
                                total_score = None
                                break
                            total_score += prob * sc
                        if total_score is None:
                            if self.debug:
                                print("_simulate_and_evaluate: spectrum had unscorable outcomes; falling back")
                        else:
                            return float(total_score)
            except Exception as e:
                if self.debug:
                    print("_simulate_and_evaluate: execute_spectrum/expand_spectrum failed:", e)
                    traceback.print_exc()
                # fall through to deterministic

        # Deterministic fallback
        try:
            outcomes = execute_deterministic(game_copy, action)
        except Exception as e:
            if self.debug:
                print("_simulate_and_evaluate: execute_deterministic failed:", e)
                traceback.print_exc()
            return None

        try:
            if not outcomes:
                if self.debug:
                    print("_simulate_and_evaluate: execute_deterministic returned no outcomes")
                return None
            first = outcomes[0]
            if isinstance(first, (list, tuple)) and len(first) >= 1:
                resultant_game = first[0]
            else:
                resultant_game = first
        except Exception:
            resultant_game = game_copy

        return self._safe_eval_base_fn(resultant_game, my_color)

    # ------------------ Expansion potential computation ------------------
    def _compute_expansion_potential(self, game: Game, action) -> float:
        """Compute the expansion potential of an action.

        Expansion potential is the average number of playable actions available to us
        in the resulting game state(s) after executing `action`.
        Returns -inf on failure to simulate/evaluate so unreliable candidates are deprioritized.
        """
        try:
            game_copy = copy_game(game)
        except Exception:
            if self.debug:
                print("_compute_expansion_potential: copy_game failed")
                traceback.print_exc()
            return -float("inf")

        # Simulate the action to get outcome branches
        outcomes = []
        try:
            if self._is_robber_or_chance(action):
                spec = None
                try:
                    spec = execute_spectrum(game_copy, action)
                except Exception:
                    try:
                        spec_map = expand_spectrum(game_copy, [action])
                        if isinstance(spec_map, dict):
                            spec = spec_map.get(action, None)
                    except Exception:
                        spec = None
                if spec:
                    outcomes = self._normalize_and_cap_spectrum(spec, self.SPECTRUM_MAX_OUTCOMES)
            else:
                det_res = execute_deterministic(game_copy, action)
                if det_res:
                    # det_res often is list of (game, prob) or similar
                    # Normalize into (game, prob) entries
                    normalized = []
                    for entry in det_res[: self.SPECTRUM_MAX_OUTCOMES]:
                        try:
                            g, p = entry
                        except Exception:
                            g = entry
                            p = 1.0
                        normalized.append((g, float(p)))
                    total_p = sum(p for _, p in normalized)
                    if total_p > 0:
                        outcomes = [(g, p / total_p) for (g, p) in normalized]
                    else:
                        n = len(normalized)
                        if n > 0:
                            outcomes = [(g, 1.0 / n) for (g, _) in normalized]

        except Exception:
            if self.debug:
                print("_compute_expansion_potential: failed to simulate action")
                traceback.print_exc()
            return -float("inf")

        if not outcomes:
            return -float("inf")

        total_expansion = 0.0
        for outcome_game, prob in outcomes:
            try:
                # Use our opponent-action derivation to count playable actions for our color
                playable = self._derive_opponent_actions(outcome_game, self.color)
                expansion = len(playable) if playable else 0
                total_expansion += prob * expansion
            except Exception:
                if self.debug:
                    print("_compute_expansion_potential: failed to derive playable actions")
                    traceback.print_exc()
                return -float("inf")

        return total_expansion

    # ------------------ NEW: compute opponent impact ------------------
    def _compute_opponent_impact(self, game: Game, action) -> float:
        """Compute the impact of a robber action on opponent expansion potential.

        Returns the difference between opponent's expansion potential before and after the action.
        On failure returns -inf so unreliable candidates are deprioritized.
        """
        try:
            game_copy = copy_game(game)
        except Exception:
            if self.debug:
                print("_compute_opponent_impact: copy_game failed")
                traceback.print_exc()
            return -float("inf")

        opp_color = self._determine_opponent_color(game, self.color)

        # Compute opponent expansion potential before action
        try:
            opp_expansion_before = self._compute_expansion_potential(game_copy, None)
        except Exception:
            if self.debug:
                print("_compute_opponent_impact: failed to compute opp_expansion_before")
                traceback.print_exc()
            return -float("inf")

        # Simulate the action to get outcome branches
        outcomes = []
        try:
            if self._is_robber_or_chance(action):
                spec = None
                try:
                    spec = execute_spectrum(game_copy, action)
                except Exception:
                    try:
                        spec_map = expand_spectrum(game_copy, [action])
                        if isinstance(spec_map, dict):
                            spec = spec_map.get(action, None)
                    except Exception:
                        spec = None
                if spec:
                    outcomes = self._normalize_and_cap_spectrum(spec, self.SPECTRUM_MAX_OUTCOMES)
            else:
                det_res = execute_deterministic(game_copy, action)
                if det_res:
                    normalized = []
                    for entry in det_res[: self.SPECTRUM_MAX_OUTCOMES]:
                        try:
                            g, p = entry
                        except Exception:
                            g = entry
                            p = 1.0
                        normalized.append((g, float(p)))
                    total_p = sum(p for _, p in normalized)
                    if total_p > 0:
                        outcomes = [(g, p / total_p) for (g, p) in normalized]
                    else:
                        n = len(normalized)
                        if n > 0:
                            outcomes = [(g, 1.0 / n) for (g, _) in normalized]
        except Exception:
            if self.debug:
                print("_compute_opponent_impact: failed to simulate action")
                traceback.print_exc()
            return -float("inf")

        if not outcomes:
            return -float("inf")

        expected_opp_expansion_after = 0.0
        for outcome_game, prob in outcomes:
            try:
                opp_expansion_after = self._compute_expansion_potential(outcome_game, None)
                expected_opp_expansion_after += prob * opp_expansion_after
            except Exception:
                if self.debug:
                    print("_compute_opponent_impact: failed to compute expansion for outcome")
                    traceback.print_exc()
                return -float("inf")

        return opp_expansion_before - expected_opp_expansion_after

    def _count_build_actions(self, game: Game, color: Color) -> int:
        """Count the number of build-type actions available to `color` in `game`."""
        try:
            playable = self._derive_opponent_actions(game, color)
            if not playable:
                return 0
            return sum(
                1 for a in playable
                if self._is_build_or_upgrade(a) or self._is_road_action(a)
            )
        except Exception:
            if self.debug:
                print("_count_build_actions: failed to derive playable actions")
                traceback.print_exc()
            return 0

    # ------------------ NEW: simulate_action_branches ------------------
    def _simulate_action_branches(self, game: Game, action) -> List[Tuple[Game, float]]:
        """Simulate action and return outcome branches (spectrum or deterministic)."""
        try:
            game_copy = copy_game(game)
        except Exception:
            if self.debug:
                print("_simulate_action_branches: copy_game failed")
                traceback.print_exc()
            return []

        outcomes: List[Tuple[Game, float]] = []
        try:
            if self._is_robber_or_chance(action):
                spec = None
                try:
                    spec = execute_spectrum(game_copy, action)
                except Exception:
                    try:
                        spec_map = expand_spectrum(game_copy, [action])
                        if isinstance(spec_map, dict):
                            spec = spec_map.get(action, None)
                    except Exception:
                        spec = None
                if spec:
                    outcomes = self._normalize_and_cap_spectrum(spec, self.SPECTRUM_MAX_OUTCOMES)
            else:
                det_res = execute_deterministic(game_copy, action)
                if det_res:
                    # normalize det_res into list of (game, prob)
                    normalized: List[Tuple[Game, float]] = []
                    for entry in det_res[: self.SPECTRUM_MAX_OUTCOMES]:
                        try:
                            g, p = entry
                        except Exception:
                            g = entry
                            p = 1.0
                        normalized.append((g, float(p)))
                    total_p = sum(p for _, p in normalized)
                    if total_p > 0:
                        outcomes = [(g, p / total_p) for (g, p) in normalized]
                    else:
                        n = len(normalized)
                        if n > 0:
                            outcomes = [(g, 1.0 / n) for (g, _) in normalized]
        except Exception:
            if self.debug:
                print("_simulate_action_branches: failed to simulate action")
                traceback.print_exc()
            return []

        return outcomes

    def _is_settlement_build(self, action) -> bool:
        """Detect settlement-building actions."""
        at = getattr(action, "action_type", None)
        try:
            return at == ActionType.BUILD_SETTLEMENT
        except Exception:
            name = getattr(action, "name", None) or getattr(action, "type", None) or action.__class__.__name__
            return "settle" in str(name).lower()

    # ------------------ NEW: expected settlement gain ------------------
    def _compute_expected_settlement_gain(self, game: Game, action) -> float:
        """Compute expected number of additional settlements reachable within SELF_LOOKAHEAD_DEPTH turns."""
        try:
            game_copy = copy_game(game)
        except Exception:
            if self.debug:
                print("_compute_expected_settlement_gain: copy_game failed")
                traceback.print_exc()
            return -float("inf")

        outcomes = self._simulate_action_branches(game_copy, action)
        if not outcomes:
            return -float("inf")

        total_gain = 0.0
        sim_nodes_used = 0

        for outcome_game, prob in outcomes:
            if sim_nodes_used >= self.SELF_LOOKAHEAD_BUDGET:
                break

            # Run bounded self-only DFS to estimate settlement gain
            stack = [(outcome_game, 0, 0)]  # (game_state, depth, settlements_gained)
            best_gain_for_branch = 0

            while stack and sim_nodes_used < self.SELF_LOOKAHEAD_BUDGET:
                state, depth, gained = stack.pop()
                sim_nodes_used += 1

                # Derive playable actions for our color
                try:
                    playable = self._derive_opponent_actions(state, self.color) or []
                except Exception:
                    if self.debug:
                        print("_compute_expected_settlement_gain: failed to derive playable actions")
                        traceback.print_exc()
                    continue

                # Filter to build/road/upgrade actions
                build_candidates = [
                    act for act in playable
                    if self._is_build_or_upgrade(act) or self._is_road_action(act)
                ]

                # Simulate each build candidate deterministically
                for act in self._sample_actions(build_candidates, state)[:5]:  # Limit to top 5 candidates
                    try:
                        det = execute_deterministic(copy_game(state), act)
                        if not det:
                            continue
                        first = det[0]
                        if isinstance(first, (list, tuple)) and len(first) >= 1:
                            next_state = first[0]
                        else:
                            next_state = first
                    except Exception:
                        continue

                    new_gained = gained + (1 if self._is_settlement_build(act) else 0)
                    if depth + 1 < self.SELF_LOOKAHEAD_DEPTH:
                        stack.append((next_state, depth + 1, new_gained))
                    else:
                        if new_gained > best_gain_for_branch:
                            best_gain_for_branch = new_gained

                # also update best_gain_for_branch with current gained
                if gained > best_gain_for_branch:
                    best_gain_for_branch = gained

            total_gain += prob * best_gain_for_branch

        return float(total_gain)

    # ------------------ NEW missing method: _evaluate_action ------------------
    def _evaluate_action(self, game: Game, action, my_color: Color) -> Optional[Tuple[float, float]]:
        """Evaluate a candidate action and return (score, vp_delta) or None on failure.

        This method unifies spectrum-based chance evaluation and deterministic execution
        and returns both the numeric score (from base_fn) and the visible VP delta (after - before).
        It is defensive to adapter signature differences and logs traces when self.debug is True.
        """
        # Diagnostic: attempt counter
        self._diag["n_eval_attempts"] = self._diag.get("n_eval_attempts", 0) + 1

        # Helper: safe eval using existing wrapper
        def safe_eval(g: Game) -> Optional[float]:
            return self._safe_eval_base_fn(g, my_color)

        # Helper: visible vp extraction (use existing helper)
        def get_vp(g: Game) -> float:
            try:
                return float(self._get_visible_vp(g, my_color))
            except Exception:
                if self.debug:
                    print("_evaluate_action: _get_visible_vp failed")
                    traceback.print_exc()
                return 0.0

        # Step A: copy game
        try:
            game_copy = copy_game(game)
        except Exception:
            if self.debug:
                print("_evaluate_action: copy_game failed:")
                traceback.print_exc()
            self._diag["n_skipped"] = self._diag.get("n_skipped", 0) + 1
            return None

        # original visible vp
        try:
            vp_orig = get_vp(game)
        except Exception:
            vp_orig = 0.0

        # Step B: if chance-like, try spectrum expansion
        if self._is_robber_or_chance(action):
            try:
                self._diag["n_spectrum_calls"] = self._diag.get("n_spectrum_calls", 0) + 1
                spec = None
                try:
                    spec = execute_spectrum(game_copy, action)
                except Exception:
                    try:
                        spec_map = expand_spectrum(game_copy, [action])
                        if isinstance(spec_map, dict):
                            spec = spec_map.get(action, None)
                    except Exception:
                        spec = None

                if spec:
                    outcomes = self._normalize_and_cap_spectrum(spec, self.SPECTRUM_MAX_OUTCOMES)
                    if outcomes:
                        weighted_score = 0.0
                        weighted_vp_delta = 0.0
                        any_scored = False
                        for og, prob in outcomes:
                            sc = safe_eval(og)
                            if sc is None:
                                # skip unscorable outcomes
                                continue
                            any_scored = True
                            vp_out = get_vp(og)
                            weighted_score += prob * sc
                            weighted_vp_delta += prob * (vp_out - vp_orig)
                        if any_scored:
                            self._diag["n_spectrum_success"] = self._diag.get("n_spectrum_success", 0) + 1
                            self._diag["n_eval_success"] = self._diag.get("n_eval_success", 0) + 1
                            return (float(weighted_score), float(weighted_vp_delta))
                        # else fall through to deterministic
            except Exception:
                if self.debug:
                    print("_evaluate_action: spectrum evaluation failed:")
                    traceback.print_exc()
                # fall through

        # Step C: deterministic execution fallback
        try:
            self._diag["n_det_calls"] = self._diag.get("n_det_calls", 0) + 1
            res = execute_deterministic(game_copy, action)
        except Exception:
            if self.debug:
                print("_evaluate_action: execute_deterministic failed:")
                traceback.print_exc()
            self._diag["n_skipped"] = self._diag.get("n_skipped", 0) + 1
            return None

        try:
            # normalize to a single resultant game
            resultant_game = None
            if res is None:
                resultant_game = game_copy
            elif isinstance(res, (list, tuple)):
                first = res[0]
                if isinstance(first, tuple) and len(first) >= 1:
                    resultant_game = first[0]
                else:
                    resultant_game = first
            else:
                # could be a single game object
                resultant_game = res if hasattr(res, "state") or hasattr(res, "current_player") else game_copy

            score = safe_eval(resultant_game)
            if score is None:
                self._diag["n_skipped"] = self._diag.get("n_skipped", 0) + 1
                return None
            vp_after = get_vp(resultant_game)
            vp_delta = float(vp_after - vp_orig)
            # success counters
            self._diag["n_eval_success"] = self._diag.get("n_eval_success", 0) + 1
            self._diag["n_det_success"] = self._diag.get("n_det_success", 0) + 1
            return (float(score), float(vp_delta))
        except Exception:
            if self.debug:
                print("_evaluate_action: normalize/eval failed:")
                traceback.print_exc()
            self._diag["n_skipped"] = self._diag.get("n_skipped", 0) + 1
            return None

    # ------------------ Decision method (public) ------------------
    def decide(self, game: Game, playable_actions: Iterable):
        """Choose an action using selective 2-ply adversarial lookahead.

        Flow:
        1) Run phase-aware 1-ply sampling and evaluation across candidates.
        2) Keep top TOP_K_1PLY candidates by 1-ply score and deepen each with opponent modeling.
        3) For each candidate, compute expected adversarial value = E_outcomes[min_opponent_response(score)].
        4) Pick candidate maximizing (expected_value, 1-ply vp_delta, repr action tie-break).

        All adapter calls are protected with try/except. On catastrophic failure we fall back to
        returning the best 1-ply candidate or the first playable action as a last resort.
        """
        actions = list(playable_actions)

        if not actions:
            if self.debug:
                print("decide: no playable_actions provided")
            return None

        if len(actions) == 1:
            if self.debug:
                print("decide: single playable action, returning it")
            return actions[0]

        # reset diagnostics for this decision
        self._diag = {k: 0 for k in self._diag}

        # Stage 1: 1-ply evaluation
        candidates = self._sample_actions(actions, game)
        self._diag["n_candidates"] = len(candidates)
        if self.debug:
            print(f"decide: sampled {len(candidates)} candidates from {len(actions)} actions")

        one_ply_results: List[Tuple[Any, float, float]] = []  # (action, score, vp_delta)

        # Resolve evaluator function robustly to avoid AttributeError
        eval_fn = getattr(self, "_evaluate_action", None) or getattr(self, "_simulate_and_evaluate", None)
        if eval_fn is None:
            if self.debug:
                print("decide: no evaluator method found; falling back to first action")
            self._diag["n_fallbacks_to_first_action"] = self._diag.get("n_fallbacks_to_first_action", 0) + 1
            return actions[0]

        for idx, a in enumerate(candidates, start=1):
            try:
                res = eval_fn(game, a, self.color)
            except Exception:
                if self.debug:
                    print("decide: evaluator raised exception for action", repr(a))
                    traceback.print_exc()
                res = None

            if self.debug:
                print(f"1-ply [{idx}/{len(candidates)}]: {repr(a)} -> {res}")

            if res is None:
                # count skipped attempts
                self._diag["n_skipped"] = self._diag.get("n_skipped", 0) + 1
                continue
            sc, vpd = res
            one_ply_results.append((a, float(sc), float(vpd)))

        if not one_ply_results:
            # Nothing evaluated successfully; fallback deterministically
            if self.debug:
                print("decide: no 1-ply evaluations succeeded; falling back to first playable action")
            self._diag["n_fallbacks_to_first_action"] = self._diag.get("n_fallbacks_to_first_action", 0) + 1
            return actions[0]

        # Stage 2: reliability checks before re-enabling 2-ply (adaptive activation)
        eval_success_rate = self._diag.get("n_eval_success", 0) / max(1, self._diag.get("n_eval_attempts", 0))
        spectrum_success_rate = (
            self._diag.get("n_spectrum_success", 0) / max(1, self._diag.get("n_spectrum_calls", 0))
            if self._diag.get("n_spectrum_calls", 0) > 0
            else 1.0
        )

        # Evaluate score gap between top two 1-ply results
        one_ply_results.sort(key=lambda t: t[1], reverse=True)
        if len(one_ply_results) > 1:
            score_gap = one_ply_results[0][1] - one_ply_results[1][1]
        else:
            score_gap = float("inf")

        # Check for road/robber candidates with high potential
        candidates_list = [t[0] for t in one_ply_results]
        road_candidates = [a for a in candidates_list if self._is_road_action(a)]
        robber_candidates = [a for a in candidates_list if self._is_robber_or_chance(a)]
        has_high_potential_road = any(
            self._compute_expansion_potential(game, a) >= 0 for a in road_candidates
        )
        has_high_potential_robber = any(
            self._compute_opponent_impact(game, a) >= 0 for a in robber_candidates
        )

        allow_2ply = (
            (eval_success_rate >= self.MIN_EVAL_SUCCESS_RATE_FOR_2PLY and spectrum_success_rate >= self.MIN_SPECTRUM_SUCCESS_RATE)
            or (score_gap < self.SCORE_AMBIGUITY_THRESHOLD)
            or has_high_potential_road
            or has_high_potential_robber
        )

        if self.debug:
            print(
                f"decide: eval_success_rate={eval_success_rate:.2f}, "
                f"spectrum_success_rate={spectrum_success_rate:.2f}, "
                f"score_gap={score_gap:.3f}, "
                f"has_high_potential_road={has_high_potential_road}, "
                f"has_high_potential_robber={has_high_potential_robber}, "
                f"allow_2ply={allow_2ply}"
            )

        if not allow_2ply:
            self._diag["n_2ply_skipped"] = self._diag.get("n_2ply_skipped", 0) + 1
            if self.debug:
                print("decide: skipping 2-ply due to low reliability or lack of high-potential candidates")
            # Fall back to best 1-ply action
            best_action_1ply = None
            best_score = -float("inf")
            best_vp = -float("inf")
            best_repr = None
            for (a, s, v) in one_ply_results:
                tie_repr = repr(a)
                is_better = False
                if best_action_1ply is None:
                    is_better = True
                elif s > best_score:
                    is_better = True
                elif s == best_score:
                    if v > best_vp:
                        is_better = True
                    elif v == best_vp and (best_repr is None or tie_repr < best_repr):
                        is_better = True
                if is_better:
                    best_action_1ply = a
                    best_score = s
                    best_vp = v
                    best_repr = tie_repr

            if best_action_1ply is not None:
                if self.debug:
                    print("decide: chosen action (1-ply fallback):", repr(best_action_1ply), "score:", best_score, "vp_delta:", best_vp)
                    print("Diagnostics:", self._diag)
                return best_action_1ply
            else:
                if self.debug:
                    print("decide: no choice after fallbacks; returning first playable action")
                    self._diag["n_fallbacks_to_first_action"] = self._diag.get("n_fallbacks_to_first_action", 0) + 1
                return actions[0]

        # Stage 3: Build candidate pool with expansion potential, settlement gain, and road guarantee
        one_ply_results.sort(key=lambda t: (t[1], t[2]), reverse=True)
        top_by_1ply = [t[0] for t in one_ply_results[:3]]  # Always include top 3 by 1-ply score
        remaining_candidates = [t[0] for t in one_ply_results[3:]]

        # Compute expansion potential and settlement gain for remaining candidates
        expansion_scores: Dict[Any, float] = {}
        settlement_gain_scores: Dict[Any, float] = {}
        for a in remaining_candidates:
            exp_potential = self._compute_expansion_potential(game, a)
            if exp_potential != -float("inf"):
                expansion_scores[a] = exp_potential

            settlement_gain = self._compute_expected_settlement_gain(game, a)
            if settlement_gain != -float("inf"):
                settlement_gain_scores[a] = settlement_gain

        # Sort remaining candidates by settlement gain, then expansion potential
        sorted_remaining = sorted(
            settlement_gain_scores.items(),
            key=lambda x: (x[1], expansion_scores.get(x[0], -float("inf"))),
            reverse=True,
        )
        additional_candidates = [a for a, _ in sorted_remaining[: max(0, self.TOP_K_1PLY - len(top_by_1ply))]]
        candidate_pool = top_by_1ply + additional_candidates

        # Guarantee inclusion of at least one high-potential road candidate
        road_candidates = [a for a in remaining_candidates if self._is_road_action(a)]
        road_scores = {a: self._compute_expected_settlement_gain(game, a) for a in road_candidates}
        best_road = None
        if road_scores:
            best_road = max(road_scores.items(), key=lambda x: x[1])[0]
            if best_road not in candidate_pool:
                candidate_pool.append(best_road)
                self._diag["n_road_candidates_included"] = self._diag.get("n_road_candidates_included", 0) + 1
                if self.debug:
                    print(f"decide: added guaranteed road candidate {repr(best_road)} with settlement_gain={road_scores[best_road]}")

        if self.debug:
            print("Candidate pool:")
            for a in candidate_pool:
                exp_potential = expansion_scores.get(a, "N/A")
                settlement_gain = settlement_gain_scores.get(a, "N/A")
                is_road = self._is_road_action(a)
                is_robber = self._is_robber_or_chance(a)
                print(
                    f"  {repr(a)} (is_road={is_road}, is_robber={is_robber}, "
                    f"expansion_potential={exp_potential}, settlement_gain={settlement_gain})"
                )

        # Stage 4: 2-ply adversarial evaluation (conservative)
        best_action = None
        best_value = -float("inf")
        best_expansion = -float("inf")
        best_vp_delta = -float("inf")
        best_repr = None
        sim_count = 0

        # Use class cap for simulated nodes
        SIMULATION_HARD_LIMIT = self.MAX_SIMULATION_NODES

        # Track how many candidates succeeded in deep simulation
        deep_successful_candidates = 0

        try:
            for a in candidate_pool:
                if sim_count >= SIMULATION_HARD_LIMIT:
                    if self.debug:
                        print("decide: reached simulation hard limit; stopping deepening")
                    break

                # Simulate our action a to produce outcome branches
                try:
                    game_copy = copy_game(game)
                except Exception as e:
                    if self.debug:
                        print("decide: copy_game failed for candidate", repr(a), e)
                        traceback.print_exc()
                    continue

                # Obtain outcome branches: prefer spectrum for chance actions
                outcomes: List[Tuple[Game, float]] = []
                try:
                    if self._is_robber_or_chance(a):
                        spec = None
                        try:
                            spec = execute_spectrum(game_copy, a)
                        except Exception:
                            try:
                                spec_map = expand_spectrum(game_copy, [a])
                                if isinstance(spec_map, dict):
                                    spec = spec_map.get(a, None)
                            except Exception:
                                spec = None

                        if spec:
                            outcomes = self._normalize_and_cap_spectrum(spec, self.SPECTRUM_MAX_OUTCOMES)
                    # Fallback to deterministic
                    if not outcomes:
                        det = execute_deterministic(game_copy, a)
                        if not det:
                            if self.debug:
                                print("decide: execute_deterministic returned empty for", repr(a))
                            continue
                        # det is list of (game, prob) often; take as provided
                        # normalize shape defensively
                        normalized = []
                        for entry in det[: self.SPECTRUM_MAX_OUTCOMES]:
                            try:
                                g, p = entry
                            except Exception:
                                g = entry
                                p = 1.0
                            normalized.append((g, float(p)))
                        # If probabilities not summing to 1, normalize
                        total_p = sum(p for _, p in normalized)
                        if total_p <= 0:
                            # assign uniform
                            n = len(normalized)
                            outcomes = [(g, 1.0 / n) for (g, _) in normalized]
                        else:
                            outcomes = [(g, p / total_p) for (g, p) in normalized]

                except Exception as e:
                    if self.debug:
                        print("decide: failed to obtain outcomes for candidate", repr(a), "error:", e)
                        traceback.print_exc()
                    continue

                # Cap outcomes just in case
                if len(outcomes) > self.SPECTRUM_MAX_OUTCOMES:
                    outcomes = outcomes[: self.SPECTRUM_MAX_OUTCOMES]

                if self.debug:
                    print(f"Candidate {repr(a)} produced {len(outcomes)} outcome(s) to evaluate")

                expected_value_a = 0.0
                expansion_potential_a = 0.0
                # find 1-ply vp delta for tie-break usage
                one_ply_vp_delta = next((v for (act, s, v) in one_ply_results if act == a), 0.0)

                # Compute robber impact if applicable
                robber_impact_a = -float("inf")
                if self._is_robber_or_chance(a):
                    try:
                        robber_impact_a = self._compute_opponent_impact(game, a)
                    except Exception:
                        if self.debug:
                            print("decide: failed to compute robber impact for", repr(a))
                            traceback.print_exc()
                        robber_impact_a = -float("inf")

                # For each outcome, model opponent adversarial response
                outcome_failures = 0
                for og, p_i in outcomes:
                    if sim_count >= SIMULATION_HARD_LIMIT:
                        break
                    # Compute expansion potential for this outcome
                    try:
                        playable = self._derive_opponent_actions(og, self.color)
                        expansion = len(playable) if playable else 0
                        expansion_potential_a += p_i * expansion
                    except Exception:
                        if self.debug:
                            print("decide: failed to compute expansion potential for outcome")
                            traceback.print_exc()
                        expansion_potential_a += p_i * -float("inf")

                    # Determine opponent color
                    opp_color = self._determine_opponent_color(og, self.color)
                    # Get opponent actions with robust fallbacks
                    try:
                        opp_actions = self._derive_opponent_actions(og, opp_color)
                    except Exception:
                        opp_actions = []

                    if not opp_actions:
                        val_i = self._simulate_and_evaluate(og, None, self.color)
                        if val_i is None:
                            outcome_failures += 1
                            continue
                        expected_value_a += p_i * val_i
                        sim_count += 1
                        continue

                    # Prune opponent actions deterministically and cap
                    opp_sampled = self._sample_opponent_actions(opp_actions, og, opp_color)[: self.OP_MAX_ACTIONS]

                    if self.debug:
                        print(f"  outcome p={p_i:.3f}: opp_actions={len(opp_actions)} -> sampled={len(opp_sampled)}")

                    # Adversarial opponent: they choose the action minimizing our final score
                    min_score_after_opp = float("inf")
                    opp_successes = 0
                    for b in opp_sampled:
                        if sim_count >= SIMULATION_HARD_LIMIT:
                            break
                        val_after_b = self._simulate_and_evaluate(og, b, self.color)
                        sim_count += 1
                        if val_after_b is None:
                            continue
                        opp_successes += 1
                        if val_after_b < min_score_after_opp:
                            min_score_after_opp = val_after_b

                    if opp_successes == 0:
                        # If no opponent simulation succeeded, evaluate the post-my-action state
                        tmp = self._simulate_and_evaluate(og, None, self.color)
                        if tmp is None:
                            outcome_failures += 1
                            continue
                        min_score_after_opp = tmp

                    expected_value_a += p_i * min_score_after_opp

                # If too many outcomes failed for this candidate, skip candidate
                if outcome_failures >= max(1, len(outcomes) // 2):
                    if self.debug:
                        print(f"decide: skipping candidate {repr(a)} due to many outcome failures ({outcome_failures})")
                    continue

                deep_successful_candidates += 1

                # Compare candidate using tie-breaks including expansion potential
                if self.debug:
                    print(
                        f"Candidate {repr(a)}: expected_value={expected_value_a}, "
                        f"expansion_potential={expansion_potential_a}, "
                        f"1-ply vp_delta={one_ply_vp_delta}"
                    )

                is_better = False
                if best_action is None:
                    is_better = True
                elif expected_value_a > best_value:
                    is_better = True
                elif expected_value_a == best_value:
                    if expansion_potential_a > best_expansion:
                        is_better = True
                    elif expansion_potential_a == best_expansion:
                        if one_ply_vp_delta > best_vp_delta:
                            is_better = True
                        elif one_ply_vp_delta == best_vp_delta:
                            tie_repr = repr(a)
                            if best_repr is None or tie_repr < best_repr:
                                is_better = True

                if is_better:
                    best_action = a
                    best_value = expected_value_a
                    best_expansion = expansion_potential_a
                    best_vp_delta = one_ply_vp_delta
                    best_repr = repr(a)

                # End loop over candidate_pool
            # End try
        except Exception:
            if self.debug:
                print("decide: unexpected error during 2-ply deepening")
                traceback.print_exc()
            # Fall back to 1-ply selection below

        # Record whether we ran 2-ply for diagnostics
        if deep_successful_candidates > 0:
            self._diag["n_2ply_runs"] = self._diag.get("n_2ply_runs", 0) + 1
        else:
            self._diag["n_2ply_skipped"] = self._diag.get("n_2ply_skipped", 0) + 1

        # Update simulated nodes total for diagnostics
        self._diag["simulated_nodes_total"] = self._diag.get("simulated_nodes_total", 0) + sim_count

        # If 2-ply produced a valid selection, return it
        if best_action is not None:
            if self.debug:
                print("decide: selected (2-ply) action:", repr(best_action), "value:", best_value)
                print("Diagnostics:", self._diag)
            return best_action

        # Otherwise, fall back to best 1-ply action using existing tie-break rules
        if self.debug:
            print("decide: falling back to best 1-ply action")
        best_action_1ply = None
        best_score = -float("inf")
        best_vp = -float("inf")
        best_repr = None
        for (a, s, v) in one_ply_results:
            tie_repr = repr(a)
            is_better = False
            if best_action_1ply is None:
                is_better = True
            elif s > best_score:
                is_better = True
            elif s == best_score:
                if v > best_vp:
                    is_better = True
                elif v == best_vp and (best_repr is None or tie_repr < best_repr):
                    is_better = True
            if is_better:
                best_action_1ply = a
                best_score = s
                best_vp = v
                best_repr = tie_repr

        if best_action_1ply is not None:
            if self.debug:
                print("decide: chosen action (1-ply fallback):", repr(best_action_1ply), "score:", best_score, "vp_delta:", best_vp)
                print("Diagnostics:", self._diag)
            return best_action_1ply

        # Last resort: return first playable action
        if self.debug:
            print("decide: no choice after fallbacks; returning first playable action")
            self._diag["n_fallbacks_to_first_action"] = self._diag.get("n_fallbacks_to_first_action", 0) + 1
            print("Diagnostics:", self._diag)
        return actions[0]
