import ast
import json
import os

from UIAgents.Agent_RPA.utils import JSON_models


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):
    self.rpa_dict = {}
    self.file_path = os.path.join(save_path, file_name)
    
    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):
    with open(self.file_path, 'w') as wf:
      json.dump(self.rpa_dict, wf, indent=4)
  
  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: JSON_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)