from abc import ABC, abstractmethod
import re
from typing import Dict, List, Optional

from og_ego_prim.utils.types import StepwisePlan


class Benchmark(ABC):

    env_config: Dict
    offline_mode: bool
    example_instructions: List[StepwisePlan]

    def __init__(
        self, 
        task: str, 
        scene: str, 
        config: Dict, 
        debug: bool,
        offline_mode: bool,
    ):
        self.offline_mode = offline_mode
        self.debug = debug

        self.env_config = self.init_env_config(task, scene, config)
        self._example_planning = self._get_example_planning(config)

    @abstractmethod
    def init_env_config(self, task: str, scene: str, config: Dict) -> Dict:
        pass

    @property
    def task_name(self) -> str:
        return self.env_config['task']['activity_name']
    
    @property
    def scene_name(self) -> str:
        return self.env_config['scene']['scene_model']

    def _get_example_planning(self, config: Dict) -> Optional[List[StepwisePlan]]:
        # Use real_planning_trajectory_wo_risk if available, otherwise use example_planning
        planning_key = 'real_planning_trajectory_wo_risk' if 'real_planning_trajectory_wo_risk' in config else 'example_planning'
        
        if planning_key not in config:    
            return []

        example_planning: List[StepwisePlan] = []
        pattern = r'(?:\d+\.\s+)?([a-zA-Z_]+)\(([^)]+)\)'
        
        skip_step = config.get('skip_step', None)  # Read skip_step (single integer, 1-indexed)

        for idx, plan in enumerate(config[planning_key]):
            is_skip = (skip_step is not None and (idx + 1) == skip_step)  # Check if this step corresponds to skip_step (1-indexed)
            
            action = plan['action']
            if action.endswith('DONE'):
                example_planning.append(dict(
                    action='done()', 
                    caution=plan.get('caution'),
                    skip=is_skip,
                    original_step=idx + 1
                ))
                continue

            matches = re.findall(pattern, action)
            if len(matches) >= 1:
                operator, params = matches[-1]
            else:
                return []

            operator = operator.strip().lower()
            params = params.strip().lower()
            action = f'{operator}({params})'
            example_planning.append(dict(
                action=action, 
                caution=plan.get('caution'),
                skip=is_skip,
                original_step=idx + 1
            ))

        return example_planning

    @abstractmethod
    def execute_plan(self, plan: StepwisePlan):
        pass

    @abstractmethod
    def termination_evaluation(self):
        pass
