import os
import sys
import logging
import subprocess
import traceback
import shutil
from src.utils import underscore_to_pascalcase, camel_to_snake
from typing import Optional

logger = logging.getLogger(__name__)

def ascend_compile(generated_code, op, context, op_engineer_dir, ascendc_device):
    logger.info(f"[{op}]Start compiling {op}")
    op = op + '_custom'
    
    op_capital=underscore_to_pascalcase(op)
    target_directory=os.path.join(op_engineer_dir, op_capital)

    try:
        compile(generated_code, "<string>", "exec")
        exec(generated_code, context)  # For Python, use exec() (be careful with untrusted code)
    except Exception as e:
        msg = traceback.format_exc()
        logger.error(f'Error in generated code {msg}')
        raise Exception(f'Error in generated code {msg}')

    # create ascendc project
    if os.path.exists(target_directory):
        logger.info(f"[{op}] [INFO] Operator project already exists, deleted")
        shutil.rmtree(target_directory)
    
    with open(os.path.join(op_engineer_dir, f'{op}.json'), 'w') as f:
        f.write(context.get('project_json_src'))
    try:
        logger.info(f"[{op}] [INFO] Begin create operator project")
        os.chdir(op_engineer_dir)
        result = subprocess.run(["msopgen", 'gen', '-i', f'{op}.json', '-c', ascendc_device, '-lan', 'cpp', '-out', op_capital], check=True, capture_output=True, text=True)
        logger.info(f"[{op}][INFO] Create operator project succeeded")
    except subprocess.CalledProcessError as e:
        logger.info(f"[{op}] [INFO] Create operator project failed!")
        logger.error("Exit Code:", e.returncode)
        logger.error("Error Output:\n", e.stdout)
        logger.error("Error Output:\n", e.stderr)
        feedback = f'Exit Code: {e.returncode}\nError Output:\n{e.stdout}'
        raise Exception(feedback) 

    # write code to specific location
    op_name = camel_to_snake(op_capital)
    host_operator_src = context.get('host_operator_src',"")
    kernel_src = context.get('kernel_src',"")

    host_operator_src = host_operator_src.replace(op, op_name)
    kernel_src = kernel_src.replace(op, op_name)

    with open(os.path.join(target_directory, 'op_host', f'{camel_to_snake(op_capital)}_tiling.h'), 'w') as f:
        f.write(context.get('host_tiling_src'))

    with open(os.path.join(target_directory, 'op_host', f'{camel_to_snake(op_capital)}.cpp'), 'w') as f:
        f.write(host_operator_src)

    with open(os.path.join(target_directory, 'op_kernel', f'{camel_to_snake(op_capital)}.cpp'), 'w') as f:
        f.write(kernel_src)

    with open(os.path.join(op_engineer_dir, 'CppExtension', 'csrc', f'op.cpp'), 'w') as f:
        f.write(context.get('python_bind_src'))

    try:
        environ_varible = 'ASCEND_CUSTOM_OPP_PATH' # this varible will purturb build if set
        os.environ.pop(environ_varible, None)
        logger.info(f"[{op}] [INFO] Begin build")
        os.chdir(target_directory)
        result = subprocess.run(["./build.sh"], check=True, capture_output=True, text=True)
        logger.info(f"[{op}] [INFO] Build succeeded")
    except subprocess.CalledProcessError as e:
        logger.error(f"[{op}] [INFO] Build failed!")
        error_output = ''
        for line in e.stdout.split('\n'):
            if '[ERROR]' in line or 'error:' in line:
                logger.error(line)
                error_output += line
                error_output += '\n'
        for line in e.stderr.split('\n'):
            if '[ERROR]' in line or 'error:' in line:
                logger.error(line)
                error_output += line
                error_output += '\n'
        feedback = f'Exit Code: {e.returncode}\nError Output:\n{error_output}'
        raise Exception(feedback)

    try:
        logger.info(f"[{op}] [INFO] Begin deploy")
        os.chdir(os.path.join(target_directory, 'build_out'))
        deploy_path = os.path.join(op_engineer_dir, 'opp')
        result = subprocess.run(["./custom_opp_openEuler_aarch64.run", f'--install-path={deploy_path}'], check=True, capture_output=True, text=True)
        logger.info(f"[{op}] [INFO] Deploy succeeded")
    except subprocess.CalledProcessError as e:
        logger.error(f"[{op}] [INFO] Deploy failed!")
        feedback = f'Exit Code: {e.returncode}\nError Output:\n{e.stdout}'
        raise Exception(feedback)

    try:
        logger.info(f"[{op}] [INFO] Begin pybind")
        os.chdir(os.path.join(op_engineer_dir, 'CppExtension'))
        pybind_path=os.path.join(op_engineer_dir, 'CppExtension', 'pybind_install')
        os.environ['PYBIND_DEST']=pybind_path
        result = subprocess.run(['bash', "build_and_run.sh"], check=True, capture_output=True, text=True)
        logger.info(f"[{op}] [INFO] Pybind succeeded\n")
    except subprocess.CalledProcessError as e:
        # Print error if build.sh fails
        logger.error(f"[{op}] [INFO] Pybind failed!")
        feedback = f'Exit Code: {e.returncode}\nError Output:\n{e.stdout}'
        raise Exception(feedback)
    # Update PYTHONPATH and sys.path
    existing_py_path = os.environ.get("PYTHONPATH", "")
    if existing_py_path:
        py_paths = existing_py_path.split(os.pathsep)
    else:
        py_paths = []
    if pybind_path not in py_paths:
        py_paths.insert(0, pybind_path)
        os.environ["PYTHONPATH"] = os.pathsep.join(py_paths)
    if pybind_path not in sys.path:
        sys.path.insert(0, pybind_path)
    
    if "custom_ops_lib" in sys.modules:
        del sys.modules["custom_ops_lib"]

    # Update ASCEND_CUSTOM_OPP_PATH
    custom_opp_path = f"{deploy_path}/vendors/customize"
    os.environ["ASCEND_CUSTOM_OPP_PATH"] = custom_opp_path

    # Update LD_LIBRARY_PATH
    if 'ascend_op_projects' not in os.environ["LD_LIBRARY_PATH"]:
        custom_lib_path = f"{deploy_path}/vendors/customize/op_api/lib/"
        existing_ld_path = os.environ.get("LD_LIBRARY_PATH", "")
        os.environ["LD_LIBRARY_PATH"] = f"{custom_lib_path}:{existing_ld_path}"
    
    try:
        compile(context['model_src'], "<string>", "exec")
        exec(context['model_src'], context)  # For Python, use exec() (be careful with untrusted code)
    except Exception as e:
        raise Exception(f'Error in generated code {e}')
