import os
from pathlib import Path

import requests
from msal import PublicClientApplication, SerializableTokenCache

GRAPH_API_ENDPOINT = "https://graph.microsoft.com/v1.0"


class OneDriveClient:
    def __init__(
        self,
        client_id,
        tenant="consumers",
        cache_path="token_cache.bin",
        root_path="/Documents/OfficeArena",
    ):
        self.client_id = client_id
        self.authority = f"https://login.microsoftonline.com/{tenant}"
        self.scope = ["Files.ReadWrite.All", "User.Read"]
        self.cache_path = cache_path
        self.root_path = root_path.strip("/")
        self.token_cache = self._load_cache()
        self.app = PublicClientApplication(
            client_id=self.client_id,
            authority=self.authority,
            token_cache=self.token_cache,
        )
        self.access_token = self._get_token()
        self.session = requests.Session()
        self.session.headers.update(self._headers())

    def _load_cache(self):
        cache = SerializableTokenCache()
        if os.path.exists(self.cache_path):
            cache.deserialize(open(self.cache_path, "r").read())
        return cache

    def _save_cache(self):
        if self.token_cache.has_state_changed:
            with open(self.cache_path, "w") as f:
                f.write(self.token_cache.serialize())

    def _get_token(self):
        accounts = self.app.get_accounts()
        if accounts:
            result = self.app.acquire_token_silent(self.scope, account=accounts[0])
        else:
            flow = self.app.initiate_device_flow(scopes=self.scope)
            print(f"Go to: {flow['verification_uri']} and enter code: {flow['user_code']}")
            result = self.app.acquire_token_by_device_flow(flow)
        if "access_token" in result:
            self._save_cache()
            return result["access_token"]
        else:
            raise Exception(f"Failed to acquire token: {result.get('error_description')}")

    def _headers(self):
        return {"Authorization": f"Bearer {self.access_token}"}

    def _resolve_path(self, path):
        path = path.strip("/")
        if path:
            return f"{GRAPH_API_ENDPOINT}/me/drive/root:/{self.root_path}/{path}"
        else:
            return f"{GRAPH_API_ENDPOINT}/me/drive/root:/{self.root_path}"

    def list_directory(self, path=""):
        url = f"{self._resolve_path(path)}:/children"
        resp = self.session.get(url)
        resp.raise_for_status()
        return resp.json().get("value", [])

    def upload_file(self, local_path, remote_path, set_public=True):
        """
        Upload a file to OneDrive and optionally set it to public access.

        Args:
            local_path (str): Local file path
            remote_path (str): Remote file path relative to root_path
            set_public (bool): Whether to set the file to public access (default: True)

        Returns:
            dict: Upload response with optional sharing link info
        """
        with open(local_path, "rb") as f:
            url = f"{self._resolve_path(remote_path)}:/content"
            resp = self.session.put(url, data=f)
            resp.raise_for_status()
            upload_result = resp.json()

        # Set public access if requested
        if set_public:
            try:
                sharing_link = self._set_public_access(remote_path)
                upload_result["sharing_link"] = sharing_link
                print(f"File '{remote_path}' uploaded and set to public access")
            except Exception as e:
                print(f"Warning: Could not set public access for '{remote_path}': {e}")

        return upload_result

    def upload_directory(self, local_dir, remote_path=""):
        for root, _, files in os.walk(local_dir):
            for file in files:
                rel_path = os.path.relpath(os.path.join(root, file), local_dir)
                remote_file_path = os.path.join(remote_path, rel_path).replace("\\", "/")
                self.upload_file(os.path.join(root, file), remote_file_path)

    def download_directory(self, remote_path, local_dir):
        items = self.list_directory(remote_path)
        os.makedirs(local_dir, exist_ok=True)
        for item in items:
            if "file" in item:
                download_url = item["@microsoft.graph.downloadUrl"]
                file_resp = self.session.get(download_url)
                file_resp.raise_for_status()
                with open(os.path.join(local_dir, item["name"]), "wb") as f:
                    f.write(file_resp.content)

    def rename_directory(self, old_path, new_name):
        url = self._resolve_path(old_path)
        resp = self.session.patch(
            url,
            headers={"Content-Type": "application/json"},
            json={"name": new_name},
        )
        resp.raise_for_status()
        return resp.json()

    def copy_paste(self, source_path, target_path, new_name=None):
        url = f"{self._resolve_path(source_path)}:/copy"
        payload = {"parentReference": {"path": f"/drive/root:/{self.root_path}/{target_path.strip('/')}"}}
        if new_name:
            payload["name"] = new_name
        resp = self.session.post(url, json=payload)
        if resp.status_code == 202:
            print("Copy started.")
        else:
            raise Exception(f"Copy failed: {resp.status_code} {resp.text}")

    def get_root_id(self):
        """Returns the ID of the OneDrive root folder."""
        if not hasattr(self, "_root_id"):
            response = self.session.get(f"{GRAPH_API_ENDPOINT}/me/drive/root")
            response.raise_for_status()
            self._root_id = response.json()["id"]
        return self._root_id

    def _list_items_by_id(self, item_id):
        url = f"{GRAPH_API_ENDPOINT}/me/drive/items/{item_id}/children"
        resp = self.session.get(url)
        resp.raise_for_status()
        return resp.json().get("value", [])

    def create_folder_recursive(self, sub_path: str, set_public=True):
        """
        Create nested folders recursively under self.root_path,
        returning the ID of the deepest folder created/found.

        Args:
            sub_path (str): Path to create relative to root_path
            set_public (bool): Whether to set folders to public access (default: True)
        """
        root_parts = Path(self.root_path).parts
        sub_parts = Path(sub_path).parts

        # Start from root folder ID
        current_id = self.get_root_id()  # Traverse/create folders for root_path (if nested)
        for part in root_parts:
            children = self._list_items_by_id(current_id)
            folder = next((i for i in children if i["name"] == part and "folder" in i), None)
            if folder:
                current_id = folder["id"]
            else:
                url = f"{GRAPH_API_ENDPOINT}/me/drive/items/{current_id}/children"
                resp = self.session.post(
                    url,
                    json={
                        "name": part,
                        "folder": {},
                        "@microsoft.graph.conflictBehavior": "replace",
                    },
                )
                resp.raise_for_status()
                current_id = resp.json()["id"]

                # Set public access for newly created folder
                if set_public:
                    try:
                        # Build path for the folder we just created
                        folder_path = "/".join(root_parts[: root_parts.index(part) + 1])
                        self._set_folder_public_access(folder_path)
                        print(f"Folder '{folder_path}' created and set to public access")
                    except Exception as e:
                        print(f"Warning: Could not set public access for folder '{part}': {e}")

        # Now create/traverse the subfolders under root_path
        for i, part in enumerate(sub_parts):
            children = self._list_items_by_id(current_id)
            folder = next((i for i in children if i["name"] == part and "folder" in i), None)
            if folder:
                current_id = folder["id"]
            else:
                url = f"{GRAPH_API_ENDPOINT}/me/drive/items/{current_id}/children"
                resp = self.session.post(
                    url,
                    json={
                        "name": part,
                        "folder": {},
                        "@microsoft.graph.conflictBehavior": "replace",
                    },
                )
                resp.raise_for_status()
                current_id = resp.json()["id"]

                # Set public access for newly created folder
                if set_public:
                    try:
                        # Build the full path for the folder we just created
                        folder_path = "/".join(sub_parts[: i + 1]) if sub_path else part
                        self._set_folder_public_access(folder_path)
                        print(f"Folder '{folder_path}' created and set to public access")
                    except Exception as e:
                        print(f"Warning: Could not set public access for folder '{part}': {e}")

        return current_id

    def upload_directory_recursive(self, local_dir, remote_path=""):
        """
        Recursively upload a local directory to OneDrive, creating folders as needed.
        This preserves the entire directory structure.
        """
        local_dir = Path(local_dir)
        if not local_dir.exists():
            raise FileNotFoundError(f"Local directory does not exist: {local_dir}")

        # First, create the base remote directory if it doesn't exist
        if remote_path:
            self.create_folder_recursive(remote_path)

        # Walk through the local directory structure
        for root, dirs, files in os.walk(local_dir):
            # Calculate the relative path from the base local directory
            rel_path = os.path.relpath(root, local_dir)

            # Build the remote path
            if rel_path == ".":
                current_remote_path = remote_path
            else:
                current_remote_path = os.path.join(remote_path, rel_path).replace("\\", "/") if remote_path else rel_path.replace("\\", "/")

            # Create directories first
            for dir_name in dirs:
                dir_remote_path = os.path.join(current_remote_path, dir_name).replace("\\", "/") if current_remote_path else dir_name
                print(f"Creating directory: {dir_remote_path}")
                self.create_folder_recursive(dir_remote_path)

            # Upload files in the current directory
            for file_name in files:
                local_file_path = os.path.join(root, file_name)
                remote_file_path = os.path.join(current_remote_path, file_name).replace("\\", "/") if current_remote_path else file_name
                print(f"Uploading: {local_file_path} -> {remote_file_path}")
                self.upload_file(local_file_path, remote_file_path)

    def get_edit_link(self, file_path):
        """
        Get the edit link for a file given its path relative to the root.

        Args:
            file_path (str): Path to the file relative to self.root_path

        Returns:
            str: The edit link URL for the file

        Raises:
            Exception: If the file is not found or doesn't support editing
        """
        url = f"{self._resolve_path(file_path)}"
        resp = self.session.get(url)
        resp.raise_for_status()

        file_info = resp.json()
        file_id = file_info["id"]

        # First, try to get an existing edit link
        permissions_url = f"{GRAPH_API_ENDPOINT}/me/drive/items/{file_id}/permissions"
        response = self.session.get(permissions_url)
        if response.status_code == 200:
            permissions = response.json().get("value", [])
            for perm in permissions:
                if perm.get("link", {}).get("type") == "edit" and "/f/" not in perm["link"]["webUrl"]:
                    print(f"Edit link for file '{file_path}' already exists; returning it.")
                    return perm["link"]["webUrl"]  # Return existing link if found

        print(f"Edit link for file '{file_path}' DOESN'T currently exist; creating it.")
        create_link_url = f"{GRAPH_API_ENDPOINT}/me/drive/items/{file_id}/createLink"
        data = {"type": "edit", "scope": "anonymous"}
        response = self.session.post(create_link_url, json=data)
        response.raise_for_status()

        return response.json()["link"]["webUrl"]

    def _set_public_access(self, file_path):
        """
        Set a file to have public edit access by creating a sharing link.

        Args:
            file_path (str): Path to the file relative to self.root_path

        Returns:
            str: The edit link URL
        """
        url = f"{self._resolve_path(file_path)}"
        resp = self.session.get(url)
        resp.raise_for_status()

        file_info = resp.json()
        file_id = file_info["id"]

        # First, try to get an existing edit link
        permissions_url = f"{GRAPH_API_ENDPOINT}/me/drive/items/{file_id}/permissions"
        response = self.session.get(permissions_url)
        if response.status_code == 200:
            permissions = response.json().get("value", [])
            for perm in permissions:
                if perm.get("link", {}).get("type") == "edit" and "/f/" not in perm["link"]["webUrl"]:
                    return perm["link"]["webUrl"]  # Return existing link if found

        # Create new edit link
        create_link_url = f"{GRAPH_API_ENDPOINT}/me/drive/items/{file_id}/createLink"
        payload = {
            "type": "edit",  # Edit access
            "scope": "anonymous",  # Public access (no sign-in required)
        }
        resp = self.session.post(create_link_url, json=payload)
        resp.raise_for_status()
        return resp.json()["link"]["webUrl"]

    def _set_folder_public_access(self, folder_path):
        """
        Set a folder to have public edit access by creating a sharing link.

        Args:
            folder_path (str): Path to the folder relative to self.root_path

        Returns:
            str: The edit link URL
        """
        url = f"{self._resolve_path(folder_path)}"
        resp = self.session.get(url)
        resp.raise_for_status()

        folder_info = resp.json()
        folder_id = folder_info["id"]

        # First, try to get an existing edit link
        permissions_url = f"{GRAPH_API_ENDPOINT}/me/drive/items/{folder_id}/permissions"
        response = self.session.get(permissions_url)
        if response.status_code == 200:
            permissions = response.json().get("value", [])
            for perm in permissions:
                if perm.get("link", {}).get("type") == "edit" and "/f/" not in perm["link"]["webUrl"]:
                    return perm["link"]["webUrl"]  # Return existing link if found

        # Create new edit link
        create_link_url = f"{GRAPH_API_ENDPOINT}/me/drive/items/{folder_id}/createLink"
        payload = {
            "type": "edit",  # Edit access
            "scope": "anonymous",  # Public access (no sign-in required)
        }
        resp = self.session.post(create_link_url, json=payload)
        resp.raise_for_status()
        return resp.json()["link"]["webUrl"]

    def download_file(self, remote_path, local_dir):
        """
        Downloads a single file from OneDrive.

        Args:
            remote_path (str): Remote file path relative to root_path.
            local_dir (str): The local directory to save the file in.

        Returns:
            str: The local path of the downloaded file.
        """
        url = self._resolve_path(remote_path)
        resp = self.session.get(url)
        resp.raise_for_status()
        file_info = resp.json()

        download_url = file_info.get("@microsoft.graph.downloadUrl")
        if not download_url:
            raise ValueError(f"Could not find download URL for '{remote_path}'")

        # The downloadUrl is a pre-authenticated URL and may not require the session's auth header.
        # Use a new request without the default headers.
        file_resp = requests.get(download_url)
        file_resp.raise_for_status()

        os.makedirs(local_dir, exist_ok=True)
        local_file_path = os.path.join(local_dir, file_info["name"])

        with open(local_file_path, "wb") as f:
            f.write(file_resp.content)

        print(f"Downloaded '{remote_path}' to '{local_file_path}'")
        return local_file_path
