"""FileLogger class for logging to a file."""

# pylint: disable-all
# Temporarily disable all pylint checkers during AST traversal to prevent crash.
# The imports checker crashes when resolving fwh_core package imports due to a bug
# in pylint/astroid: https://github.com/pylint-dev/pylint/issues/10185
# pylint: enable=all
# Re-enable all pylint checkers for the checking phase. This allows other checks
# (code quality, style, undefined names, etc.) to run normally while bypassing
# the problematic imports checker that would crash during AST traversal.

import json
import shutil
from collections.abc import Mapping
from pathlib import Path
from typing import Any

import matplotlib.figure
import mlflow
import numpy
import PIL.Image
import plotly.graph_objects
from omegaconf import DictConfig, OmegaConf

from fwh_core.logging.logger import Logger


class FileLogger(Logger):
    """Logs to a file."""

    def __init__(self, file_path: str):
        self.file_path = file_path
        try:
            Path(self.file_path).parent.mkdir(parents=True, exist_ok=True)
        except PermissionError as e:
            raise RuntimeError(f"Failed to create directory for logging: {e}") from e

    def log_config(self, config: DictConfig, resolve: bool = False) -> None:
        """Log config to the file."""
        with open(self.file_path, "a") as f:
            _config = OmegaConf.to_container(config, resolve=resolve)
            print(f"Config: {_config}", file=f)

    def log_metrics(self, step: int, metric_dict: Mapping[str, Any]) -> None:
        """Log metrics to the file."""
        with open(self.file_path, "a") as f:
            print(f"Metrics at step {step}: {metric_dict}", file=f)

    def log_params(self, param_dict: Mapping[str, Any]) -> None:
        """Log params to the file."""
        with open(self.file_path, "a") as f:
            print(f"Params: {param_dict}", file=f)

    def log_tags(self, tag_dict: Mapping[str, Any]) -> None:
        """Log tags to the file."""
        with open(self.file_path, "a") as f:
            print(f"Tags: {tag_dict}", file=f)

    def log_figure(
        self,
        figure: matplotlib.figure.Figure | plotly.graph_objects.Figure,
        artifact_file: str,
        **kwargs,
    ) -> None:
        """Save figure to file system."""
        figure_path = Path(self.file_path).parent / artifact_file
        figure_path.parent.mkdir(parents=True, exist_ok=True)

        # Handle different figure types
        if isinstance(figure, matplotlib.figure.Figure):
            figure.savefig(str(figure_path), **kwargs)
        elif isinstance(figure, plotly.graph_objects.Figure):
            if str(figure_path).endswith(".html"):
                figure.write_html(str(figure_path), **kwargs)
            else:
                figure.write_image(str(figure_path), **kwargs)
        else:
            raise ValueError(f"Unsupported figure type: {type(figure)}")

        with open(self.file_path, "a") as f:
            print(f"Figure saved: {figure_path}", file=f)

    def _save_image_to_path(
        self, image: numpy.ndarray | PIL.Image.Image | mlflow.Image, image_path: Path, **kwargs
    ) -> bool:
        """Save image to specified path. Returns True if successful, False otherwise."""
        try:
            if isinstance(image, PIL.Image.Image):
                image.save(str(image_path), **kwargs)
            elif isinstance(image, numpy.ndarray):
                PIL.Image.fromarray(image).save(str(image_path), **kwargs)
            elif isinstance(image, mlflow.Image):
                # MLflow Image objects need special handling - convert to PIL first
                pil_image = image.to_pil()
                pil_image.save(str(image_path), **kwargs)
            else:
                # Unsupported image type
                with open(self.file_path, "a") as f:
                    print(f"Image type {type(image).__name__} not supported for file saving", file=f)
                return False
            return True
        except Exception as e:
            # Log any save errors
            with open(self.file_path, "a") as f:
                print(f"Failed to save image: {e}", file=f)
            return False

    def log_image(
        self,
        image: numpy.ndarray | PIL.Image.Image | mlflow.Image,
        artifact_file: str | None = None,
        key: str | None = None,
        step: int | None = None,
        **kwargs,
    ) -> None:
        """Save image to file system."""
        # Parameter validation - ensure we have either artifact_file or (key + step)
        if not artifact_file and not (key and step is not None):
            with open(self.file_path, "a") as f:
                print("Image logging failed - need either artifact_file or (key + step)", file=f)
            return

        if artifact_file:
            # Artifact mode
            image_path = Path(self.file_path).parent / artifact_file
            image_path.parent.mkdir(parents=True, exist_ok=True)

            if self._save_image_to_path(image, image_path, **kwargs):
                with open(self.file_path, "a") as f:
                    print(f"Image saved: {image_path}", file=f)
        else:
            # Time-stepped mode (we know key and step are valid due to validation above)
            filename = f"{key}_step_{step}.png"
            image_path = Path(self.file_path).parent / filename
            image_path.parent.mkdir(parents=True, exist_ok=True)

            if self._save_image_to_path(image, image_path, **kwargs):
                with open(self.file_path, "a") as f:
                    print(f"Time-stepped image saved: {image_path}", file=f)

    def log_artifact(self, local_path: str, artifact_path: str | None = None) -> None:
        """Copy artifact to the log directory."""
        source_path = Path(local_path)
        dest_path = Path(self.file_path).parent / (artifact_path or source_path.name)
        dest_path.parent.mkdir(parents=True, exist_ok=True)

        if source_path.is_file():
            shutil.copy2(local_path, dest_path)
        else:
            shutil.copytree(local_path, dest_path, dirs_exist_ok=True)

        with open(self.file_path, "a") as f:
            print(f"Artifact copied: {local_path} -> {dest_path}", file=f)

    def log_json_artifact(self, data: dict | list, artifact_name: str) -> None:
        """Save JSON data as an artifact to the log directory."""
        json_path = Path(self.file_path).parent / artifact_name
        json_path.parent.mkdir(parents=True, exist_ok=True)

        with open(json_path, "w") as f:
            json.dump(data, f, indent=2)

        with open(self.file_path, "a") as f:
            print(f"JSON artifact saved: {json_path}", file=f)

    def close(self) -> None:
        """Close the logger."""
        pass
