Skip to main content

Tools to download, read, process, and plot e-CALLISTO FITS dynamic spectra.

Project description

ecallistolib

Python 3.10+ License: MIT

A Python library to download, read, process, and plot e-CALLISTO FITS dynamic spectra.

e-CALLISTO (Compact Astronomical Low-frequency Low-cost Instrument for Spectroscopy and Transportable Observatory) is an international network of solar radio spectrometers that monitor solar radio emissions in the frequency range of approximately 45–870 MHz.


Table of Contents


Features

  • 📥 Download – List and download FITS files directly from the e-CALLISTO data archive
  • 📖 Read – Parse e-CALLISTO FITS files (.fit, .fit.gz) into structured Python objects
  • 🔧 Process – Apply noise reduction techniques (mean subtraction, clipping, scaling)
  • ✂️ Crop – Extract frequency and time ranges from spectra
  • 🔗 Combine – Merge multiple spectra along the time or frequency axis
  • 📊 Plot – Generate publication-ready dynamic spectrum visualizations
  • ⚠️ Error Handling – Custom exceptions for robust error management

Installation

From PyPI (Stable)

pip install ecallistolib

Optional Dependencies

Install optional features as needed:

pip install ecallistolib"[download,plot]"

From Source (Development)

git clone https://github.com/saandev/ecallistolib.git
cd ecallistolib
pip install -e .

Optional Dependencies

Install optional features as needed:

# For downloading data from the e-CALLISTO archive
pip install -e ".[download]"

# For plotting
pip install -e ".[plot]"

# Install all optional dependencies
pip install -e ".[download,plot]"

Quick Start

import ecallistolib as ecl

# Read a FITS file
spectrum = ecl.read_fits("ALASKA_20230101_120000_01.fit.gz")

# Apply noise reduction
cleaned = ecl.noise_reduce_mean_clip(spectrum)

# Plot the result
fig, ax, im = ecl.plot_dynamic_spectrum(cleaned, title="Solar Radio Burst")

Usage Guide

Reading FITS Files

The library can read standard e-CALLISTO FITS files:

import ecallistolib as ecl

# Read a single FITS file
spectrum = ecl.read_fits("path/to/STATION_YYYYMMDD_HHMMSS_NN.fit.gz")

# Access the data
print(f"Data shape: {spectrum.shape}")          # (n_freq, n_time)
print(f"Frequencies: {spectrum.freqs_mhz}")     # Frequency axis in MHz
print(f"Time samples: {spectrum.time_s}")       # Time axis in seconds
print(f"Source file: {spectrum.source}")        # Original file path
print(f"Metadata: {spectrum.meta}")             # Station, date, etc.

Parsing Filenames

Extract metadata from e-CALLISTO filenames:

parts = ecl.parse_callisto_filename("ALASKA_20230615_143000_01.fit.gz")

print(parts.station)        # "ALASKA"
print(parts.date_yyyymmdd)  # "20230615"
print(parts.time_hhmmss)    # "143000"
print(parts.focus)          # "01"

Downloading Data

Download FITS files directly from the e-CALLISTO archive:

from datetime import date
import ecallistolib as ecl

# List available files for a specific day, hour, and station
remote_files = ecl.list_remote_fits(
    day=date(2023, 6, 15),
    hour=14,                    # UTC hour (0-23)
    station_substring="alaska"  # Case-insensitive station filter
)

print(f"Found {len(remote_files)} files:")
for rf in remote_files:
    print(f"  - {rf.name}: {rf.url}")

# Download the files
saved_paths = ecl.download_files(remote_files, out_dir="./data")

for path in saved_paths:
    print(f"Downloaded: {path}")

Processing Data

Noise Reduction

Apply mean-subtraction and clipping to enhance signal visibility:

import ecallistolib as ecl

spectrum = ecl.read_fits("my_spectrum.fit.gz")

# Apply noise reduction with default parameters
cleaned = ecl.noise_reduce_mean_clip(spectrum)

# Or customize the parameters
cleaned = ecl.noise_reduce_mean_clip(
    spectrum,
    clip_low=-5.0,              # Lower clipping threshold
    clip_high=20.0,             # Upper clipping threshold
    scale=2500.0 / 255.0 / 25.4 # Scaling factor (None to disable)
)

