import numpy as np
from typing import Iterable, Tuple, Optional

#这个不好
def find_plateau_point_pwlf(y, n_segments=2):
    """
    用 pwlf 拟合两段线性模型，找出斜率从第一段到第二段大幅降低的断点。
    假设：第一段是上升／斜率明显的；第二段斜率接近 0（即“平缓”）。
    
    参数：
      x: 一维 numpy 数组，长度 >= 3，严格单调递增或至少非减。
      y: 一维 numpy 数组，与 x 同长度。
      n_segments: 分段数，默认为2（上升段 + 平缓段）。
    
    返回：
      plateau_x: 检测到“进入平缓区域”的断点对应的 x 值（float），
                 或 None 如果拟合失败或第二段斜率不够小。
    """
    # 基本检查
    y = np.asarray(y, dtype=float)
    x = np.arange(1, len(y)+1)
    
    # 拟合 piecewise linear 两段
    try:
        pw = pwlf.PiecewiseLinFit(x, y)
        # 拟合两段线
        breaks = pw.fit(n_segments)  # breaks 是长度为 n_segments+1 的断点 x 值，包括最小 x 和最大 x
        # breaks[0] == x.min(), breaks[-1] == x.max()
    except Exception as e:
        print(e)
        return None
    
    # 断点位置（中间那个断点，也就是 breaks[1]）
    # 如果 n_segments=2，那么 breaks 是 [x0, x_break, xN]
    x_break = breaks[1]
    
    # 计算第一段斜率 和 第二段斜率
    # 对第一段 (x[0] … x_break) 做线性拟合；对第二段 (x_break … end) 做线性拟合
    # 我们用 numpy.polyfit
    # 找索引分界：第一个 x >= x_break
    idx_break = np.searchsorted(x, x_break)
    if idx_break < 2 or (len(x) - idx_break) < 2:
        # 段太短，不可靠
        return None
    
    # 拟合
    slope1, intercept1 = np.polyfit(x[:idx_break], y[:idx_break], deg=1)
    slope2, intercept2 = np.polyfit(x[idx_break - 1:], y[idx_break - 1:], deg=1)
    # 注意 idx_break -1 来确保第二段含断点那一点
    
    # 判断第二段斜率是否“足够平缓”
    # 你可以定义一个阈值，比如第二段斜率幅度 < 第一段斜率的某个比例，或绝对值 < 某个小数
    # 下面用一个默认判断：斜率2 的绝对值 < 10% 的斜率1（如果斜率1 很大）
    
    if np.abs(slope2) < np.abs(slope1) * 0.1:
        return x_break
    else:
        # 如果第二段斜率还不是很平缓，可能整体趋势还在稳定上升
        return None


def find_latency(x: Iterable[int],
                 y: Iterable[float],
                 window: int = 10,
                 improve_percent: float = 0.01
                 ) -> Tuple[Optional[int], Optional[int]]:
    """
    找到第一个满足“连续 (window-1) 次相对提升都不超过 improve_percent”的点 i，
    返回 (x[i], i)。若无则返回 (None, None)。

    参数说明：
      x: 可迭代的一维整数数组（按从小到大排序）
      y: 与 x 等长的数值型数组
      window: 窗口大小 w（从第 w 个点开始判断）；实际比较的相邻对数为 w-1
      improve_percent: 相对提升阈值 p（例如 0.01 表示 1%）

    注意：
      - 函数假设 x 和 y 长度相同。
      - 对于 y[j-1] == 0 的情况，若 y[j] == 0 则认为相对提升为 0（通过），否则视为超过阈值（不通过）。
      - 若 y 中有负值，相对提升的语义会变复杂（建议传入非负数序列）。
    """
    x_arr = np.asarray(x)
    y_arr = np.asarray(y, dtype=float)

    if x_arr.ndim != 1 or y_arr.ndim != 1:
        raise ValueError("x 和 y 必须为一维序列")
    if len(x_arr) != len(y_arr):
        raise ValueError("x 和 y 长度必须相等")
    n = len(y_arr)
    if window < 2:
        raise ValueError("window 最小应为 2（至少一次相邻比较）")

    p = float(improve_percent)

    # 从索引 i = window-1 开始判断（0-based）
    for i in range(window - 1, n):
        ok = True
        # 比较的 j 范围：从 i-(window-2) 到 i （高端索引），对应的比较是 y[j] vs y[j-1]
        start_j = i - (window - 2)
        for j in range(start_j, i + 1):
            prev = j - 1
            y_prev = y_arr[prev]
            y_curr = y_arr[j]

            # 处理 y_prev == 0 的稳健比较
            if np.isclose(y_prev, 0.0):
                # 如果两者都近似为 0，则认为相对提升 ~0；否则视为提升超过阈值
                if not np.isclose(y_curr, 0.0):
                    ok = False
                    break
                # else 都为 0，视为通过
            else:
                rel = (y_curr - y_prev) / y_prev
                if rel > p:
                    ok = False
                    break

        if ok:
            return int(x_arr[i]), int(i)

    return None, None