import logging
from typing import Union

import discord
import humanfriendly
import psutil
import requests
import yaml
from discord.utils import utcnow

# imagine a world where devs imported exactly what they needed
# but it's just not possible /s

logger = logging.getLogger(__name__)


# Basic Loading with some error handling
def load_config():
    try:
        with open("config.yaml", "r") as file:
            config = yaml.safe_load(file)
            logger.info("Configuration file loaded successfully.")
            return config
    except FileNotFoundError:
        logger.error("Config file not found. Please ensure 'config.yaml' exists.")
        return {}
    except yaml.YAMLError as exc:
        logger.error(f"Error parsing YAML config: {exc}")
        return {}


# Unused Saving
def save_config(config):
    try:
        with open("config.yaml", "w") as file:
            yaml.safe_dump(config, file)
            logger.info("Configuration file saved successfully.")
    except Exception as e:
        logger.error(f"Error saving config: {e}")


# is_mod, takes a Member, returns a bool
# not the best as DM'ing the bot won't produce a list of roles and thus won't count mods.
# i just don't want to rewrite the yaml
def is_mod(member):
    config = load_config()
    mod_ids = config.get("mods", [])

    if str(member.id) in mod_ids:
        return True

    for role in member.roles:
        if str(role.id) in mod_ids:
            return True

    return False


# Unused Decorator for commands.
# You'll come to learn there's a good amount of stuff that's just easier with commands rather than slash commands.
# See robocraft.py for some examples
def mod_only():
    def wrapper(func):
        async def wrapped(self, ctx, *args, **kwargs):
            if not is_mod(ctx.author):
                logger.warning(
                    f"{ctx.author} tried to access {ctx.command} without permission."
                )
                await ctx.send("You do not have permission to use this command.")
                return
            return await func(self, ctx, *args, **kwargs)

        return wrapped

    return wrapper


# Gross function, but it works.
def handle_api_request(route_key, data=None) -> tuple[Union[dict, list, None], str]:
    # route_key is your YAML key
    # Data defaults to None so we don't have to set it

    config = load_config()

    # If you wanted to set up multiple API URLS, you could use them like routes (a list), or just copy how api_url is setup.
    api_url = config.get("api_url")
    api_token = config.get("api_token")
    routes = config.get("routes", {})

    if not api_url:  # you fucked up
        logger.error("API URL is not configured.")
        return None, "API URL is not configured."

    if route_key not in routes:  # you also fucked up
        logger.warning(f"Invalid route key: {route_key}")
        return None, "Invalid API route key."

    route = routes[route_key]
    url = f"{api_url}{route}"
    headers = {"Authorization": api_token}

    try:  # Very prone to errors. Server shite.
        if data:
            response = requests.post(url, json=data, headers=headers)
        else:
            response = requests.get(url, headers=headers)

        if response.status_code == 200:
            logger.info(f"API request to {url} succeeded.")
            return response.json(), "Success!"
        else:
            return None, f"Error {response.status_code}: {response.text}"

    except Exception as e:
        logger.error(f"Request failed: {e}")
        return None, f"Request failed: {str(e)}"


# Baller about me embed, don't worry too much about this
# Good insight into how embeds work though.
def about_me_embed(bot):
    cpu = psutil.cpu_percent()
    memory = psutil.virtual_memory().percent
    disk = psutil.disk_usage("/")
    disk = {
        "used_gb": round(disk.used / (1024.0**3), 2),
        "available_gb": round(disk.free / (1024.0**3), 2),
        "total_gb": round(disk.total / (1024.0**3), 2),
    }
    uptime = utcnow() - bot.uptime
    if bot.src:  # remember that?
        description = f"This bot is [**Open Source!**](<{bot.src}>)"
    else:
        description = "A Discord Bot Written in Python!"  # idk

    # make the embed
    embed = discord.Embed(
        title=f"About: {bot.user.name}",
        description=description,
        color=discord.Color.random(),
    )
    # add fields
    embed.add_field(name="CPU: ", value=f"{cpu}% of {psutil.cpu_count()} cores")
    embed.add_field(
        name="Memory: ",
        value=f"{memory}% of {round(psutil.virtual_memory().total / (1024.0 ** 3), 2)} GB",
    )
    embed.add_field(
        name="Disk Usage: ", value=f"{disk['used_gb']} GB used of {disk['total_gb']} GB"
    )
    embed.add_field(name="Uptime: ", value=f"{humanfriendly.format_timespan(uptime)}")

    # add avatar and footer
    embed.set_image(url=bot.user.display_avatar.url)
    embed.set_footer(text="Made with :3 by Flo ❤️", icon_url=bot.user.display_avatar.url)
    return embed