# Processing metadata is recorded
print(cleaned.meta["noise_reduction"])
# {'method': 'mean_subtract_clip', 'clip_low': -5.0, 'clip_high': 20.0, 'scale': 3.88...}

Algorithm Details:

  1. Subtract the mean intensity over time for each frequency channel (removes baseline)
  2. Clip values to the specified range
  3. Apply optional scaling factor

Background Subtraction Only

If you want to visualize the result before clipping is applied:

import ecallistolib as ecl

spectrum = ecl.read_fits("my_spectrum.fit.gz")

# Apply only background subtraction (no clipping)
bg_subtracted = ecl.background_subtract(spectrum)

# This is equivalent to the first step of noise_reduce_mean_clip
# Each frequency channel now has zero mean

Cropping & Slicing

Extract specific frequency or time ranges from a spectrum:

Crop by Physical Values

import ecallistolib as ecl

spectrum = ecl.read_fits("my_spectrum.fit.gz")

# Crop to specific frequency range (in MHz)
cropped = ecl.crop_frequency(spectrum, freq_min=100, freq_max=300)

# Crop to specific time range (in seconds)
cropped = ecl.crop_time(spectrum, time_min=10, time_max=60)

# Crop both axes at once
cropped = ecl.crop(spectrum, freq_range=(100, 300), time_range=(10, 60))

Slice by Array Index

# Get first 100 frequency channels
sliced = ecl.slice_by_index(spectrum, freq_slice=slice(0, 100))

# Get every other time sample (downsampling)
sliced = ecl.slice_by_index(spectrum, time_slice=slice(None, None, 2))

# Combine slices
sliced = ecl.slice_by_index(spectrum, freq_slice=slice(50, 150), time_slice=slice(0, 500))

Cropping Preserves Metadata

cropped = ecl.crop(spectrum, freq_range=(100, 200))

# Check what was cropped
print(cropped.meta["cropped"])
# {'frequency': {'min': 100, 'max': 200}}

Combining Spectra

Combine Along Frequency (Vertical Stacking)

Combine two spectra from the same observation but different frequency bands (e.g., focus 01 and 02):

import ecallistolib as ecl

# Check if files can be combined
if ecl.can_combine_frequency("file_01.fit.gz", "file_02.fit.gz"):
    combined = ecl.combine_frequency("file_01.fit.gz", "file_02.fit.gz")
    print(f"Combined shape: {combined.shape}")

Requirements for frequency combination:

  • Same station, date, and time
  • Different focus numbers (01 vs 02)
  • Matching time axes

Combine Along Time (Horizontal Concatenation)

Concatenate multiple spectra recorded consecutively:

import ecallistolib as ecl

files = [
    "ALASKA_20230615_140000_01.fit.gz",
    "ALASKA_20230615_141500_01.fit.gz",
    "ALASKA_20230615_143000_01.fit.gz",
]

# Check compatibility
if ecl.can_combine_time(files):
    combined = ecl.combine_time(files)
    print(f"Combined shape: {combined.shape}")
    print(f"Total duration: {combined.time_s[-1] - combined.time_s[0]:.1f} seconds")

Requirements for time combination:

  • Same station, date, and focus
  • Matching frequency axes

Plotting

Create dynamic spectrum visualizations with full customization:

import ecallistolib as ecl
import matplotlib.pyplot as plt

spectrum = ecl.read_fits("my_spectrum.fit.gz")
cleaned = ecl.noise_reduce_mean_clip(spectrum)

# Basic plot
fig, ax, im = ecl.plot_dynamic_spectrum(cleaned, title="Solar Radio Observation")
plt.show()

# Customized plot with clipping values, colormap, and figure size
fig, ax, im = ecl.plot_dynamic_spectrum(
    cleaned,
    title="Type III Solar Burst",
    cmap="magma",            # Matplotlib colormap
    vmin=-5,                  # Colormap lower bound
    vmax=20,                  # Colormap upper bound
    figsize=(12, 6),          # Figure size in inches
    interpolation="bilinear"  # Any matplotlib imshow kwarg
)
plt.savefig("spectrum.png", dpi=150, bbox_inches="tight")

Plotting Raw Data

import ecallistolib as ecl

spectrum = ecl.read_fits("my_spectrum.fit.gz")

# Plot raw spectrum without any processing
fig, ax, im = ecl.plot_raw_spectrum(
    spectrum,
    title="Raw Spectrum",
    cmap="viridis",
    figsize=(10, 5)
)

