Source code for astropy.units.format.fits

# Licensed under a 3-clause BSD style license - see LICENSE.rst

"""
Handles the "FITS" unit format.
"""

from typing import Literal

import numpy as np

from astropy.units.core import CompositeUnit, UnitBase
from astropy.units.enums import DeprecatedUnitAction
from astropy.units.errors import UnitScaleError
from astropy.utils import classproperty

from . import Base
from .generic import _GenericParserMixin


[docs] class FITS(Base, _GenericParserMixin): """ The FITS standard unit format. This supports the format defined in the Units section of the `FITS Standard <https://fits.gsfc.nasa.gov/fits_standard.html>`_. """ @classproperty(lazy=True) def _units(cls) -> dict[str, UnitBase]: from astropy import units as u names = { "Celsius": u.deg_C, "deg C": u.deg_C, "dbyte": u.Unit("dbyte", 0.1 * u.byte), "as": u.attosecond, } bases = [ "m", "g", "s", "rad", "sr", "K", "A", "mol", "cd", "Hz", "J", "W", "V", "N", "Pa", "C", "Ohm", "S", "F", "Wb", "T", "H", "lm", "lx", "a", "yr", "eV", "pc", "Jy", "mag", "R", "bit", "byte", "G", "barn", ] # fmt: skip prefixes = [ "y", "z", "a", "f", "p", "n", "u", "m", "c", "d", "", "da", "h", "k", "M", "G", "T", "P", "E", "Z", "Y", ] # fmt: skip for key in (prefix + base for base in bases for prefix in prefixes): if key not in names: names[key] = getattr(u, key) simple_units = [ "deg", "arcmin", "arcsec", "mas", "min", "h", "d", "Ry", "solMass", "u", "solLum", "solRad", "AU", "lyr", "count", "ct", "photon", "ph", "pixel", "pix", "D", "Sun", "chan", "bin", "voxel", "adu", "beam", "erg", "Angstrom", "angstrom", ] # fmt: skip names.update((unit, getattr(u, unit)) for unit in simple_units) return names
[docs] @classmethod def to_string( cls, unit: UnitBase, fraction: bool | Literal["inline", "multiline"] = False, deprecations: DeprecatedUnitAction = DeprecatedUnitAction.WARN, ) -> str: # Remove units that aren't known to the format unit = cls._decompose_to_known_units(unit) parts = [] base = np.log10(unit.scale) if base % 1.0 != 0.0: raise UnitScaleError( "The FITS unit format is not able to represent scales " "that are not powers of 10. Multiply your data by " f"{unit.scale:e}." ) elif unit.scale != 1.0: # We could override format_exponential_notation to set the # scale factor but that would give the wrong impression that # all values in FITS are set that way. So, instead do it # here, and use a unity-scale unit for the rest. parts.append(f"10**{int(base)}") unit = CompositeUnit(1, unit.bases, unit.powers) if unit.bases: parts.append(super().to_string(unit, fraction=fraction)) return cls._scale_unit_separator.join(parts)
[docs] @classmethod def parse(cls, s: str, debug: bool = False) -> UnitBase: result = cls._do_parse(s, debug) if hasattr(result, "function_unit"): raise ValueError("Function units are not yet supported for FITS units.") return result