Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add some routines to get background subtracted images #174

Merged
merged 28 commits into from
Dec 3, 2024
Merged
Show file tree
Hide file tree
Changes from 26 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
a7230c6
Add some routines to get background subtracted images
jeanconn Aug 28, 2024
ab2a8f1
Break into more subroutines
jeanconn Sep 17, 2024
ee6086f
Add retry
jeanconn Sep 17, 2024
7a3290a
First round simplifications
jeanconn Sep 20, 2024
820211a
Second round simplifications
jeanconn Sep 24, 2024
f18ea88
Improve docstring and start/stop errors
jeanconn Sep 25, 2024
99361f1
Use default padding and smoothing in get_tccd_data
jeanconn Oct 21, 2024
f341fb4
Reorg and progress
jeanconn Nov 6, 2024
4080ac5
Move some imports to avoid circular issues
jeanconn Nov 19, 2024
f0d8d90
Add more docstrings
jeanconn Nov 20, 2024
84d95c3
Add more tests
jeanconn Nov 20, 2024
149c9c2
Update pre-commit config
jeanconn Nov 20, 2024
de58e09
Remove old import
jeanconn Nov 20, 2024
3edd838
Fix long line
jeanconn Nov 20, 2024
f263a89
Minor renaming and doc improvements from feedback
jeanconn Nov 25, 2024
71eb86a
Fix some mica -> cxc pieces
jeanconn Nov 25, 2024
b3f58b1
Be more strict about bounds
jeanconn Nov 25, 2024
81bc8f9
Remove unmasking as no longer needed
jeanconn Nov 27, 2024
5cba36b
Wrong type in docstring
jeanconn Nov 27, 2024
6e20ccf
Add some handling for zero length img_table
jeanconn Nov 27, 2024
9b79677
Change shapes for zero-length output
taldcroft Dec 2, 2024
28625f4
Update tests (now failing)
taldcroft Dec 2, 2024
07d1c35
Docstring updates
taldcroft Dec 2, 2024
4a8a73c
Update dark map extraction to be per-pixel to work near edges
jeanconn Dec 2, 2024
3ea9707
Update test to work at positive ccd edges
jeanconn Dec 2, 2024
45e0108
Pre-ruff formatting fix
jeanconn Dec 2, 2024
4c87665
Oops, edge at -512
jeanconn Dec 2, 2024
37560e5
Move the numba function to module scope
taldcroft Dec 3, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions .pre-commit-config.yaml
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
repos:
- repo: https://github.com/psf/black
rev: 23.11.0
rev: 24.10.0
hooks:
- id: black
language_version: python3.11

- repo: https://github.com/pycqa/isort
rev: 5.12.0
rev: 5.13.2
hooks:
- id: isort
name: isort (python)
Expand Down
105 changes: 104 additions & 1 deletion chandra_aca/aca_image.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,12 @@

import numba
import numpy as np
import requests
from astropy.table import Table
from cxotime import CxoTimeLike
from ska_helpers import retry

__all__ = ["ACAImage", "centroid_fm", "AcaPsfLibrary", "EIGHT_LABELS"]
__all__ = ["ACAImage", "centroid_fm", "AcaPsfLibrary", "EIGHT_LABELS", "get_aca_images"]