Plotting Background Subtracted (Before Clipping)

import ecallistolib as ecl

spectrum = ecl.read_fits("my_spectrum.fit.gz")

# Plot after background subtraction but before clipping
fig, ax, im = ecl.plot_background_subtracted(
    spectrum,
    vmin=-10,
    vmax=30,
    cmap="RdBu_r"  # Diverging colormap for +/- values
)

Time Axis Formats

Display time in seconds or Universal Time (UT):

import ecallistolib as ecl

spectrum = ecl.read_fits("my_spectrum.fit.gz")

# Default: time in seconds
ecl.plot_dynamic_spectrum(spectrum, time_format="seconds")

# Time in UT format (HH:MM:SS)
ecl.plot_dynamic_spectrum(spectrum, time_format="ut")

Time Axis Converter

Convert between elapsed seconds and UT time programmatically:

import ecallistolib as ecl

spectrum = ecl.read_fits("my_spectrum.fit.gz")

# Create converter from spectrum metadata
converter = ecl.TimeAxisConverter.from_dynamic_spectrum(spectrum)

# Convert seconds to UT
print(converter.seconds_to_ut(100))    # "12:01:40"
print(converter.seconds_to_ut(3661))   # "13:01:01"

# Convert UT to seconds
print(converter.ut_to_seconds("12:01:40"))  # 100.0
print(converter.ut_to_seconds("13:00:00"))  # 3600.0

Using a Custom Axes

import matplotlib.pyplot as plt
import ecallistolib as ecl

fig, axes = plt.subplots(1, 2, figsize=(14, 5))

spectrum1 = ecl.read_fits("file1.fit.gz")
spectrum2 = ecl.read_fits("file2.fit.gz")

ecl.plot_raw_spectrum(spectrum1, ax=axes[0], title="Raw")
ecl.plot_dynamic_spectrum(
    ecl.noise_reduce_mean_clip(spectrum2), 
    ax=axes[1], 
    title="Noise Reduced",
    vmin=-5, vmax=20
)

plt.tight_layout()
plt.show()

API Reference

DynamicSpectrum

The core data structure representing an e-CALLISTO dynamic spectrum.

@dataclass(frozen=True)
class DynamicSpectrum:
    data: np.ndarray           # Intensity data, shape (n_freq, n_time)
    freqs_mhz: np.ndarray      # Frequency axis in MHz, shape (n_freq,)
    time_s: np.ndarray         # Time axis in seconds, shape (n_time,)
    source: Optional[Path]     # Original file path
    meta: Mapping[str, Any]    # Metadata dictionary

Properties

Property Type Description
shape tuple[int, int] Returns (n_freq, n_time)

Methods

Method Description
copy_with(**changes) Returns a new DynamicSpectrum with specified fields replaced

I/O Functions

read_fits(path: str | Path) -> DynamicSpectrum

Read an e-CALLISTO FITS file.

Parameter Type Description
path str | Path Path to the FITS file (.fit or .fit.gz)

Returns: DynamicSpectrum object with data, frequencies, time, and metadata.


parse_callisto_filename(path: str | Path) -> CallistoFileParts

Parse an e-CALLISTO filename.

Returns: CallistoFileParts with attributes:

  • station – Station name
  • date_yyyymmdd – Date string
  • time_hhmmss – Time string
  • focus – Focus/channel number

Download Functions

list_remote_fits(day, hour, station_substring, base_url=..., timeout_s=10.0) -> List[RemoteFITS]

List available FITS files from the e-CALLISTO archive.

Parameter Type Description
day date Target date
hour int UTC hour (0–23)
station_substring str Case-insensitive station filter
base_url str Archive base URL (optional)
timeout_s float Request timeout in seconds

Returns: List of RemoteFITS objects with name and url attributes.


download_files(items, out_dir, timeout_s=30.0) -> List[Path]

Download FITS files to a local directory.

Parameter Type Description
items Iterable[RemoteFITS] Files to download
out_dir str | Path Output directory
timeout_s float Request timeout per file

Returns: List of saved file paths.


Processing Functions

noise_reduce_mean_clip(ds, clip_low=-5.0, clip_high=20.0, scale=...) -> DynamicSpectrum

Apply noise reduction via mean subtraction and clipping.

