Source code for astrocut.fits_cutout

import warnings
from datetime import date
from pathlib import Path
from time import monotonic
from typing import List, Literal, Optional, Tuple, Union

import numpy as np
from astropy import log as astropy_log
from astropy.coordinates import SkyCoord
from astropy.io import fits
from astropy.io.fits import HDUList
from astropy.nddata import NoOverlapError
from astropy.units import Quantity
from astropy.utils.decorators import deprecated_renamed_argument
from astropy.wcs import WCS
from s3path import S3Path

from . import __version__, log
from .exceptions import DataWarning, InvalidQueryError
from .image_cutout import ImageCutout


[docs] class FITSCutout(ImageCutout): """ Class for creating cutouts from FITS files. Parameters ---------- input_files : list List of input image files. coordinates : str | `~astropy.coordinates.SkyCoord` Coordinates of the center of the cutout. cutout_size : int | array | list | tuple | `~astropy.units.Quantity` Size of the cutout array. fill_value : int | float Value to fill the cutout with if the cutout is outside the image. limit_rounding_method : str Method to use for rounding the cutout limits. Options are 'round', 'ceil', and 'floor'. extension : int | list | 'all' Optional, default None. The extension(s) to cutout from. If None, the first extension with data is used. single_outfile : bool Optional, default True. If True, all cutouts are written to a single file or HDUList. verbose : bool If True, log messages are printed to the console. fsspec_kwargs : dict Optional, default None. Keyword arguments to pass through to `s3fs` for cloud-hosted files. Attributes ---------- cutouts_by_file : dict The cutouts as a list of `FITSCutout.CutoutInstance` objects stored by input filename. fits_cutouts : list The cutouts as a list of `astropy.io.fits.HDUList` objects. hdu_cutouts_by_file : dict The cutouts as `astropy.io.fits.ImageHDU` objects stored by input filename. Methods ------- cutout() Generate cutouts from a list of input images. write_as_fits(output_dir, cutout_prefix) Write the cutouts to files in FITS format. """ def __init__( self, input_files: List[Union[str, Path, S3Path]], coordinates: Union[SkyCoord, str], cutout_size: Union[int, np.ndarray, Quantity, List[int], Tuple[int]] = 25, fill_value: Union[int, float] = np.nan, limit_rounding_method: str = "round", extension: Optional[Union[int, List[int], Literal["all"]]] = None, single_outfile: bool = True, verbose: bool = False, *, fsspec_kwargs: Optional[dict] = None, ): # Superclass constructor super().__init__(input_files, coordinates, cutout_size, fill_value, limit_rounding_method, verbose) # If a single extension is given, make it a list if isinstance(extension, int): extension = [extension] self._extension = extension # Assigning other attributes self._single_outfile = single_outfile self._fits_cutouts = None self.hdu_cutouts_by_file = {} self._fsspec_kwargs = fsspec_kwargs # Make the cutouts upon initialization self.cutout() def _construct_fits_from_hdus(self, cutout_hdus: List[fits.ImageHDU]) -> fits.HDUList: """ Make one or more cutout HDUs into a single HDUList object. Parameters ---------- cutout_hdus : list The `~astropy.io.fits.hdu.image.ImageHDU` object(s) to be written to the fits file. Returns ------- response : `~astropy.io.fits.HDUList` The HDUList object. """ # Setting up the Primary HDU keywords = dict() if self._coordinates: keywords = { "RA_OBJ": (self._coordinates.ra.deg, "[deg] right ascension"), "DEC_OBJ": (self._coordinates.dec.deg, "[deg] declination"), } # Build the primary HDU with keywords primary_hdu = fits.PrimaryHDU() primary_hdu.header.extend( [ ("ORIGIN", "STScI/MAST", "institution responsible for creating this file"), ("DATE", str(date.today()), "file creation date"), ("PROCVER", __version__, "software version"), ] ) for kwd in keywords: primary_hdu.header[kwd] = keywords[kwd] return fits.HDUList([primary_hdu] + cutout_hdus) @property def fits_cutouts(self): """ Return the cutouts as a list `astropy.io.fits.HDUList` objects. """ if not self._fits_cutouts: fits_cutouts = [] if self._single_outfile: # one output file for all input files cutout_hdus = [x for file in self.hdu_cutouts_by_file for x in self.hdu_cutouts_by_file[file]] fits_cutouts = [self._construct_fits_from_hdus(cutout_hdus)] else: # one output file per input file for file, cutout_list in self.hdu_cutouts_by_file.items(): fits_cutouts.append(self._construct_fits_from_hdus(cutout_list)) self._fits_cutouts = fits_cutouts return self._fits_cutouts def _parse_extensions(self, input_file: Union[str, Path, S3Path], infile_exts: np.ndarray) -> List[int]: """ Given a list of image extensions available in the file with infile_name, cross-match with user input extensions to figure out which extensions to use for cutout. Parameters ---------- input_file : str | Path | S3Path The path to the input file. infile_exts : list List of image extensions available in the file. Returns ------- cutout_exts : list List of extensions to be cutout. """ # Skip files with no image data if len(infile_exts) == 0: warnings.warn(f"No image extensions with data found in {input_file}, skipping...", DataWarning) return [] if self._extension is None: cutout_exts = infile_exts[:1] # Take the first image extension elif self._extension == "all": cutout_exts = infile_exts # Take all the extensions else: # User input extentions cutout_exts = [x for x in infile_exts if x in self._extension] if len(cutout_exts) < len(self._extension): empty_exts = ",".join([str(x) for x in self._extension if x not in cutout_exts]) warnings.warn( f"Not all requested extensions in {input_file} are image extensions or have " f"data, extension(s) {empty_exts} will be skipped.", DataWarning, ) return cutout_exts def _load_file_data(self, input_file: Union[str, Path, S3Path]) -> Tuple[fits.HDUList, List[int]]: """ Load the data from an input file and determine which extension(s) to cutout from. Parameters ---------- input_file : str | Path | S3Path The path to the input file. Returns -------- hdulist : `~astropy.io.fits.HDUList` The HDU list for the input file. cutout_inds : list The indices of the extension(s) to cutout from. """ # Account for cloud-hosted files fsspec_kwargs = None if "s3://" in input_file: fsspec_kwargs = {"anon": True} if self._fsspec_kwargs: fsspec_kwargs.update(self._fsspec_kwargs) # Open the file hdulist = fits.open(input_file, mode="denywrite", memmap=True, fsspec_kwargs=fsspec_kwargs) # Sorting out which extension(s) to cutout infile_exts = np.where([hdu.is_image and hdu.size > 0 for hdu in hdulist])[0] cutout_inds = self._parse_extensions(input_file, infile_exts) return (hdulist, cutout_inds) def _get_img_wcs(self, hdu_header: fits.Header) -> Tuple[WCS, bool]: """ Get the WCS for an image. Parameters ---------- hdu_header : `~astropy.io.fits.Header` The header for the image HDU. Returns -------- img_wcs : `~astropy.wcs.WCS` The WCS for the image. no_sip : bool Whether the image WCS has no SIP information. """ # We are going to reroute the logging to a string stream temporarily so we can # intercept any message from astropy, chiefly the "Inconsistent SIP distortion information" # INFO message which will indicate that we need to remove existing SIP keywords # from a WCS whose CTYPE does not include SIP. In this we are taking the CTYPE to be # correct and adjusting the header keywords to match. hdlrs = astropy_log.handlers astropy_log.handlers = [] with astropy_log.log_to_list() as log_list: img_wcs = WCS(hdu_header, relax=True) for hd in hdlrs: astropy_log.addHandler(hd) no_sip = False if len(log_list) > 0: if "Inconsistent SIP distortion information" in log_list[0].msg: # Remove sip coefficients img_wcs.sip = None no_sip = True else: # Message(s) we didn't prepare for we want to go ahead and display for log_rec in log_list: astropy_log.log(log_rec.levelno, log_rec.msg, extra={"origin": log_rec.name}) return (img_wcs, no_sip) def _hducut( self, cutout_data: np.ndarray, cutout_wcs: WCS, hdu_header: fits.Header, no_sip: bool, ind: int, primary_filename: fits.Header, is_empty: bool, ) -> fits.ImageHDU: """ Create a cutout HDU from an image HDU. Parameters ---------- cutout_data : `numpy.ndarray` The cutout data. img_wcs : `~astropy.wcs.WCS` The WCS for the image. hdu_header : `~astropy.io.fits.Header` The header for the image HDU. no_sip : bool Whether the image WCS has no SIP information. ind : int The index of the extension in the original file. primary_filename : str The filename in the header of the primary HDU. is_empty : bool Indicates if the cutout has no image data. Returns ------- response : `~astropy.io.fits.ImageHDU` The cutout HDU. """ # Updating the header with the new wcs info if no_sip: hdu_header.update(cutout_wcs.to_header(relax=False)) else: hdu_header.update(cutout_wcs.to_header(relax=True)) # relax arg is for sip distortions if they exist # Naming the extension and preserving the original name hdu_header["O_EXT_NM"] = (hdu_header.get("EXTNAME"), "Original extension name.") hdu_header["EXTNAME"] = "CUTOUT" # Moving the filename, if present, into the ORIG_FLE keyword hdu_header["ORIG_FLE"] = (hdu_header.get("FILENAME"), "Original image filename.") hdu_header.remove("FILENAME", ignore_missing=True) # Check that there is data in the cutout image if is_empty: hdu_header["EMPTY"] = (True, "Indicates no data in cutout image.") # Create the cutout HDU cutout_hdu = fits.ImageHDU(header=hdu_header, data=cutout_data) # Adding a few more keywords cutout_hdu.header["ORIG_EXT"] = (ind, "Extension in original file.") if not cutout_hdu.header.get("ORIG_FLE") and primary_filename: cutout_hdu.header["ORIG_FLE"] = primary_filename return cutout_hdu def _cutout_file(self, file: Union[str, Path, S3Path]): """ Create cutouts from a single file. Parameters ---------- file : str | Path | S3Path The path to the file. """ # Load data hdulist, cutout_inds = self._load_file_data(file) if not len(cutout_inds): # No image extensions with data were found hdulist.close() return # Create HDU cutouts cutouts = [] fits_cutouts = [] num_empty = 0 for ind in cutout_inds: try: # Get HDU, header, and WCS img_hdu = hdulist[ind] hdu_header = fits.Header(img_hdu.header, copy=True) img_wcs, no_sip = self._get_img_wcs(hdu_header) primary_filename = hdulist[0].header.get("FILENAME") # Create the cutout # Eventually, this will be replaced by a call to Cutout2D cutout = self.CutoutInstance(img_hdu.section, img_wcs, self) # Save the cutout data to use when outputting as an image # Eventually, the values here will be a list of Cutout2D objects is_empty = (cutout.data == 0).all() or (np.isnan(cutout.data)).all() if is_empty: num_empty += 1 else: cutouts.append(cutout) # Also save the cutouts as ImageHDU objects for FITS output fits_cutouts.append( self._hducut(cutout.data, cutout.wcs, hdu_header, no_sip, ind, primary_filename, is_empty) ) except OSError as err: warnings.warn( f"Error {err} encountered when performing cutout on {file}, extension {ind}, skipping...", DataWarning, ) num_empty += 1 except NoOverlapError: warnings.warn( f"Cutout footprint does not overlap with data in {file}, extension {ind}, skipping...", DataWarning, ) num_empty += 1 except InvalidQueryError: warnings.warn( f"Cutout footprint does not overlap with data in {file}, extension {ind}, skipping...", DataWarning, ) num_empty += 1 except ValueError as err: if "Input position contains invalid values" in str(err): warnings.warn( f"Cutout footprint does not overlap with data in {file}, extension {ind}, skipping...", DataWarning, ) num_empty += 1 else: raise # Close HDUList hdulist.close() if num_empty == len(cutout_inds): # No extensions have cutout data warnings.warn(f"Cutout of {file} contains no data, skipping...", DataWarning) else: # At least one extension has cutout data # Save cutouts self.cutouts_by_file[file] = cutouts self.hdu_cutouts_by_file[file] = fits_cutouts
[docs] def cutout(self) -> Union[str, List[str], List[fits.HDUList]]: """ Generate cutouts from a list of input images. Returns ------- cutout_path : Path | list Cutouts as memory objects or path(s) to the written cutout files. Raises ------ InvalidQueryError If no cutouts contain data. """ # Track start time start_time = monotonic() # Cutout each input file for file in self._input_files: self._cutout_file(file) # If no cutouts contain data, raise exception if not self.cutouts_by_file: raise InvalidQueryError("Cutout contains no data! (Check image footprint.)") # Log total time elapsed log.debug("Total time: %.2f sec", monotonic() - start_time) return self.fits_cutouts
[docs] def write_as_fits(self, output_dir: Union[str, Path] = ".", cutout_prefix: str = "cutout") -> List[str]: """ Write the cutouts to memory or to a file in FITS format. Returns ------- cutout_paths : list A list of paths to the cutout FITS files. """ Path(output_dir).mkdir(parents=True, exist_ok=True) if self._single_outfile: # one output file for all input files log.debug("Returning cutout as a single FITS file.") cutout_fits = self.fits_cutouts[0] filename = self._make_cutout_filename(cutout_prefix) cutout_path = Path(output_dir, filename) with warnings.catch_warnings(): warnings.simplefilter("ignore") cutout_fits.writeto(cutout_path, overwrite=True, checksum=True) # Return file path or memory object return [cutout_path.as_posix()] else: # one output file per input file log.debug("Returning cutouts as individual FITS files.") cutout_paths = [] for i, file in enumerate(self.hdu_cutouts_by_file): cutout_fits = self.fits_cutouts[i] filename = self._make_cutout_filename(Path(file).stem) cutout_path = Path(output_dir, filename) with warnings.catch_warnings(): warnings.simplefilter("ignore") cutout_fits.writeto(cutout_path, overwrite=True, checksum=True) # Append file path or memory object cutout_paths.append(cutout_path.as_posix()) log.debug("Cutout filepaths: {}".format(cutout_paths)) return cutout_paths
[docs] def write_as_zip(self, output_dir: Union[str, Path] = ".", filename: Union[str, Path, None] = None) -> str: """ Package the FITS cutouts into a zip archive without writing intermediate files. Parameters ---------- output_dir : str | Path, optional Directory where the zip will be created. Default '.'. filename : str | Path | None, optional Name (or path) of the output zip file. If not provided, defaults to 'astrocut_{ra}_{dec}_{size}.zip'. If provided without a '.zip' suffix, the suffix is added automatically. Returns ------- str Path to the created zip file. """ def build_entries(): if self._single_outfile: # Mirror the single-file naming used by write_as_fits arcname = self._make_cutout_filename("cutout") hdu = self.fits_cutouts[0] yield arcname, hdu else: # One file per input; mirror write_as_fits naming for i, file in enumerate(self.hdu_cutouts_by_file): arcname = self._make_cutout_filename(Path(file).stem) hdu = self.fits_cutouts[i] yield arcname, hdu return self._write_cutouts_to_zip(output_dir=output_dir, filename=filename, build_entries=build_entries)
class CutoutInstance: """ Represents an individual cutout with its own data and WCS. Eventually, this will be replaced by `astropy.nddata.Cutout2D` objects. Parameters ---------- img_data : `~astropy.io.fits.Section` The data for the image. img_wcs : `~astropy.wcs.WCS` The WCS for the image. parent : `FITSCutout` The parent FITSCutout object. Attributes ---------- data : `numpy.ndarray` The cutout data array. shape : tuple The shape of the cutout data array. shape_input : tuple The shape of the input image data array. slices_original : tuple Slices for the minimal bounding box of the cutout with respect to the original array. xmin_original : int The minimum x value of the cutout in the original array. xmax_original : int The maximum x value of the cutout in the original array. ymin_original : int The minimum y value of the cutout in the original array. ymax_original : int The maximum y value of the cutout in the original array. wcs : `~astropy.wcs.WCS` The WCS for the cutout. """ def __init__(self, img_data: fits.Section, img_wcs: WCS, parent: "FITSCutout"): # Calculate cutout limits cutout_lims = parent._get_cutout_limits(img_wcs) # Extract data from Section self.data = self._get_cutout_data(img_data, cutout_lims, parent) self.shape = self.data.shape self.shape_input = img_data.shape self.wcs = self._get_cutout_wcs(img_wcs, cutout_lims) def _get_cutout_data(self, data: fits.Section, cutout_lims: np.ndarray, parent: "FITSCutout") -> np.ndarray: """ Extract the cutout data from an image. Parameters ---------- data : `~astropy.io.fits.Section` The data for the image. cutout_lims : `numpy.ndarray` The cutout pixel limits in an array of the form [[ymin,ymax],[xmin,xmax]] parent : `FITSCutout` The parent FITSCutout object. Needed for access to certain attributes. Returns -------- cutout_data : `numpy.ndarray` The cutout data. """ log.debug("Original image shape: %s", data.shape) # Get the limits for the cutout # These limits are not guaranteed to be within the image footprint (xmin, xmax), (ymin, ymax) = cutout_lims ymax_img, xmax_img = data.shape # Check the cutout is on the image if (xmax <= 0) or (xmin >= xmax_img) or (ymax <= 0) or (ymin >= ymax_img): raise InvalidQueryError("Cutout location is not in image footprint!") # Adjust limits to fit within image bounds xmin_clipped, xmax_clipped = max(0, xmin), min(xmax_img, xmax) ymin_clipped, ymax_clipped = max(0, ymin), min(ymax_img, ymax) # Compute padding required (before and after in x and y) padding = np.array( [ (max(0, -ymin), max(0, ymax - ymax_img)), # (top, bottom) (max(0, -xmin), max(0, xmax - xmax_img)), # (left, right) ] ) # Extract the cutout img_cutout = data[ymin_clipped:ymax_clipped, xmin_clipped:xmax_clipped] # Assign relevant attributes self.slices_original = slice(ymin_clipped, ymax_clipped + 1), slice(xmin_clipped, xmax_clipped + 1) self.xmin_original, self.xmax_original = xmin_clipped, xmax_clipped self.ymin_original, self.ymax_original = ymin_clipped, ymax_clipped # Adding padding to the cutout so that it's the expected size if padding.any(): # only do if we need to pad img_cutout = np.pad(img_cutout, padding, "constant", constant_values=parent._fill_value) log.debug("Image cutout shape: %s", img_cutout.shape) return img_cutout def _get_cutout_wcs(self, img_wcs: WCS, cutout_lims: np.ndarray) -> WCS: """ Starting with the full image WCS and adjusting it for the cutout WCS. Adjusts CRPIX values and adds physical WCS keywords. Parameters ---------- img_wcs : `~astropy.wcs.WCS` WCS for the image the cutout is being cut from. cutout_lims : `numpy.ndarray` The cutout pixel limits in an array of the form [[ymin,ymax],[xmin,xmax]] Returns -------- response : `~astropy.wcs.WCS` The cutout WCS object including SIP distortions if present. """ # relax = True is important when the WCS has sip distortions, otherwise it has no effect wcs_header = img_wcs.to_header(relax=True) # Adjusting the CRPIX values wcs_header["CRPIX1"] -= cutout_lims[0, 0] wcs_header["CRPIX2"] -= cutout_lims[1, 0] # Adding the physical WCS keywords wcs_header.set("WCSNAMEP", "PHYSICAL", "name of world coordinate system alternate P") wcs_header.set("WCSAXESP", 2, "number of WCS physical axes") wcs_header.set("CTYPE1P", "RAWX", "physical WCS axis 1 type CCD col") wcs_header.set("CUNIT1P", "PIXEL", "physical WCS axis 1 unit") wcs_header.set("CRPIX1P", 1, "reference CCD column") wcs_header.set("CRVAL1P", cutout_lims[0, 0] + 1, "value at reference CCD column") wcs_header.set("CDELT1P", 1.0, "physical WCS axis 1 step") wcs_header.set("CTYPE2P", "RAWY", "physical WCS axis 2 type CCD col") wcs_header.set("CUNIT2P", "PIXEL", "physical WCS axis 2 unit") wcs_header.set("CRPIX2P", 1, "reference CCD row") wcs_header.set("CRVAL2P", cutout_lims[1, 0] + 1, "value at reference CCD row") wcs_header.set("CDELT2P", 1.0, "physical WCS axis 2 step") return WCS(wcs_header)
[docs] @deprecated_renamed_argument( "correct_wcs", None, "1.0.0", warning_type=DeprecationWarning, message="`correct_wcs` is non-operational and will be removed in a future version.", ) def fits_cut( input_files: List[Union[str, Path, S3Path]], coordinates: Union[SkyCoord, str], cutout_size: Union[int, np.ndarray, Quantity, List[int], Tuple[int]] = 25, correct_wcs: bool = False, extension: Optional[Union[int, List[int], Literal["all"]]] = None, single_outfile: bool = True, cutout_prefix: str = "cutout", output_dir: Union[str, Path] = ".", memory_only: bool = False, fill_value: Union[int, float] = np.nan, limit_rounding_method: str = "round", verbose=False, *, fsspec_kwargs: Optional[dict] = None, ) -> Union[str, List[str], List[HDUList]]: """ Takes one or more FITS files with the same WCS/pointing, makes the same cutout in each file, and returns the result either in a single FITS file with one cutout per extension or in individual fits files. The memory_only flag allows the cutouts to be returned as `~astropy.io.fits.HDUList` objects rather than saving to disk. Note: No checking is done on either the WCS pointing or pixel scale. If images don't line up the cutouts will also not line up. This function is maintained for backwards compatibility. For maximum flexibility, we recommend using ``FITSCutout`` directly. Parameters ---------- input_files : list List of fits image files to cutout from. The image is assumed to be in the first extension. coordinates : str or `~astropy.coordinates.SkyCoord` object The position around which to cutout. It may be specified as a string ("ra dec" in degrees) or as the appropriate `~astropy.coordinates.SkyCoord` object. cutout_size : int, array-like, `~astropy.units.Quantity` The size of the cutout array. If ``cutout_size`` is a scalar number or a scalar `~astropy.units.Quantity`, then a square cutout of ``cutout_size`` will be created. If ``cutout_size`` has two elements, they should be in ``(ny, nx)`` order. Scalar numbers in ``cutout_size`` are assumed to be in units of pixels. `~astropy.units.Quantity` objects must be in pixel or angular units. extension : int, list of ints, None, or 'all' Optional, default None. Default is to cutout the first extension that has image data. The user can also supply one or more extensions to cutout from (integers), or 'all'. single_outfile : bool Default True. If true return all cutouts in a single fits file with one cutout per extension, if False return cutouts in individual fits files. If returing a single file the filename will have the form: <cutout_prefix>_<ra>_<dec>_<size x>_<size y>.fits. If returning multiple files each will be named: <original filemame base>_<ra>_<dec>_<size x>_<size y>.fits. cutout_prefix : str Default value "cutout". Only used if single_outfile is True. A prefix to prepend to the cutout filename. output_dir : str Default value '.'. The directory to save the cutout file(s) to. memory_only : bool Default value False. If set to true, instead of the cutout file(s) being written to disk the cutout(s) are returned as a list of `~astropy.io.fit.HDUList` objects. If set to True cutout_prefix and output_dir are ignored, however single_outfile can still be used to set the number of returned `~astropy.io.fits.HDUList` objects. fill_value : int | float Value to fill the cutout with if the cutout is outside the image. limit_rounding_method : str Method to use for rounding the cutout limits. Options are 'round', 'ceil', and 'floor'. verbose : bool Default False. If true intermediate information is printed. fsspec_kwargs : dict Optional, default None. Keyword arguments to pass through to `s3fs` for cloud-hosted files. Returns ------- response : str or list If single_outfile is True, returns the single output filepath. Otherwise, returns a list of all the output filepaths. If memory_only is True, a list of `~astropy.io.fit.HDUList` objects is returned instead of file name(s). """ fits_cutout = FITSCutout( input_files, coordinates, cutout_size, fill_value, limit_rounding_method, extension, single_outfile, verbose=verbose, fsspec_kwargs=fsspec_kwargs, ) if memory_only: return fits_cutout.fits_cutouts cutout_paths = fits_cutout.write_as_fits(output_dir, cutout_prefix) return cutout_paths[0] if len(cutout_paths) == 1 else cutout_paths
[docs] def img_cut( input_files: List[Union[str, Path, S3Path]], coordinates: Union[SkyCoord, str], cutout_size: Union[int, np.ndarray, Quantity, List[int], Tuple[int]] = 25, stretch: str = "asinh", minmax_percent: Optional[List[int]] = None, minmax_value: Optional[List[int]] = None, invert: bool = False, img_format: str = ".jpg", colorize: bool = False, cutout_prefix: str = "cutout", output_dir: Union[str, Path] = ".", extension: Optional[Union[int, List[int], Literal["all"]]] = None, fill_value: Union[int, float] = np.nan, limit_rounding_method: str = "round", verbose=False, *, fsspec_kwargs: Optional[dict] = None, ) -> Union[str, List[str]]: """ Takes one or more fits files with the same WCS/pointing, makes the same cutout in each file, and returns the result either as a single color image or in individual image files. Note: No checking is done on either the WCS pointing or pixel scale. If images don't line up the cutouts will also not line up. This function is maintained for backwards compatibility. For maximum flexibility, we recommend using ``FITSCutout`` directly. Parameters ---------- input_files : list List of fits image files to cutout from. The image is assumed to be in the first extension. coordinates : str or `~astropy.coordinates.SkyCoord` object The position around which to cutout. It may be specified as a string ("ra dec" in degrees) or as the appropriate `~astropy.coordinates.SkyCoord` object. cutout_size : int, array-like, `~astropy.units.Quantity` The size of the cutout array. If ``cutout_size`` is a scalar number or a scalar `~astropy.units.Quantity`, then a square cutout of ``cutout_size`` will be created. If ``cutout_size`` has two elements, they should be in ``(ny, nx)`` order. Scalar numbers in ``cutout_size`` are assumed to be in units of pixels. `~astropy.units.Quantity` objects must be in pixel or angular units. stretch : str Optional, default 'asinh'. The stretch to apply to the image array. Valid values are: asinh, sinh, sqrt, log, linear minmax_percent : array Optional, default [0.5,99.5]. Interval based on a keeping a specified fraction of pixels (can be asymmetric) when scaling the image. The format is [lower percentile, upper percentile], where pixel values below the lower percentile and above the upper percentile are clipped. Only one of minmax_percent and minmax_value should be specified. minmax_value : array Optional. Interval based on user-specified pixel values when scaling the image. The format is [min value, max value], where pixel values below the min value and above the max value are clipped. Only one of minmax_percent and minmax_value should be specified. invert : bool Optional, default False. If True the image is inverted (light pixels become dark and vice versa). img_format : str Optional, default 'jpg'. The output image file type. Valid values are "jpg" and "png". colorize : bool Optional, default False. If True a single color image is produced as output, and it is expected that three files are given as input. cutout_prefix : str Default value "cutout". Only used when producing a color image. A prefix to prepend to the cutout filename. output_dir : str Defaul value '.'. The directory to save the cutout file(s) to. fill_value : int | float Value to fill the cutout with if the cutout is outside the image. limit_rounding_method : str Method to use for rounding the cutout limits. Options are 'round', 'ceil', and 'floor'. extension : int, list of ints, None, or 'all' Optional, default None. Default is to cutout the first extension that has image data. The user can also supply one or more extensions to cutout from (integers), or "all". verbose : bool Default False. If true intermediate information is printed. fsspec_kwargs : dict Optional, default None. Keyword arguments to pass through to `s3fs` for cloud-hosted files. Returns ------- response : str or list If colorize is True, returns the single output filepath. Otherwise, returns a list of all the output filepaths. """ fits_cutout = FITSCutout( input_files, coordinates, cutout_size, fill_value, limit_rounding_method, extension, verbose=verbose, fsspec_kwargs=fsspec_kwargs, ) cutout_paths = fits_cutout.write_as_img( stretch, minmax_percent, minmax_value, invert, colorize, img_format, output_dir, cutout_prefix ) return cutout_paths