# Plug-and-Play Image Restoration

This repository contains the implementation for image deblurring using Plug-and-Play (PnP) algorithms with our proposed denoiser.

## Requirements

- Python 3.8+
- PyTorch 1.10+
- NumPy
- SciPy
- scikit-image
- matplotlib
- PIL (Pillow)
- OpenCV (cv2)
- hdf5storage
- einops
- PyYAML

Install dependencies:
```bash
pip install torch numpy scipy scikit-image matplotlib pillow opencv-python hdf5storage einops pyyaml
```

## Project Structure

```
├── classes/
│   ├── PnP_class.py        # Main PnP restoration class
│   ├── blur_utils.py       # Blur operators and utilities
│   ├── utils_image.py      # Image utilities
│   └── utils_restoration.py # Restoration utilities
├── images/
│   ├── CBSD10/             # Test images
│   └── kernels/            # Blur kernels
├── ours_model/
│   ├── model_loader.py     # Model loader
│   ├── ours_models.py      # Model architecture
│   └── configs-50g-new/    # Model config and checkpoint
├── playgrounds/
│   └── playground.ipynb    # Example notebook
└── Experiments/            # Output directory
```

## Quick Start

1. Navigate to the `playgrounds` directory:
```bash
cd playgrounds
```

2. Open and run `playground.ipynb` in Jupyter:
```bash
jupyter notebook playground.ipynb
```

## Usage

The main class `img_PnP` handles image restoration:

```python
from classes.PnP_class import *
from ours_model.model_loader import *

# Load image and setup forward model
my_image = img_PnP(
    image_path="../images/CBSD10/0001.png",
    forward_model_name='deblurring',
    forward_model_args={'scale_factor': 1, 'kernel_id': 7, 'device': 'cuda:0'},
    noise_level=0.03,
    color_mode='RGB',
    crop=True
)

# Load denoiser
denoiser = load_model(
    config_folder="../ours_model/configs-50g-new",
    device='cuda:0'
).to('cuda:0')

# Run PnP reconstruction
denoiser_args = {'guide_image': my_image.observed, 'h': 8.0/255.0}
algo_params = {'name': 'HQS', 'step_size': 6.5, 'transpose': False, 'clip': False}

my_image.PnP(
    denoiser=OURS,
    denoiser_args=denoiser_args,
    denoiser_object=denoiser,
    num_iterations=100,
    algo_params=algo_params
)

# Save results
my_image.get_images(plot=True, save=True)
```

## Parameters

### Forward Model Arguments
- `kernel_id`: Blur kernel index (0-7: Levin09 kernels, 8: Gaussian, 9: Box)
- `scale_factor`: Downsampling factor (1 for deblurring)
- `device`: CUDA device (e.g., 'cuda:0')

### Algorithm Parameters
- `name`: Algorithm type ('HQS', 'FBS', 'DRS', 'RED')
- `step_size`: Step size parameter
- `transpose`: Set to `False` for our denoiser
- `clip`: Whether to clip output to [0, 1]

### Denoiser Arguments
- `guide_image`: Guide image for filtering
- `h`: Filtering strength parameter

## Supported Algorithms

- **HQS**: Half-Quadratic Splitting
- **FBS**: Forward-Backward Splitting
- **DRS**: Douglas-Rachford Splitting
- **RED**: Regularization by Denoising