Parameter Type Default Description
ds DynamicSpectrum Input spectrum
clip_low float -5.0 Lower clipping threshold
clip_high float 20.0 Upper clipping threshold
scale float | None ~3.88 Scaling factor (None to disable)

Returns: New DynamicSpectrum with processed data and updated metadata.


background_subtract(ds) -> DynamicSpectrum

Subtract mean over time for each frequency channel (background subtraction only, no clipping).

Parameter Type Description
ds DynamicSpectrum Input spectrum

Returns: New DynamicSpectrum with background subtracted. Useful for visualizing data before clipping is applied.


Cropping Functions

crop_frequency(ds, freq_min=None, freq_max=None) -> DynamicSpectrum

Crop a spectrum to a frequency range.

Parameter Type Description
ds DynamicSpectrum Input spectrum
freq_min float | None Minimum frequency in MHz (inclusive)
freq_max float | None Maximum frequency in MHz (inclusive)

Raises: CropError if range is invalid or results in empty data.


crop_time(ds, time_min=None, time_max=None) -> DynamicSpectrum

Crop a spectrum to a time range.

Parameter Type Description
ds DynamicSpectrum Input spectrum
time_min float | None Minimum time in seconds
time_max float | None Maximum time in seconds

Raises: CropError if range is invalid or results in empty data.


crop(ds, freq_range=None, time_range=None) -> DynamicSpectrum

Crop a spectrum along both axes at once.

Parameter Type Description
ds DynamicSpectrum Input spectrum
freq_range tuple | None (min, max) frequency in MHz
time_range tuple | None (min, max) time in seconds

slice_by_index(ds, freq_slice=None, time_slice=None) -> DynamicSpectrum

Slice a spectrum by array indices.

Parameter Type Description
ds DynamicSpectrum Input spectrum
freq_slice slice | None Slice for frequency axis
time_slice slice | None Slice for time axis

Combine Functions

can_combine_frequency(path1, path2, time_atol=0.01) -> bool

Check if two files can be combined along the frequency axis.


combine_frequency(path1, path2) -> DynamicSpectrum

Combine two spectra vertically (frequency stacking).


can_combine_time(paths, freq_atol=0.01) -> bool

Check if files can be combined along the time axis.


combine_time(paths) -> DynamicSpectrum

Concatenate spectra horizontally (time concatenation).


Plotting Functions

plot_dynamic_spectrum(ds, title="...", cmap="inferno", figsize=None, vmin=None, vmax=None, ax=None, show_colorbar=True, time_format="seconds", **imshow_kwargs)

Plot a dynamic spectrum with full customization.

Parameter Type Default Description
ds DynamicSpectrum Spectrum to plot
title str "Dynamic Spectrum" Plot title
cmap str "inferno" Matplotlib colormap
figsize tuple | None None Figure size as (width, height) in inches
vmin float | None None Colormap lower bound (clipping)
vmax float | None None Colormap upper bound (clipping)
ax Axes | None None Existing axes (creates new if None)
show_colorbar bool True Whether to display colorbar
time_format str "seconds" "seconds" or "ut" for time axis format
**imshow_kwargs Additional kwargs passed to matplotlib.imshow()

Returns: Tuple of (fig, ax, im).


plot_raw_spectrum(ds, title="Raw Spectrum", cmap="viridis", ...)

Plot raw spectrum data without any processing. Accepts the same parameters as plot_dynamic_spectrum.


plot_background_subtracted(ds, title="Background Subtracted", cmap="RdBu_r", ...)

Plot spectrum after background subtraction (before clipping). Automatically applies background_subtract() and plots the result. Accepts the same parameters as plot_dynamic_spectrum.


TimeAxisConverter

Convert between elapsed seconds and UT time.

@dataclass
class TimeAxisConverter:
    ut_start_sec: float  # UT observation start in seconds since midnight
Method Description
seconds_to_ut(seconds) Convert elapsed seconds to UT string (HH:MM:SS)
ut_to_seconds(ut_str) Convert UT string to elapsed seconds
from_dynamic_spectrum(ds) Create converter from spectrum metadata

Exceptions

The library provides a hierarchy of custom exceptions for robust error handling:

Exception Description
ECallistoError Base exception for all library errors
InvalidFITSError Raised when a FITS file is invalid or cannot be read
InvalidFilenameError Raised when a filename doesn't match e-CALLISTO naming convention
DownloadError Raised when downloading files from the archive fails
CombineError Raised when spectra cannot be combined
CropError Raised when cropping parameters are invalid

