{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ============================================================================\n",
    "# CHUNK 1: SETUP - Algorithms, Helper Functions, and Definitions\n",
    "# ============================================================================\n",
    "\n",
    "import numpy as np\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.signal import convolve2d\n",
    "from hj_prox import compute_prox_HJ\n",
    "import time\n",
    "from typing import Tuple\n",
    "\n",
    "eps = 1e-5\n",
    "\n",
    "# Set deterministic behavior\n",
    "torch.backends.cudnn.deterministic = True\n",
    "torch.backends.cudnn.benchmark = False\n",
    "\n",
    "# Plotting configuration\n",
    "plt.rcParams.update({'font.size': 16})\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# Helper Functions\n",
    "# ============================================================================\n",
    "\n",
    "def create_gaussian_kernel(size, sigma):\n",
    "    \"\"\"Create a 2D Gaussian kernel for blurring.\"\"\"\n",
    "    kernel = np.zeros((size, size), dtype=float)\n",
    "    center = size // 2\n",
    "    for i in range(size):\n",
    "        for j in range(size):\n",
    "            x, y = i - center, j - center\n",
    "            kernel[i, j] = np.exp(-(x**2 + y**2) / (2 * sigma**2))\n",
    "    return kernel / kernel.sum()\n",
    "\n",
    "\n",
    "def blur_image(img, kernel_size=9, sigma=2.0):\n",
    "    \"\"\"Apply Gaussian blur to an image.\"\"\"\n",
    "    kernel = create_gaussian_kernel(kernel_size, sigma)\n",
    "    blurred = convolve2d(img, kernel, mode='same', boundary='symm')\n",
    "    return blurred, kernel\n",
    "\n",
    "\n",
    "def compute_tv(x: torch.Tensor) -> float:\n",
    "    \"\"\"Compute Total Variation of an image using forward differences.\"\"\"\n",
    "    dx = torch.zeros_like(x)\n",
    "    dy = torch.zeros_like(x)\n",
    "    \n",
    "    # Forward differences\n",
    "    dx[:-1, :] = x[1:, :] - x[:-1, :]\n",
    "    dy[:, :-1] = x[:, 1:] - x[:, :-1]\n",
    "    \n",
    "    # TV norm\n",
    "    tv = torch.sqrt(dx**2 + dy**2 + 1e-8)\n",
    "    return tv.sum().item()\n",
    "\n",
    "\n",
    "def compute_tv_pytorch(x_batch: torch.Tensor, H: int, W: int) -> torch.Tensor:\n",
    "    \"\"\"Compute Total Variation for a batch of images.\"\"\"\n",
    "    batch_size = x_batch.shape[0]\n",
    "    x_imgs = x_batch.view(batch_size, H, W)\n",
    "    dx = torch.zeros_like(x_imgs)\n",
    "    dy = torch.zeros_like(x_imgs)\n",
    "    dx[:, :-1, :] = x_imgs[:, 1:, :] - x_imgs[:, :-1, :]\n",
    "    dy[:, :, :-1] = x_imgs[:, :, 1:] - x_imgs[:, :, :-1]\n",
    "    # Enforce zero gradient at bottom/right\n",
    "    dx[:, -1, :] = 0\n",
    "    dy[:, :, -1] = 0\n",
    "    tv = torch.sqrt(dx**2 + dy**2 + 1e-8)\n",
    "    return tv.sum(dim=(1,2))\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# Algorithm 1: PDHG (Chambolle-Pock) with Analytical Operators\n",
    "# ============================================================================\n",
    "\n",
    "def chambolle_pock_tv_deblur(\n",
    "    blurred_img: np.ndarray,\n",
    "    kernel: np.ndarray,\n",
    "    regularization: float = 0.02,\n",
    "    iterations: int = 500,\n",
    "    tau: float = None,\n",
    "    sigma: float = None,\n",
    "    theta: float = 1.0,\n",
    "    verbose: bool = True,\n",
    "    device: str = 'cpu'\n",
    ") -> Tuple[np.ndarray, list]:\n",
    "    \"\"\"\n",
    "    Deblur image using TV regularization with Chambolle-Pock algorithm.\n",
    "    \n",
    "    Solves: minimize 0.5 * ||K*x - y||^2 + λ * TV(x)\n",
    "    \n",
    "    Args:\n",
    "        blurred_img: Blurred input image\n",
    "        kernel: Blur kernel\n",
    "        regularization: TV regularization parameter (λ)\n",
    "        iterations: Number of iterations\n",
    "        tau: Primal step size\n",
    "        sigma: Dual step size\n",
    "        theta: Extrapolation parameter (typically 1.0)\n",
    "        verbose: Print progress\n",
    "        device: 'cpu' or 'cuda'\n",
    "    \n",
    "    Returns:\n",
    "        Deblurred image and list of objective values\n",
    "    \"\"\"\n",
    "    H, W = blurred_img.shape\n",
    "    y = torch.from_numpy(blurred_img).float().to(device)\n",
    "    \n",
    "    # Initialize primal variable\n",
    "    x = y.clone()\n",
    "    x_bar = y.clone()\n",
    "    \n",
    "    # Initialize dual variables\n",
    "    z = torch.zeros_like(y)  # Dual for data fidelity\n",
    "    p1 = torch.zeros_like(y)  # Dual for TV (gradient x)\n",
    "    p2 = torch.zeros_like(y)  # Dual for TV (gradient y)\n",
    "    \n",
    "    # Prepare kernel for FFT convolution\n",
    "    kernel_torch = torch.from_numpy(kernel).float().to(device)\n",
    "    kernel_padded = torch.zeros(H, W, device=device)\n",
    "    kh, kw = kernel.shape\n",
    "    kernel_padded[:kh, :kw] = kernel_torch\n",
    "    \n",
    "    # Center the kernel for FFT\n",
    "    kernel_padded = torch.roll(kernel_padded, shifts=(-kh//2, -kw//2), dims=(0, 1))\n",
    "    kernel_fft = torch.fft.fft2(kernel_padded)\n",
    "    kernel_fft_conj = torch.conj(kernel_fft)\n",
    "    \n",
    "    # Set step sizes if not provided\n",
    "    if tau is None or sigma is None:\n",
    "        L = 12.0  # Conservative estimate of operator norm\n",
    "        tau = 0.9 / L\n",
    "        sigma = 0.9 / L\n",
    "        if verbose:\n",
    "            print(f\"Using step sizes: tau={tau:.4f}, sigma={sigma:.4f}\")\n",
    "    \n",
    "    losses = []\n",
    "    \n",
    "    for it in range(iterations):\n",
    "        x_old = x.clone()\n",
    "        \n",
    "        # ---- Dual update ----\n",
    "        # Update z (data fidelity dual)\n",
    "        x_bar_fft = torch.fft.fft2(x_bar)\n",
    "        Kx_bar = torch.real(torch.fft.ifft2(x_bar_fft * kernel_fft))\n",
    "        z = (z + sigma * (Kx_bar - y)) / (1.0 + sigma)\n",
    "        \n",
    "        # Update p (TV dual) with forward differences\n",
    "        grad_x = torch.zeros_like(x_bar)\n",
    "        grad_y = torch.zeros_like(x_bar)\n",
    "        grad_x[:-1, :] = x_bar[1:, :] - x_bar[:-1, :]\n",
    "        grad_y[:, :-1] = x_bar[:, 1:] - x_bar[:, :-1]\n",
    "        \n",
    "        p1 = p1 + sigma * grad_x\n",
    "        p2 = p2 + sigma * grad_y\n",
    "        \n",
    "        # Project onto L∞ ball\n",
    "        norm = torch.sqrt(p1**2 + p2**2 + 1e-8)\n",
    "        factor = torch.clamp(regularization / norm, max=1.0)\n",
    "        p1 = p1 * factor\n",
    "        p2 = p2 * factor\n",
    "        \n",
    "        # ---- Primal update ----\n",
    "        # K^T z\n",
    "        z_fft = torch.fft.fft2(z)\n",
    "        KTz = torch.real(torch.fft.ifft2(z_fft * kernel_fft_conj))\n",
    "        \n",
    "        # Divergence of p (negative adjoint of gradient)\n",
    "        div_p = torch.zeros_like(x)\n",
    "        # Backward differences\n",
    "        div_p[1:, :] -= p1[:-1, :]\n",
    "        div_p[:-1, :] += p1[:-1, :]\n",
    "        div_p[:, 1:] -= p2[:, :-1]\n",
    "        div_p[:, :-1] += p2[:, :-1]\n",
    "        \n",
    "        # Update x\n",
    "        x = x - tau * (KTz - div_p)\n",
    "        \n",
    "        # Extrapolation\n",
    "        x_bar = x + theta * (x - x_old)\n",
    "        \n",
    "        # Monitor convergence\n",
    "        if it % 1 == 0:\n",
    "            x_fft = torch.fft.fft2(x)\n",
    "            Kx = torch.real(torch.fft.ifft2(x_fft * kernel_fft))\n",
    "            \n",
    "            data_term = 0.5 * torch.sum((Kx - y)**2).item()\n",
    "            tv_term = compute_tv(x)\n",
    "            loss = data_term + regularization * tv_term\n",
    "            losses.append(loss)\n",
    "            \n",
    "            if verbose and it % 50 == 0:\n",
    "                primal_res = torch.norm(x - x_old).item()\n",
    "                print(f\"Iter {it}: Loss={loss:.6f}, Data={data_term:.6f}, \"\n",
    "                      f\"TV={tv_term:.6f}, ||Δx||={primal_res:.6f}\")\n",
    "    \n",
    "    return x.cpu().numpy(), losses\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# Algorithm 2: PDHG with HJ-Prox\n",
    "# ============================================================================\n",
    "\n",
    "def pdhg_hjprox_deblur(\n",
    "    blurred_img: np.ndarray,\n",
    "    kernel: np.ndarray,\n",
    "    regularization: float = 0.02,\n",
    "    iterations: int = 500,\n",
    "    tau: float = None,\n",
    "    sigma: float = None,\n",
    "    theta: float = 1.0,\n",
    "    int_samples: int = 100,\n",
    "    delta: float = 1e-2,\n",
    "    verbose: bool = True,\n",
    "    device: str = 'cpu',\n",
    "    init_img: np.ndarray = None\n",
    ") -> Tuple[np.ndarray, list]:\n",
    "    \"\"\"\n",
    "    Deblur image using PDHG with HJ-Prox for TV regularization.\n",
    "    \n",
    "    Solves: minimize 0.5 * ||K*x - y||^2 + λ * TV(x)\n",
    "    \n",
    "    Uses HJ-Prox to compute the proximal operator of TV.\n",
    "    \n",
    "    Args:\n",
    "        blurred_img: Blurred input image\n",
    "        kernel: Blur kernel\n",
    "        regularization: TV regularization parameter (λ)\n",
    "        iterations: Number of iterations\n",
    "        tau: Primal step size\n",
    "        sigma: Dual step size\n",
    "        theta: Extrapolation parameter\n",
    "        int_samples: Number of samples for HJ-Prox\n",
    "        delta: Smoothing parameter for HJ-Prox\n",
    "        verbose: Print progress\n",
    "        device: 'cpu' or 'cuda'\n",
    "        init_img: Initial image (if None, uses blurred_img)\n",
    "    \n",
    "    Returns:\n",
    "        Deblurred image and list of objective values\n",
    "    \"\"\"\n",
    "    H, W = blurred_img.shape\n",
    "    y = torch.from_numpy(blurred_img).float().to(device)\n",
    "    \n",
    "    # Initialize x and x_bar with provided image or default to blurred image\n",
    "    if init_img is not None:\n",
    "        if init_img.shape != blurred_img.shape:\n",
    "            raise ValueError(f\"init_img shape {init_img.shape} must match blurred_img shape {blurred_img.shape}\")\n",
    "        x = torch.from_numpy(init_img).float().to(device)\n",
    "        x_bar = x.clone()\n",
    "    else:\n",
    "        x = y.clone()\n",
    "        x_bar = y.clone()\n",
    "    \n",
    "    z = torch.zeros_like(y)\n",
    "    \n",
    "    # Prepare kernel for convolution via FFT\n",
    "    kernel_torch = torch.from_numpy(kernel).float().to(device)\n",
    "    kernel_padded = torch.zeros(H, W, device=device)\n",
    "    kh, kw = kernel.shape\n",
    "    kernel_padded[:kh, :kw] = kernel_torch\n",
    "    # Center before FFT to ensure correct alignment\n",
    "    kernel_padded = torch.roll(kernel_padded, shifts=(-kh//2, -kw//2), dims=(0, 1))\n",
    "    kernel_fft = torch.fft.fft2(kernel_padded)\n",
    "    kernel_fft_conj = torch.conj(kernel_fft)\n",
    "    \n",
    "    # Set step sizes if not provided\n",
    "    if tau is None or sigma is None:\n",
    "        L = 3.0\n",
    "        tau = 1.0 / L\n",
    "        sigma = 1.0 / L\n",
    "        if verbose:\n",
    "            print(f\"Using step sizes: tau={tau:.4f}, sigma={sigma:.4f}\")\n",
    "    \n",
    "    losses = []\n",
    "    \n",
    "    # Define TV function for HJ-Prox (batch-aware)\n",
    "    def tv_batch(xb):\n",
    "        return regularization * compute_tv_pytorch(xb, H, W)\n",
    "    \n",
    "    for it in range(iterations):\n",
    "        x_old = x.clone()\n",
    "        \n",
    "        # Dual update: K x_bar\n",
    "        x_bar_fft = torch.fft.fft2(x_bar)\n",
    "        Kx = torch.real(torch.fft.ifft2(x_bar_fft * kernel_fft))\n",
    "        z_update = z + sigma * Kx\n",
    "        # Closed-form prox for data fidelity (quadratic)\n",
    "        z = (z_update - sigma * y) / (1.0 + sigma)\n",
    "        \n",
    "        # Primal update\n",
    "        z_fft = torch.fft.fft2(z)\n",
    "        KTz = torch.real(torch.fft.ifft2(z_fft * kernel_fft_conj))\n",
    "        x_grad = x - tau * KTz\n",
    "        x_flat = x_grad.view(-1, 1)\n",
    "        \n",
    "        # Compute delta with annealing schedule\n",
    "        delta_k = 2300000 / (it + 1)**(2 + eps)\n",
    "        \n",
    "        # Proximal step using HJ-Prox for TV\n",
    "        x_prox, _, _ = compute_prox_HJ(\n",
    "            x_flat, tau, f=tv_batch, delta=delta_k,\n",
    "            int_samples=int_samples, alpha=1.0\n",
    "        )\n",
    "        \n",
    "        x = x_prox.view(H, W)\n",
    "        \n",
    "        # Over-relaxation\n",
    "        x_bar = x + theta * (x - x_old)\n",
    "        \n",
    "        # Monitor convergence\n",
    "        if it % 1 == 0:\n",
    "            x_fft = torch.fft.fft2(x)\n",
    "            Kx_curr = torch.real(torch.fft.ifft2(x_fft * kernel_fft))\n",
    "            data_term = 0.5 * torch.sum((Kx_curr - y)**2).item()\n",
    "            tv_term = compute_tv_pytorch(x.unsqueeze(0).view(1, -1), H, W).item()\n",
    "            loss = data_term + regularization * tv_term\n",
    "            losses.append(loss)\n",
    "            \n",
    "            if verbose:\n",
    "                diff = torch.norm(x - x_old).item()\n",
    "                print(f\"Iter {it}: Loss={loss:.6f}, Δx={diff:.6f}\")\n",
    "    \n",
    "    return x.cpu().numpy(), losses\n",
    "\n",
    "\n",
    "print(\"✓ All algorithms and helper functions loaded successfully\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ============================================================================\n",
    "# CHUNK 2: DATA GENERATION\n",
    "# ============================================================================\n",
    "\n",
    "print(\"\\n\" + \"=\"*60)\n",
    "print(\"Creating synthetic test image and generating data...\")\n",
    "print(\"=\"*60)\n",
    "\n",
    "# Image dimensions\n",
    "H, W = 64, 64\n",
    "\n",
    "# Create test image with rectangles\n",
    "original_img = np.zeros((H, W))\n",
    "original_img[10:20, 10:30] = 1.0\n",
    "original_img[30:45, 20:40] = 0.7\n",
    "original_img[15:35, 35:55] = 0.5\n",
    "\n",
    "# Parameters\n",
    "kernel_size = 5\n",
    "blur_sigma = 0.2\n",
    "noise_level = 0.15\n",
    "regularization = 0.125\n",
    "iterations = 20000\n",
    "\n",
    "# Create blurred image\n",
    "print(f\"Applying Gaussian blur (kernel size={kernel_size}, sigma={blur_sigma})...\")\n",
    "blurred_img, blur_kernel = blur_image(original_img, kernel_size, blur_sigma)\n",
    "\n",
    "# Add noise\n",
    "if noise_level > 0:\n",
    "    np.random.seed(42)\n",
    "    blurred_img += noise_level * np.random.randn(*blurred_img.shape)\n",
    "    blurred_img = np.clip(blurred_img, 0, 1)\n",
    "    print(f\"Added Gaussian noise (σ={noise_level})\")\n",
    "\n",
    "print(f\"✓ Data generated: Image size {H}x{W}\")\n",
    "print(f\"✓ Regularization parameter λ = {regularization}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# ============================================================================\n",
    "# CHUNK 3: RUN ALGORITHM 1 - Analytical PDHG (Chambolle-Pock)\n",
    "# ============================================================================\n",
    "\n",
    "print(\"\\n\" + \"=\"*60)\n",
    "print(\"Running Algorithm 1: PDHG with Analytical Operators...\")\n",
    "print(\"=\"*60)\n",
    "\n",
    "start_time = time.time()\n",
    "\n",
    "deblurred_img_Analytical, analytical_losses = chambolle_pock_tv_deblur(\n",
    "    blurred_img,\n",
    "    blur_kernel,\n",
    "    regularization=regularization,\n",
    "    iterations=20000,\n",
    "    tau=0.025*0.0088,\n",
    "    sigma=0.1,\n",
    "    theta=1.0,\n",
    "    verbose=True,\n",
    "    device='cpu'\n",
    ")\n",
    "\n",
    "elapsed_time = time.time() - start_time\n",
    "\n",
    "# Clip for visualization and metrics\n",
    "deblurred_clipped = np.clip(deblurred_img_Analytical, 0, 1)\n",
    "\n",
    "# Compute metrics\n",
    "mse_blurred = np.mean((original_img - blurred_img)**2)\n",
    "mse_deblurred = np.mean((original_img - deblurred_clipped)**2)\n",
    "\n",
    "psnr_blurred = 20 * np.log10(1.0 / np.sqrt(mse_blurred)) if mse_blurred > 0 else float('inf')\n",
    "psnr_deblurred = 20 * np.log10(1.0 / np.sqrt(mse_deblurred)) if mse_deblurred > 0 else float('inf')\n",
    "\n",
    "print(f\"\\n✓ Analytical PDHG completed\")\n",
    "print(f\"  - Runtime: {elapsed_time:.2f} seconds\")\n",
    "print(f\"  - Final objective: {analytical_losses[-1]:.6f}\")\n",
    "print(f\"  - PSNR (blurred): {psnr_blurred:.2f} dB\")\n",
    "print(f\"  - PSNR (deblurred): {psnr_deblurred:.2f} dB\")\n",
    "print(f\"  - PSNR improvement: {psnr_deblurred - psnr_blurred:.2f} dB\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ============================================================================\n",
    "# CHUNK 4: RUN ALGORITHM 2 - PDHG with HJ-Prox\n",
    "# ============================================================================\n",
    "\n",
    "print(\"\\n\" + \"=\"*60)\n",
    "print(\"Running Algorithm 2: PDHG with HJ-Prox...\")\n",
    "print(\"=\"*60)\n",
    "\n",
    "start_time = time.time()\n",
    "\n",
    "deblurred_img_HJ, hj_losses = pdhg_hjprox_deblur(\n",
    "    blurred_img,\n",
    "    blur_kernel,\n",
    "    regularization=regularization,\n",
    "    iterations=20000,\n",
    "    tau=0.025*0.0088,\n",
    "    sigma=0.1,\n",
    "    theta=1.0,\n",
    "    int_samples=1000,\n",
    "    delta=0.0001,\n",
    "    verbose=True,\n",
    "    device='cpu',\n",
    "    init_img=blurred_img\n",
    ")\n",
    "\n",
    "elapsed_time = time.time() - start_time\n",
    "\n",
    "print(f\"\\n✓ PDHG-HJ completed\")\n",
    "print(f\"  - Runtime: {elapsed_time:.2f} seconds\")\n",
    "print(f\"  - Final objective: {hj_losses[-1]:.6f}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ============================================================================\n",
    "# CHUNK 5: GENERATE FIGURES\n",
    "# ============================================================================\n",
    "\n",
    "print(\"\\n\" + \"=\"*60)\n",
    "print(\"Generating figures...\")\n",
    "print(\"=\"*60)\n",
    "\n",
    "# Global font settings\n",
    "plt.rcParams.update({'font.size': 16})\n",
    "\n",
    "# --- Figure 1: Original Image ---\n",
    "plt.figure(figsize=(11, 10))\n",
    "plt.imshow(original_img, cmap='gray')\n",
    "plt.title(\"Original Image\", fontsize=40, pad=8)\n",
    "plt.axis('off')\n",
    "plt.tight_layout()\n",
    "plt.savefig('pdhg_original.pdf', format='pdf', dpi=600, bbox_inches='tight')\n",
    "plt.show()\n",
    "plt.close()\n",
    "\n",
    "# --- Figure 2: PDHG-HJ Deblurred Image ---\n",
    "plt.figure(figsize=(11, 10))\n",
    "plt.imshow(deblurred_img_HJ, cmap='gray')\n",
    "plt.title(\"PDHG-HJ\", fontsize=40, pad=8)\n",
    "plt.axis('off')\n",
    "plt.tight_layout()\n",
    "plt.savefig('pdhg_hj_deblurred.pdf', format='pdf', dpi=600, bbox_inches='tight')\n",
    "plt.show()\n",
    "plt.close()\n",
    "\n",
    "# --- Figure 3: PDHG Analytical Deblurred Image ---\n",
    "plt.figure(figsize=(11, 10))\n",
    "plt.imshow(deblurred_clipped, cmap='gray')\n",
    "plt.title(\"PDHG\", fontsize=40, pad=8)\n",
    "plt.axis('off')\n",
    "plt.tight_layout()\n",
    "plt.savefig('pdhg_analytical_deblurred.pdf', format='pdf', dpi=600, bbox_inches='tight')\n",
    "plt.show()\n",
    "plt.close()\n",
    "\n",
    "# --- Figure 4: Objective Function Convergence ---\n",
    "plt.rcParams.update({'font.size': 20})\n",
    "plt.figure(figsize=(11, 10))\n",
    "plt.semilogy(hj_losses, '-', linewidth=3, label=f'PDHG-HJ: {hj_losses[-1]:.3f}')\n",
    "plt.semilogy(analytical_losses, '--', linewidth=3, label=f'PDHG: {analytical_losses[-1]:.3f}')\n",
    "plt.title(\"PDHG Convergence\", fontsize=40)\n",
    "plt.ylabel(\"Objective (log scale)\", fontsize=40)\n",
    "plt.xlabel(\"Iteration\", fontsize=40)\n",
    "plt.legend(fontsize=40, loc='upper left')\n",
    "plt.grid(True, which='both', alpha=0.3)\n",
    "plt.gca().tick_params(axis='both', which='major', labelsize=40)\n",
    "plt.tight_layout(pad=2.0)\n",
    "plt.savefig('pdhg_convergence.pdf', format='pdf', dpi=600, bbox_inches='tight')\n",
    "plt.show()\n",
    "plt.close()\n",
    "\n",
    "print(\"✓ All figures saved: pdhg_original.pdf, pdhg_hj_deblurred.pdf, pdhg_analytical_deblurred.pdf, pdhg_convergence.pdf\")\n",
    "print(\"\\n\" + \"=\"*60)\n",
    "print(\"✓ ALL EXPERIMENTS COMPLETED SUCCESSFULLY\")\n",
    "print(\"=\"*60)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.10"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