EIGHT_LABELS = np.array(
[
Expand Down Expand Up @@ -828,3 +832,102 @@ def get_psf_image(

out = ACAImage(psf, row0=row0, col0=col0) if aca_image else (psf, row0, col0)
return out


@retry.retry(exceptions=requests.exceptions.RequestException, delay=5, tries=3)
def get_aca_images(
start: CxoTimeLike, stop: CxoTimeLike, bgsub=True, source="maude", **maude_kwargs
) -> Table:
"""
Get ACA images and ancillary data from either the MAUDE or CXC data sources.

The returned table of ACA images and ancillary data will include the default
columns returned by chandra_aca.maude_decom.get_aca_images or
mica.archive.aca_l0.get_aca_images. If bgsub is True (the default) then the
table will also include columns::

name dtype unit
--------------------- ------- -----------
IMG_BGSUB float64 DN
IMG_DARK float64 DN
T_CCD_SMOOTH float64 degC

where:

- 'IMG_BGSUB': background subtracted image
- 'IMG_DARK': dark current image
- 'T_CCD_SMOOTH': smoothed CCD temperature

The IMG_DARK individual values are only calculated if within the 1024x1024
dark current map, otherwise they are set to 0. In practice this is not an
issue in that IMG and IMG_BGSUB must be within the CCD to be tracked.

Parameters
----------
start : CxoTimeLike
Start time.
stop : CxoTimeLike
Stop time (CXC sec).
bgsub : bool
Include background subtracted images in output table.
source : str
Data source for image and temperature telemetry ('maude' or 'cxc'). For 'cxc',
the image telemetry is from mica and temperature telemetry is from CXC L0 via
cheta.
**maude_kwargs
Additional kwargs for maude data source.

Returns
-------
imgs_table : astropy.table.Table
Table of ACA images and ancillary data.
"""
import mica.archive.aca_dark
import mica.archive.aca_l0

import chandra_aca.dark_subtract
import chandra_aca.maude_decom

# Get aca images over the time range
if source == "maude":
imgs_table = chandra_aca.maude_decom.get_aca_images(start, stop, **maude_kwargs)
t_ccds = chandra_aca.dark_subtract.get_tccd_data(
imgs_table["TIME"], source=source, **maude_kwargs
)
elif source == "cxc":
imgs_table = mica.archive.aca_l0.get_aca_images(start, stop)
t_ccds = chandra_aca.dark_subtract.get_tccd_data(
imgs_table["TIME"], source=source
)
else:
raise ValueError(f"source must be 'maude' or 'cxc', not {source}")

# Get background subtracted values if bgsub is True.
# There's nothing to do if there are no images (len(imgs_table) == 0),
# so special case that.
if bgsub and len(imgs_table) > 0:
dark_data = mica.archive.aca_dark.get_dark_cal_props(
imgs_table["TIME"].min(),
select="nearest",
include_image=True,
aca_image=False,
)
img_dark = dark_data["image"]
tccd_dark = dark_data["ccd_temp"]
imgs_dark = chandra_aca.dark_subtract.get_dark_current_imgs(
imgs_table, img_dark, tccd_dark, t_ccds
)
imgs_bgsub = imgs_table["IMG"] - imgs_dark
imgs_bgsub.clip(0, None)

imgs_table["IMG_BGSUB"] = imgs_bgsub
imgs_table["IMG_DARK"] = imgs_dark
imgs_table["T_CCD_SMOOTH"] = t_ccds

if bgsub and len(imgs_table) == 0:
# Add the columns to the table even if there are no rows
imgs_table["IMG_BGSUB"] = np.zeros(shape=(0, 8, 8))
imgs_table["IMG_DARK"] = np.zeros(shape=(0, 8, 8))
imgs_table["T_CCD_SMOOTH"] = np.zeros(shape=0)

return imgs_table
207 changes: 207 additions & 0 deletions chandra_aca/dark_subtract.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,207 @@
import numba
import numpy as np
import requests.exceptions
import scipy.signal
from cheta import fetch_sci
from ska_helpers import retry
from ska_numpy import smooth

from chandra_aca.dark_model import dark_temp_scale_img

__all__ = [
"get_tccd_data",
"get_aca_images_bgd_sub",
"get_dark_current_imgs",
"get_dark_backgrounds",
]


@retry.retry(exceptions=requests.exceptions.RequestException, delay=5, tries=3)
def get_tccd_data(
times, smooth_window=30, median_window=3, source="maude", maude_channel=None
):
"""
Get the CCD temperature for given times and interpolate and smooth.

This is a light wrapper around cheta.fetch_sci.Msid("aacccdpt", start, stop)
to handle an option to use the maude data source in an explicit way.

Parameters
----------
times: np.array (float)
Sampling times for CCD temperature data (CXC seconds).
source : str, optional
Source of CCD temperature data ('maude' (default) or 'cxc' for cheta archive).
median_window : int, optional
Median filter window to remove outliers (default=3).
smooth_window : int, optional
Smooth data using a hanning window of this length in samples (default=30).
maude_channel : str, optional
Maude channel to use (default is flight).

Returns
-------
vals : np.array
CCD temperature values.
"""

# If no times are given, return an empty array.
if len(times) == 0:
return np.array([])

pad = 600 # 600 seconds default pad

# increase the pad if the smooth window is large
pad = np.max([smooth_window * 32.8 / 2, pad])

fetch_start = times[0] - pad
fetch_stop = times[-1] + pad

if source == "maude":
# Override the cheta data_source to be explicit about maude source.
data_source = "maude allow_subset=False"
if maude_channel is not None:
data_source += f" channel={maude_channel}"
with fetch_sci.data_source(data_source):
dat = fetch_sci.Msid("aacccdpt", fetch_start, fetch_stop)
elif source == "cxc":
with fetch_sci.data_source("cxc"):
dat = fetch_sci.Msid("aacccdpt", fetch_start, fetch_stop)
else:
raise ValueError(f"Unknown source: {source}")

yin = dat.vals.copy()

if median_window > 0:
# Filter the data using a median filter from scipy
yin = scipy.signal.medfilt(yin, kernel_size=median_window)

if smooth_window > 0:
# And then smooth it with hanning and window_len = smooth_window
yin = smooth(yin, window_len=smooth_window)

# Interpolate the data to the requested times
vals = np.interp(times, dat.times, yin)

return vals


def get_aca_images_bgd_sub(img_table, t_ccd_vals, img_dark, tccd_dark):
jeanconn marked this conversation as resolved.
Show resolved Hide resolved
jeanconn marked this conversation as resolved.
Show resolved Hide resolved
"""
Get the background subtracted ACA images.

Parameters
----------
img_table : astropy.table.Table
Table of ACA images with columns 'IMG', 'IMGROW0_8X8', 'IMGCOL0_8X8', 'INTEG'.
t_ccd_vals : np.array
CCD temperature values at the times of the ACA images (deg C).
img_dark : np.array
Dark calibration image. Must be 1024x1024 (e-/s).
tccd_dark : float
Reference temperature of the dark calibration image (deg C).

Returns
-------
tuple (imgs_bgsub, imgs_dark)
imgs_bgsub : np.array
Background subtracted ACA images (DN).
imgs_dark : np.array
Dark current images (DN).
"""
imgs_dark = get_dark_current_imgs(img_table, img_dark, tccd_dark, t_ccd_vals)
imgs_bgsub = img_table["IMG"] - imgs_dark
imgs_bgsub.clip(0, None)

return imgs_bgsub, imgs_dark


def get_dark_current_imgs(img_table, img_dark, tccd_dark, t_ccds):
"""
Get the scaled dark current values for a table of ACA images.

This scales the dark current to the appropriate temperature and integration time,
returning the dark current in DN matching the ACA images in img_table.

Parameters
----------
img_table : astropy.table.Table
Table of ACA images with columns 'IMG', 'IMGROW0_8X8', 'IMGCOL0_8X8', 'INTEG'.
img_dark : 1024x1024 array
Dark calibration image.
tccd_dark : float
Reference temperature of the dark calibration image.
t_ccds : array
Cheta temperature values at the times of img_table.

Returns
-------
imgs_dark : np.array (len(img_table), 8, 8)
Temperature scaled dark current for each ACA image in img_table (DN).

"""
if len(img_table) != len(t_ccds):
raise ValueError("img_table and t_ccds must have the same length")

if img_dark.shape != (1024, 1024):
raise ValueError("dark image shape is not 1024x1024")

imgs_dark_unscaled = get_dark_backgrounds(
img_dark,
img_table["IMGROW0_8X8"],
img_table["IMGCOL0_8X8"],
)

imgs_dark = np.zeros((len(img_table), 8, 8), dtype=np.float64)

# Scale the dark current at each dark cal 8x8 image to the ccd temperature and e-/s
for i, (img_dark_unscaled, t_ccd_i, img) in enumerate(
zip(imgs_dark_unscaled, t_ccds, img_table, strict=True)
):
img_scaled = dark_temp_scale_img(img_dark_unscaled, tccd_dark, t_ccd_i)
# Default to using 1.696 integ time if not present in the image table
integ = img["INTEG"] if "INTEG" in img.colnames else 1.696
imgs_dark[i] = img_scaled * integ / 5

return imgs_dark


def get_dark_backgrounds(raw_dark_img, imgrow0, imgcol0, size=8):
"""
Get dark background cutouts at a set of ACA image positions.

Parameters
----------
raw_dark_img : np.array
Dark calibration image.
imgrow0 : np.array (int)
Row of ACA image.
imgcol0 : np.array (int)
Column of ACA image.
size : int, optional
Size of ACA image (default=8).

Returns
-------
imgs_dark : np.array (len(imgrow0), size, size)
Dark backgrounds for image locations sampled from raw_dark_img (e-/s).
Pixels outside raw_dark_img are set to 0.0.
"""
CCD_MAX = 1024
CCD_MIN = 0

@numba.jit(nopython=True)
def staggered_aca_slice(array_in, array_out, row, col):
for idx in np.arange(len(row)):
row_out = np.zeros((size, size), dtype=np.float64)
for i, r in enumerate(np.arange(row[idx], row[idx] + size)):
for j, c in enumerate(np.arange(col[idx], col[idx] + size)):
if r >= CCD_MIN and r < CCD_MAX and c >= CCD_MIN and c < CCD_MAX:
row_out[i, j] = array_in[r, c]
taldcroft marked this conversation as resolved.
Show resolved Hide resolved
array_out[idx] = row_out

imgs_dark = np.zeros([len(imgrow0), size, size], dtype=np.float64)
staggered_aca_slice(
raw_dark_img.astype(float), imgs_dark, 512 + imgrow0, 512 + imgcol0
)
return imgs_dark
Loading
Loading