Skip to main content

Topology-Preserving Image Smoothing using ASFT algorithm

Project description

Topology-Preserving Smoothing - Python Implementation

A Python implementation of the Alternate Sequential Filter controlled by Topology (ASFT) algorithm for smoothing binary images while preserving their topology (number of connected components and holes).

Example: Einstein Image

Original Smoothed (ASFT-MED) Smoothed (ASFT)

Smoothed with: -s 1 -r 3 -c 4 --medial for ASFT-MED and with the same parameters for the pure ASFT.

The algorithm smooths jagged boundaries while preserving topology - all connected components and holes from the original image are maintained.

More Examples

Example 1: Pixelized stains #1

Input Output

Example 2: Pixelized stains #2

Input Output

Example 3: QR-code (Connectivity Comparison)

Input Output (C8) Output (C4)

Example 4: Flying heron

Input Output

Example 5: Trees

Input Output

Reference

Based on the paper:

M. Couprie and G. Bertrand: "Topology preserving alternating sequential filter for smoothing 2D and 3D objects", Journal of Electronic Imaging, Vol. 13, No. 4, pp. 720-730, 2004.

See also the dedicated web-page:

Topological Smoothing by M. Couprie, G. Bertrand

Installation

From PyPI (when published)

# Recommended installation (includes numba for 10-100x speedup)
pip install toposmooth[fast]

# If you have issues installing numba (e.g., unsupported platform):
pip install toposmooth

From Source

# Clone the repository
git clone https://github.com/vyastrebov/toposmooth.git
cd toposmooth

# Recommended: install with numba
pip install -e ".[fast]"

# Without numba (if installation fails):
pip install -e .

Required packages

  • numpy
  • scipy
  • Pillow
  • numba (included with [fast], provides 10-100x speedup)

Supported Image Formats

The algorithm supports any bitmap format readable by Pillow:

  • PNG (.png) - recommended for lossless output
  • JPEG (.jpg, .jpeg) - input only (lossy compression not ideal for binary output)
  • PGM (.pgm) - portable graymap, matches C implementation
  • BMP (.bmp)
  • TIFF (.tiff, .tif)
  • And many others...

The output format is automatically determined by the file extension.

Usage

Command Line

After installation, you can use the toposmooth command or run as a Python module:

# Using the installed command
toposmooth input.png output.png

# Or using Python module syntax
python -m toposmooth input.png output.png

# Match C asftmed command: ./asftmed input.pgm 4 3 output.pgm
toposmooth input.pgm output.pgm -s 1 -r 3 -c 4

# With all options
toposmooth input.jpg output.png \
    -s 1           # Scale factor (default: 4, use 1 to match input dimensions)
    -r 5           # Maximum smoothing radius (default: 5)
    -c 4           # Connectivity for white value (1): 4 (neighbor) or 8 (next-neighbour)
    -t 128         # Binarization threshold (default: mean intensity, if the input is not binary)
    --medial       # Use medial axis constraints - strictly preserves the form (default, recommended)
    --no-medial    # Use plain ASFT without medial axis constraints (removes small features)
    --save-binary binary.png  # Save the binarized input image

# Show version
toposmooth --version

Python API

from toposmooth import topology_preserving_smooth, load_image, asftmed
from PIL import Image
import numpy as np

# Load and process an image
img = load_image('input.png')
smoothed, binary = topology_preserving_smooth(
    img,
    scale=1,           # Scale factor (1 = no scaling, matches C)
    smooth_radius=5,   # Maximum ASFT radius
    connex=4,          # Object connectivity (4 or 8)
    threshold=None,    # Binarization threshold (None = mean)
    use_medial=True    # Use medial axis constraints (recommended)
)

# Save result in any format
Image.fromarray(smoothed).save('output.png')   # PNG
Image.fromarray(smoothed).save('output.jpg')   # JPEG
Image.fromarray(smoothed).save('output.pgm')   # PGM

Parameters

  • scale: Integer scaling factor. The image is upscaled before smoothing to allow sub-pixel smoothing with respect to original pixel size. Use scale=1 to match the original image dimensions.

  • smooth_radius (rmax): Maximum radius of disk structuring elements. The algorithm applies opening/closing operations with disks of radius 1, 2, ..., rmax. Larger values produce smoother results but take longer (linear complexity with rmax), but the default value rmax=5 produces already very good results.

  • connex: Object connectivity (4 or 8). For 8-connectivity, diagonal neighbors (white pixels, value=1) are considered connected.

  • threshold: Binarization threshold. Pixels above this value become foreground (255), others become background (0). If None, uses the mean intensity.

  • use_medial / --medial: When enabled (default), uses medial axis constraints to preserve thin features, recommended.