Error Handling Example

import ecallistolib as ecl
from ecallistolib import InvalidFITSError, CropError

try:
    spectrum = ecl.read_fits("corrupted_file.fit")
except FileNotFoundError:
    print("File not found")
except InvalidFITSError as e:
    print(f"Invalid FITS file: {e}")

try:
    cropped = ecl.crop(spectrum, freq_range=(1000, 2000))  # Out of range
except CropError as e:
    print(f"Cropping failed: {e}")

Examples

Complete Workflow

from datetime import date
import ecallistolib as ecl
import matplotlib.pyplot as plt

# 1. Download data
remote = ecl.list_remote_fits(date(2023, 6, 15), hour=12, station_substring="alaska")
paths = ecl.download_files(remote[:2], out_dir="./data")

# 2. Read and combine
if ecl.can_combine_time(paths):
    spectrum = ecl.combine_time(paths)
else:
    spectrum = ecl.read_fits(paths[0])

# 3. Process
cleaned = ecl.noise_reduce_mean_clip(spectrum)

# 4. Plot
fig, ax, im = ecl.plot_dynamic_spectrum(
    cleaned,
    title=f"e-CALLISTO Observation - {spectrum.meta.get('station', 'Unknown')}",
    cmap="plasma"
)
plt.savefig("observation.png", dpi=200)
plt.show()

Working with Metadata

import ecallistolib as ecl

spectrum = ecl.read_fits("my_file.fit.gz")

# Access metadata
print(f"Station: {spectrum.meta.get('station')}")
print(f"Date: {spectrum.meta.get('date')}")
print(f"UT Start: {spectrum.meta.get('ut_start_sec')} seconds")

# After processing, metadata is preserved and extended
processed = ecl.noise_reduce_mean_clip(spectrum)
print(f"Processing applied: {processed.meta.get('noise_reduction')}")

Data Format

e-CALLISTO FITS files follow a standard naming convention:

STATION_YYYYMMDD_HHMMSS_NN.fit.gz
Field Description
STATION Observatory name (e.g., ALASKA, GLASGOW)
YYYYMMDD Observation date
HHMMSS Observation start time (UTC)
NN Focus/channel number (typically 01 or 02)

The FITS files contain:

  • Primary HDU: 2D array of intensity values
  • Extension 1: Binary table with frequency and time axes

Contributing

Contributions are welcome! Please feel free to submit issues or pull requests.

  1. Fork the repository
  2. Create your feature branch (git checkout -b feature/amazing-feature)
  3. Commit your changes (git commit -m 'Add amazing feature')
  4. Push to the branch (git push origin feature/amazing-feature)
  5. Open a Pull Request

Running Tests

pip install pytest
pytest

License

This project is licensed under the MIT License. See the LICENSE file for details.


Acknowledgments


Links

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

ecallistolib-0.2.2.tar.gz (30.3 kB view details)

Uploaded Source

Built Distribution

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

ecallistolib-0.2.2-py3-none-any.whl (21.7 kB view details)

Uploaded Python 3

File details

Details for the file ecallistolib-0.2.2.tar.gz.

File metadata

  • Download URL: ecallistolib-0.2.2.tar.gz
  • Upload date:
  • Size: 30.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.0

File hashes

Hashes for ecallistolib-0.2.2.tar.gz
Algorithm Hash digest
SHA256 c55e58f3ad792d72c5cafc6c6e462a0c56b4eb3414c7b6ea8c6e82a484050e02
MD5 e162718c039a11d2a57c83ab6deeb42c
BLAKE2b-256 61c4890a2699d6103586351cd32b176816feb2c09c2ea52567ed552f3dad7c42

See more details on using hashes here.

File details

Details for the file ecallistolib-0.2.2-py3-none-any.whl.

File metadata

  • Download URL: ecallistolib-0.2.2-py3-none-any.whl
  • Upload date:
  • Size: 21.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.0

File hashes

Hashes for ecallistolib-0.2.2-py3-none-any.whl
Algorithm Hash digest
SHA256 3f6e65a47c7fda151a9478b9d25be5c131d97484684da5fbc8fef2550bf092be
MD5 5290a4cbf4709c4a18f86036037ee2d4
BLAKE2b-256 371fe756230254b2923cca8c9611fc84767f3797a041b87d557778da958bf532

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