from __future__ import annotations

from pathlib import Path

import numpy as np
import pandas as pd


def round_half_up(x: float) -> int:
    """Round to nearest integer with halves rounding away from zero."""
    if np.isnan(x):
        raise ValueError("Cannot round NaN")
    if x >= 0:
        return int(np.floor(x + 0.5))
    return int(np.ceil(x - 0.5))


def spearman_trend(yearly_series: pd.Series) -> str:
    """Classify monotonic trend using Spearman correlation on yearly averages."""
    s = yearly_series.dropna().sort_index()
    if len(s) < 2:
        return "Not Applicable"

    # Years are already ordered; Spearman reduces to Pearson corr between positions and value ranks.
    positions = pd.Series(np.arange(len(s), dtype=float))
    ranks = pd.Series(s.to_numpy(dtype=float)).rank(method="average")
    rho = positions.corr(ranks)

    if pd.isna(rho):
        return "Not Applicable"

    if rho >= 0.5:
        return "Increasing"
    if rho <= -0.5:
        return "Decreasing"
    return "Stable"


def main() -> None:
    base_dir = Path(__file__).resolve().parent
    data_path = base_dir / "air-quality-data-in-india" / "source" / "city_day.csv"
    out_path = base_dir / "result.txt"

    df = pd.read_csv(data_path, parse_dates=["Date"])

    # BTX is commonly defined as Benzene + Toluene + Xylene.
    df["BTX"] = df[["Benzene", "Toluene", "Xylene"]].sum(axis=1, min_count=1)
    df["Year"] = df["Date"].dt.year

    yearly = (
        df.groupby("Year")
        .agg(avg_BTX=("BTX", "mean"), avg_O3=("O3", "mean"))
        .sort_index()
    )

    btx_2017 = yearly.loc[2017, "avg_BTX"] if 2017 in yearly.index else np.nan
    btx_2018 = yearly.loc[2018, "avg_BTX"] if 2018 in yearly.index else np.nan

    if pd.isna(btx_2017) or pd.isna(btx_2018) or btx_2017 == 0:
        factor_str = "Not Applicable"
    else:
        factor_str = str(round_half_up(float(btx_2018 / btx_2017)))

    trend_str = spearman_trend(yearly.loc[2015:2020, "avg_O3"])

    if factor_str == "Not Applicable" or trend_str == "Not Applicable":
        result = "Not Applicable"
    else:
        result = f"{factor_str}; {trend_str}"

    out_path.write_text(result + "\n", encoding="utf-8")


if __name__ == "__main__":
    main()
