import numpy as np
from scipy.integrate import solve_ivp


def solve_VdP(initial_conditions, t_eval):
    def VdP(t, y):
        """ODE system."""
        mu = 1
        x1, x2 = y[0], y[1]
        y_dot = np.zeros(2)

        y_dot[0] = x2
        y_dot[1] = mu * (1 - x1 ** 2) * x2 - x1
        return y_dot

    assert initial_conditions.ndim == 2
    assert initial_conditions.shape[1] == 2
    trajectories = []

    for idx, ic in enumerate(initial_conditions):
        solution = solve_ivp(
            VdP, t_span=(t_eval[0], t_eval[-1]), y0=ic, t_eval=t_eval, method='DOP853'
        )
        trajectories.append(solution["y"])

    return np.array(trajectories)


def solve_lorenz_system(initial_conditions, t_eval):
    assert initial_conditions.ndim == 2
    assert initial_conditions.shape[1] == 3

    def lorenz(t, y):
        """ODE system."""
        sigma = 10.
        beta = 8. / 3
        rho = 28.
        x, y, z = y[0], y[1], y[2]
        y_dot = np.zeros(3)

        y_dot[0] = sigma * (y - x)
        y_dot[1] = x * (rho - z) - y
        y_dot[2] = x * y - beta * z
        return y_dot

    trajectories = []

    for idx, ic in enumerate(initial_conditions):
        solution = solve_ivp(
            lorenz, t_span=(t_eval[0], t_eval[-1]), y0=ic, t_eval=t_eval, method='DOP853'
        )
        trajectories.append(solution["y"])

    return np.array(trajectories)


def solve_rossler_system(initial_conditions, t_eval):
    assert initial_conditions.ndim == 2
    assert initial_conditions.shape[1] == 3

    def rossler(t, y):
        """ODE system."""
        x = y
        a = 0.15
        b = 0.2
        c = 10
        y_dot = np.zeros(3)

        y_dot[0] = -x[1] - x[2]
        y_dot[1] = x[0] + a * x[1]
        y_dot[2] = b + x[2] * (x[0] - c)
        return y_dot

    trajectories = []

    for idx, ic in enumerate(initial_conditions):
        solution = solve_ivp(
            rossler, t_span=[t_eval[0], t_eval[-1]], y0=ic, t_eval=t_eval, method='DOP853'
        )
        trajectories.append(solution["y"])

    return np.array(trajectories)