Algorithms

ASFT-MED (default, --medial)

Uses medial axes of both foreground and background as constraints. This prevents:

  • Thin features from being eroded away
  • Small gaps from being filled in

Plain ASFT (--no-medial)

Standard ASFT without constraints. May lose very thin features.

How It Works

  1. Binarization: The input image is converted to binary using the threshold.

  2. Scaling (optional): The binary image is upscaled by the scale factor.

  3. Medial Axis Computation (ASFT-MED only):

    • Compute medial axis of foreground (skeleton)
    • Compute medial axis of background (inverse skeleton)
  4. ASFT: For each radius r from 1 to rmax:

    • Apply homotopic pseudo-closing (fills small holes while preserving topology)
    • Apply homotopic pseudo-opening (removes small protrusions while preserving topology)
    • With medial axis constraints, these operations are bounded to preserve thin structures
  5. The result is a smoothed binary image with the same topology as the input.

Performance

  • With numba (pip install toposmooth[fast]): ~1-2 seconds for a 520×372 image with scale=1, radius=3
  • Without numba: Much slower (minutes instead of seconds) - only use if numba won't install
  • The algorithm scales roughly as O(scale² × rmax × width × height)
  • First run with numba may take a few seconds for JIT compilation

Validation

The Python implementation produces pixel-perfect results matching the C implementation for Einstein's binary portrait.

# C version
./asftmed test/einstein.pgm 4 3 c_output.pgm

# Python version
toposmooth test/einstein.pgm py_output.pgm -s 1 -r 3 -c 4

Test Shapes

Run python test_shapes.py to generate comparison panels for various test shapes:

  1. Rectangle - Grid-aligned rectangle
  2. Square - Grid-aligned square
  3. L-Shape - Grid-aligned L shape
  4. Squares Edge Touch - Two squares sharing a full edge (1 component for both connectivities)
  5. Squares Single Point Touch - Two squares touching at exactly one 4-neighbor pixel (1 component for both)
  6. Squares Diagonal Touch - Two squares touching only at diagonal (2 components for 4-conn, 1 for 8-conn)
  7. Rotated Square 30° - Square rotated by 30 degrees
  8. Rotated Square 45° - Square rotated by 45 degrees

Each test shows the original shape and results for:

  • Connectivity 4 and 8
  • ASFT and ASFT-MED algorithms
  • Scale factors 1 and 4

Results are saved to test_results/ folder, including all_tests_overview.png with all shapes combined.

Sample Test Result

Test Overview

Information

Authors: Claude Opus 4.5 in Cursor environment (mainly), Vladislav A. Yastrebov (marginally) Validator: Vladislav A. Yastrebov (CNRS, Mines Paris) License: GPL-2.0 Date: Nov-Dec 2025

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

toposmooth-1.0.0.tar.gz (21.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

toposmooth-1.0.0-py3-none-any.whl (19.1 kB view details)

Uploaded Python 3

File details

Details for the file toposmooth-1.0.0.tar.gz.

File metadata

  • Download URL: toposmooth-1.0.0.tar.gz
  • Upload date:
  • Size: 21.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.2

File hashes

Hashes for toposmooth-1.0.0.tar.gz
Algorithm Hash digest
SHA256 05da320a5b22a663e2ced03b84551e6a7452778ac9f6c4fe8714b7dfd68726d7
MD5 21719128bec2c83d093a8409ac1fe57c
BLAKE2b-256 cbc2490670cbe877a92b7a187e8fc77747d90c77c4d95271981db34557ee1727

See more details on using hashes here.

File details

Details for the file toposmooth-1.0.0-py3-none-any.whl.

File metadata

  • Download URL: toposmooth-1.0.0-py3-none-any.whl
  • Upload date:
  • Size: 19.1 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.2

File hashes

Hashes for toposmooth-1.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 f9abd14a93a71959dcda269fdb43d713ac0a118241d6061849c8a63db7084f15
MD5 0328cedd28b7e7f4fd61a710a4df1c6d
BLAKE2b-256 82e3ea71b81ac0d4d6e1d01f8a6d7d7d4c98cbbf20b8c3f97064979f1d3075a0

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page