import os
import json
import time
import csv
import zipfile
import pandas
import pandas as pd

import os
import sys
import subprocess
import nbformat
from nbformat.v4 import new_notebook, new_code_cell

from kaggle.api.kaggle_api_extended import KaggleApi
import ast

import argparse
import json


def get_metrics(score, competition, csv_dir, file_path, api):
    print(f"Downloading leaderboard for {competition}...")
    lb_zip_path = api.competition_leaderboard_download(competition, path=csv_dir, quiet=True)
    

    lb_zip_path = file_path.replace('.csv', '.zip').replace('.py', '.zip')
    with zipfile.ZipFile(lb_zip_path, 'r') as zip_ref:
        csv_names = [name for name in zip_ref.namelist() if name.endswith('.csv')]
        if not csv_names:
            print(f"No CSV file found in {lb_zip_path}")
            return None
        inner_csv_name = csv_names[0]
        extracted_csv_path = os.path.join(csv_dir, f"{competition}_leaderboard.csv")
        with zip_ref.open(inner_csv_name) as source, open(extracted_csv_path, 'wb') as target:
            target.write(source.read())
    
    df = pd.read_csv(extracted_csv_path, encoding='utf-8')

    df['Score'] = pd.to_numeric(df['Score'], errors='coerce')

    my_pos = None

    for i in range(1, len(df)):
        prev_score = df.loc[i-1, 'Score']
        curr_score = df.loc[i, 'Score']
        if (float(curr_score) <= float(score) <= float(prev_score)) or (float(curr_score) >= float(score) >= float(prev_score)):
            my_pos = i  
            break
    
    time.sleep(10)
    return {
        'competition': competition,
        'leaderboard_position': my_pos,
        'beaten_users': 1 - my_pos/len(df),
        'score': score
    }
    


def submit_files(csv_dir, kaggle_username=None, kaggle_key=None):
    api = KaggleApi()
    api.authenticate()
    
    results = []
    
    for filename in os.listdir(csv_dir):
        if filename.endswith('.csv'):
            competition = filename.replace('.csv', '')
            file_path = os.path.join(csv_dir, filename)
            
            print(f"Submitting {filename} to {competition}...")
            try:
                api.competition_submit(file_path, "Auto submission", competition)
            except Exception as e:
                print(f"Error submitting {filename}: {e}")
                continue
            
            time.sleep(60)
            
            subs = api.competition_submissions(competition)
            my_sub = sorted(subs, key=lambda x: x.date, reverse=True)[0]
            score = my_sub.public_score
            team_name = my_sub.team_name  
            results.append(get_metrics(score, competition, csv_dir, file_path, api))
            
           
    
    return results


def py_to_ipynb(py_path, preproc_code, ipynb_path):
    with open(py_path, 'r', encoding='utf-8') as f:
        code = f.read()
    with open('kaggle_moving.py', 'r', encoding='utf-8') as f:
        move_code = f.read()
    nb = new_notebook(cells=[new_code_cell(move_code), new_code_cell(preproc_code), new_code_cell(code)])
    with open(ipynb_path, 'w', encoding='utf-8') as f:
        nbformat.write(nb, f)

def create_kaggle_metadata(title, competition_id, kaggle_username):
    return {
        "title": title,
        "id": f"{kaggle_username}/{title.replace(' ', '_')}",
        "language": "python",
        "kernel_type": "notebook",
        "is_private": "true",
        "enable_gpu": "false",
        "enable_internet": "false",
        "code_file": f'{competition_id}.ipynb',
        "competition_sources": [competition_id]
    }

def write_kernel_metadata(ipynb_dir, title, competition_id, kaggle_username):
    meta = create_kaggle_metadata(title, competition_id, kaggle_username)
    import json
    with open(os.path.join(ipynb_dir, 'kernel-metadata.json'), 'w', encoding='utf-8') as f:
        json.dump(meta, f, indent=2)

def extract_function(filename, func_name):
    with open(filename, "r", encoding="utf-8") as f:
        source = f.read()
    tree = ast.parse(source)
    for node in tree.body:
        if isinstance(node, ast.FunctionDef) and node.name == func_name:
            start_line = node.lineno - 1
            end_line = node.end_lineno
            lines = source.splitlines()
            func_code = "\n".join(lines[start_line:end_line])
            return func_code
    return None

def get_function_names(filename):
    with open(filename, "r", encoding="utf-8") as f:
        tree = ast.parse(f.read(), filename=filename)
    return [node.name for node in ast.walk(tree) if isinstance(node, ast.FunctionDef)]


def submit_notebook(folder_path, kaggle_username):
    api = KaggleApi()
    api.authenticate()
    assert os.path.isdir(folder_path), "Directory not founded"
    results = []
    preproc_code = ''
    functions = get_function_names('./custom_dataset_preparation.py')
    for filename in os.listdir(folder_path):
        if filename.endswith('.py'):
            py_path = os.path.join(folder_path, filename)
            competition_id = os.path.splitext(filename)[0]
            for func in functions:
                print(func, competition_id)
                if func.replace('prepare_', '').replace('_', '-') in competition_id:
                    preproc_code += 'import os\nimport zipfile\nimport shutil\nimport pandas as pd\nimport glob\n'
                    preproc_code += extract_function("./custom_dataset_preparation.py", func)
                    preproc_code += f"\n{func}('/kaggle/working/{competition_id}/')"
            base = competition_id
            ipynb_name = base + ".ipynb"
            ipynb_path = os.path.join(folder_path, ipynb_name)

            py_to_ipynb(py_path, preproc_code,  ipynb_path)
            print(f"Notebook created: {ipynb_path}")

            write_kernel_metadata(folder_path, base, competition_id, kaggle_username)

            cmd = [
                "kaggle", "kernels", "push", "-p", folder_path
            ]
            subprocess.run(cmd, check=True)

            competition_url = f"https://www.kaggle.com/competitions/{competition_id}"
            print(f"Link to competition: {competition_url}")

            kernel_slug = base.lower().replace('_', '-').replace(' ', '-')
            kernel_url = f"https://www.kaggle.com/code/{kaggle_username}/{kernel_slug}"
            print(f"Link to notebook: {kernel_url}")

            score = input(f"Please, enter score for competition '{competition_id}': ")
            results.append(get_metrics(score, competition_id, folder_path, py_path, api))

    return results



parser = argparse.ArgumentParser(description='Submit and get leaderboard, then submit notebook.')
parser.add_argument('path', help='Path to dir with submissions')
parser.add_argument('user', help='Kaggle username')

args = parser.parse_args()

res1 = submit_files(args.path)
res2 = submit_notebook(args.path, args.user)

res = res1 + res2

with open('competition_report.json', 'w', encoding='utf-8') as f:
    json.dump(res, f, ensure_ascii=False, indent=4)