import ast
import json
import os
import shutil
from pathlib import Path

from . import models
from .agent_utils import print_with_color


def define_functions_from_code(code_str, local_scope):
    """
    Parses the given code string and defines functions within the provided local scope,
    ignoring functions that contain only 'pass' or '...'.
    
    Parameters:
    - code_str: A string containing Python code to parse and execute.
    - local_scope: A dictionary representing the local scope where the functions will be defined.
    """
    try:
        parsed_ast = ast.parse(code_str)
    except Exception:
        return

    for node in ast.walk(parsed_ast):
        if isinstance(node, ast.FunctionDef):
            if len(node.body) == 1 and isinstance(node.body[0], (ast.Pass, ast.Expr)):
                continue
            exec(compile(ast.Module(body=[node], type_ignores=[]), filename="<ast>", mode="exec"), local_scope)

class RPABank:
    def __init__(self, save_path="./", file_name="rpa_bank.json", load_local_bank: bool=True,
                 backup_dir="rpa_bank_backups", auto_backup=True, start_timestamp=None):
        """
        Initialize the RPABank with backup support.
        
        :param save_path: Base directory for saving the bank file
        :param file_name: Filename or absolute path for the bank file
        :param load_local_bank: Whether to load existing bank file
        :param backup_dir: Directory name for backups (relative to save_path)
        :param auto_backup: Whether to auto-backup on save
        :param start_timestamp: Program start timestamp for consistent backup naming
        """
        self.rpa_dict = {}
        self.file_path = os.path.join(save_path, file_name)
        
        # Convert to Path for easier manipulation
        file_path_obj = Path(self.file_path)
        self.save_path = file_path_obj.parent
        self.backup_dir = self.save_path / backup_dir
        
        # Backup settings
        self.auto_backup = auto_backup
        if start_timestamp is None:
            import datetime
            start_timestamp = datetime.datetime.now().strftime("%Y-%m-%d_%H-%M-%S")
        self.start_timestamp = start_timestamp
        self.backup_file_path = self.backup_dir / f"rpa_bank_{self.start_timestamp}.json"
        
        # Create backup directory
        self.backup_dir.mkdir(parents=True, exist_ok=True)
        
        if load_local_bank and os.path.exists(self.file_path):
            self.load(self.file_path)
    
    def clear(self):
        self.rpa_dict = {}
    
    def load(self, file_path: str):
        with open(file_path, 'r') as rf:
            self.rpa_dict = json.load(rf)
        
    def save(self):
        """
        Save rpa_bank to file with backup support.
        
        Result:
          - Backup the old file before overwriting (if exists)
          - Overwrite the old file with new content
          - Backup the new file (overwrite the old backup file created before)
        """
        file_path_obj = Path(self.file_path)
        
        # Backup old file before overwriting (if exists)
        if self.auto_backup and file_path_obj.exists():
            try:
                shutil.copy2(file_path_obj, self.backup_file_path)
                print_with_color(f"📦 Backed up old version to: {self.backup_file_path.name}", 'cyan')
            except Exception as e:
                print_with_color(f"⚠️  Failed to backup old file: {e}", 'yellow')
        
        # Save the current rpa_dict to file
        with open(self.file_path, 'w') as wf:
            json.dump(self.rpa_dict, wf, indent=4)
        
        print_with_color(f"💾 Saved rpa_bank to: {self.file_path}", 'green')
        
        # Create backup after saving (ensures we always have a backup)
        if self.auto_backup:
            self._create_post_save_backup()
    
    def _create_post_save_backup(self):
        """Create backup after saving."""
        try:
            # Copy the just-saved JSON file to back up
            file_path_obj = Path(self.file_path)
            shutil.copy2(file_path_obj, self.backup_file_path)
            print_with_color(f"📦 Backup created: {self.backup_file_path.name}", 'cyan')
        except Exception as e:
            print_with_color(f"⚠️  Failed to create backup: {e}", 'yellow')
        
    def save_temp(self, task_type: str = None, save_path: str = None, file_name='candidate_rpa.json'):
        """
        task_type:
            None: save the whole rpa bank to file
            not None: save the target rpa to file
        """
        content = self.rpa_dict if task_type is None else self.rpa_dict[task_type]
        with open(os.path.join(save_path, file_name), 'w') as wf:
            json.dump(content, wf, indent=4)

    def add_rpa(self, rpa_info: models.RPAInfo):
        self.rpa_dict[rpa_info.task_type] = {
            'rpa_description': rpa_info.rpa_description,
            'rpa_params': rpa_info.parameters,
            'example_usage':rpa_info.example_usage,
            'rpa_code': rpa_info.rpa_code,
            'verified_tasks': []
        }
    
    # Not used
    def get_prompt_relevant_rpa(self, query_task_type, local_scope):
        # Use exact matching to retrieve relevant rpas
        if query_task_type in self.rpa_dict:
            rpa_description = self.rpa_dict[query_task_type]['rpa_description']
            rpa_input = self.rpa_dict[query_task_type]['rpa_input']
            rpa_code = self.rpa_dict[query_task_type]['rpa_code']
            if self.rpa_dict[query_task_type]['success']>0:
                define_functions_from_code(rpa_code, local_scope)
                output = f"\nHere is the code for a relevant rpa:\nThe task is to: {rpa_description}\n```python\n{rpa_code}\n```"
                if self.rpa_dict[query_task_type]['success']==1:
                    output += "\nPlease pay close attention to the process and details of this successful code when writing code. Also, be aware of pause moments and potential randomness (the current environment may differ from this one). You can call helper methods directly in rpa code, but not variables."
            else:
                output = f"\nHere is an failure record from a previous task:\nThe task is to: {rpa_description}\n{rpa_code}\nPlease also analyze the strategy or mistakes in this failure record, and consider how to alter the strategy or avoid the same problems."
        else:
            output = ""
        return output

    def merge_from(self, other_bank):
        """
        Merge another RPABank into this one.
        If both contain the same task_type, the other_bank's entry will overwrite this one.
        """
        self.rpa_dict.update(other_bank.rpa_dict)
    
    def update_verified_tasks(self, task_type: str, verified_tasks: list[str]):
        new_list = list(set(self.rpa_dict[task_type]['verified_tasks'] + verified_tasks))
        self.rpa_dict[task_type]['verified_tasks_num'] = len(new_list)
        self.rpa_dict[task_type]['verified_tasks'] = new_list
    
    def update_based_on_task(self, task_type: str, based_on_task: int):
        """
            based_on_task indicates which task the RPA was generated from
        """
        self.rpa_dict[task_type]['based_on_task'] = str(based_on_task)