import numpy as np
import matplotlib.pyplot as plt

def r1(x, y):
    return (x+y)**2

def r2(x, y):
    eps = 1e-9
    return np.log((1.0 + x + eps)/(1.0 + y + eps))


def compute_pareto_front(num_points=101):
    grid = np.linspace(0, 1, num_points)
    all_points = []

    for x in grid:
        for y in grid:
            val1 = r1(x, y)
            val2 = r2(x, y)
            all_points.append((val1, val2))

    pts = np.array(all_points)  

    is_dominated = np.zeros(len(pts), dtype=bool)
    for i in range(len(pts)):
        if is_dominated[i]:
            continue
        for j in range(len(pts)):
            if i == j:
                continue
            if (pts[j,0] >= pts[i,0]) and (pts[j,1] >= pts[i,1]) and \
               ((pts[j,0] > pts[i,0]) or (pts[j,1] > pts[i,1])):
                is_dominated[i] = True
                break

    pareto_pts = pts[~is_dominated]
    return pareto_pts


def solve_constrained(b, num_points=101):
    xs = np.linspace(0, 1, num_points)
    ys = np.linspace(0, 1, num_points)

    solutions = []

    for x in xs:
        best_r1 = -1e9
        best_r2 = None
        feasible_found = False
        for y in ys:
            val2 = r2(x, y)
            if val2 >= b:
                feasible_found = True
                val1 = r1(x, y)
                if val1 > best_r1:
                    best_r1 = val1
                    best_r2 = val2

        if feasible_found:
            solutions.append((best_r1, best_r2))

    return np.array(solutions)


if __name__ == "__main__":

    pareto_pts = compute_pareto_front(num_points=101)

    plt.figure(figsize=(7,6))

    plt.scatter(pareto_pts[:,0], pareto_pts[:,1],
                label="Pareto Frontier", s=20, alpha=0.7, color="goldenrod")
    

    b_values = [0.1]

    for b_value in b_values:
        constrained_solutions = solve_constrained(b_value, num_points=101)
        
        if len(constrained_solutions)>0:
            plt.scatter(constrained_solutions[:,0], constrained_solutions[:,1],
                        label=f"Constrained solutions (b={b_value})", s=30)
        else:
            print(f"No feasible points for b={b_value} (r2 >= {b_value}).")

    plt.xlabel("r1")
    plt.ylabel("r2")
    plt.title("Pareto Front vs. Constrained Solutions")
    plt.legend()
    plt.tight_layout()
    plt.show()
