Source code for gammapy.catalog.hawc

# Licensed under a 3-clause BSD style license - see LICENSE.rst
"""HAWC catalogs (https://www.hawc-observatory.org)."""
import abc
import numpy as np
from astropy.table import Table
from gammapy.modeling.models import Model, SkyModel
from gammapy.utils.scripts import make_path
from .core import SourceCatalog, SourceCatalogObject

__all__ = [
    "SourceCatalog2HWC",
    "SourceCatalog3HWC",
    "SourceCatalogObject2HWC",
    "SourceCatalogObject3HWC",
]


class SourceCatalogObjectHWCBase(SourceCatalogObject, abc.ABC):
    """Base class for the HAWC catalogs objects."""

    _source_name_key = "source_name"

    def __str__(self):
        return self.info()

    def info(self, info="all"):
        """Summary information string.

        Parameters
        ----------
        info : {'all', 'basic', 'position', 'spectrum'}
            Comma separated list of options
        """
        if info == "all":
            info = "basic,position,spectrum"

        ss = ""
        ops = info.split(",")
        if "basic" in ops:
            ss += self._info_basic()
        if "position" in ops:
            ss += self._info_position()
        if "spectrum" in ops:
            ss += self._info_spectrum()

        return ss

    def _info_basic(self):
        """Print basic information."""
        return (
            f"\n*** Basic info ***\n\n"
            f"Catalog row index (zero-based) : {self.row_index}\n"
            f"Source name : {self.name}\n"
        )

    def _info_position(self):
        """Print position information."""
        return (
            f"\n*** Position info ***\n\n"
            f"RA: {self.data.ra:.3f}\n"
            f"DEC: {self.data.dec:.3f}\n"
            f"GLON: {self.data.glon:.3f}\n"
            f"GLAT: {self.data.glat:.3f}\n"
            f"Position error: {self.data.pos_err:.3f}\n"
        )

    def _info_spectrum(self):
        """Print spectral information."""
        ss = "\n*** Spectral info ***\n\n"
        ss += self._info_spectrum_one(0)

        if self.n_models == 2:
            ss += self._info_spectrum_one(1)
        else:
            ss += "No second spectrum available"

        return ss


[docs]class SourceCatalogObject2HWC(SourceCatalogObjectHWCBase): """One source from the HAWC 2HWC catalog. Catalog is represented by `~gammapy.catalog.SourceCatalog2HWC`. """ @property def n_models(self): """Number of models (1 or 2).""" if hasattr(self.data, "spec1_dnde"): return 1 if np.isnan(self.data.spec1_dnde) else 2 else: return 1 def _get_idx(self, which): if which == "point": return 0 elif which == "extended": if self.n_models == 2: return 1 else: raise ValueError(f"No extended source analysis available: {self.name}") else: raise ValueError(f"Invalid which: {which!r}") def _info_spectrum_one(self, idx): d = self.data ss = f"Spectrum {idx}:\n" val, err = d[f"spec{idx}_dnde"].value, d[f"spec{idx}_dnde_err"].value ss += f"Flux at 7 TeV: {val:.3} +- {err:.3} cm-2 s-1 TeV-1\n" val, err = d[f"spec{idx}_index"], d[f"spec{idx}_index_err"] ss += f"Spectral index: {val:.3f} +- {err:.3f}\n" radius = d[f"spec{idx}_radius"] ss += f"Test Radius: {radius:1}\n\n" return ss
[docs] def spectral_model(self, which="point"): """Spectral model (`~gammapy.modeling.models.PowerLawSpectralModel`). * ``which="point"`` -- Spectral model under the point source assumption. * ``which="extended"`` -- Spectral model under the extended source assumption. Only available for some sources. Raise ValueError if not available. """ idx = self._get_idx(which) pars = { "reference": "7 TeV", "amplitude": self.data[f"spec{idx}_dnde"], "index": -self.data[f"spec{idx}_index"], } errs = { "amplitude": self.data[f"spec{idx}_dnde_err"], "index": self.data[f"spec{idx}_index_err"], } model = Model.create("PowerLawSpectralModel", "spectral", **pars) for name, value in errs.items(): model.parameters[name].error = value return model
[docs] def spatial_model(self, which="point"): """Spatial model (`~gammapy.modeling.models.SpatialModel`). * ``which="point"`` - `~gammapy.modeling.models.PointSpatialModel`. * ``which="extended"`` - `~gammapy.modeling.models.DiskSpatialModel`. Only available for some sources. Raise ValueError if not available. """ idx = self._get_idx(which) pars = {"lon_0": self.data.glon, "lat_0": self.data.glat, "frame": "galactic"} if idx == 0: tag = "PointSpatialModel" else: tag = "DiskSpatialModel" pars["r_0"] = self.data[f"spec{idx}_radius"] errs = { "lat_0": self.data.pos_err, "lon_0": self.data.pos_err / np.cos(self.data.glat), } model = Model.create(tag, "spatial", **pars) for name, value in errs.items(): model.parameters[name].error = value return model
[docs] def sky_model(self, which="point"): """Sky model (`~gammapy.modeling.models.SkyModel`). * ``which="point"`` - Sky model for point source analysis. * ``which="extended"`` - Sky model for extended source analysis. Only available for some sources. Raise ValueError if not available. According to the paper, the radius of the extended source model is only a rough estimate of the source size, based on the residual excess. """ return SkyModel( spatial_model=self.spatial_model(which), spectral_model=self.spectral_model(which), name=self.name, )
[docs]class SourceCatalog2HWC(SourceCatalog): """HAWC 2HWC catalog. One source is represented by `~gammapy.catalog.SourceCatalogObject2HWC`. The data is from tables 2 and 3 in the paper [1]_. The catalog table contains 40 rows / sources. The paper mentions 39 sources e.g. in the abstract. The difference is due to Geminga, which was detected as two "sources" by the algorithm used to make the catalog, but then in the discussion considered as one source. References ---------- .. [1] Abeysekara et al, "The 2HWC HAWC Observatory Gamma Ray Catalog", On ADS: `2017ApJ...843...40A <https://ui.adsabs.harvard.edu/abs/2017ApJ...843...40A>`__ """ tag = "2hwc" """Catalog name""" description = "2HWC catalog from the HAWC observatory" """Catalog description""" source_object_class = SourceCatalogObject2HWC def __init__(self, filename="$GAMMAPY_DATA/catalogs/2HWC.ecsv"): table = Table.read(make_path(filename), format="ascii.ecsv") source_name_key = "source_name" super().__init__(table=table, source_name_key=source_name_key)
[docs]class SourceCatalogObject3HWC(SourceCatalogObjectHWCBase): """One source from the HAWC 3HWC catalog. Catalog is represented by `~gammapy.catalog.SourceCatalog3HWC`. """ @property def n_models(self): """Number of models.""" return 1 def _info_spectrum_one(self, idx): d = self.data ss = f"Spectrum {idx}:\n" val, errn, errp = ( d[f"spec{idx}_dnde"].value, d[f"spec{idx}_dnde_errn"].value, d[f"spec{idx}_dnde_errp"].value, ) ss += f"Flux at 7 TeV: {val:.3} {errn:.3} + {errp:.3} cm-2 s-1 TeV-1\n" val, errn, errp = ( d[f"spec{idx}_index"], d[f"spec{idx}_index_errn"], d[f"spec{idx}_index_errp"], ) ss += f"Spectral index: {val:.3f} {errn:.3f} + {errp:.3f}\n" radius = d[f"spec{idx}_radius"] ss += f"Test Radius: {radius:1}\n\n" return ss @property def is_pointlike(self): return self.data["spec0_radius"] == 0.0
[docs] def spectral_model(self): """Spectral model as a `~gammapy.modeling.models.PowerLawSpectralModel` object.""" pars = { "reference": "7 TeV", "amplitude": self.data["spec0_dnde"], "index": -self.data["spec0_index"], } errs = { "index": 0.5 * (self.data["spec0_index_errp"] + np.abs(self.data["spec0_index_errn"])), "amplitude": 0.5 * (self.data["spec0_dnde_errp"] + np.abs(self.data["spec0_dnde_errn"])), } model = Model.create("PowerLawSpectralModel", "spectral", **pars) for name, value in errs.items(): model.parameters[name].error = value return model
[docs] def spatial_model(self): """Spatial model as a `~gammapy.modeling.models.SpatialModel` object.""" pars = {"lon_0": self.data.glon, "lat_0": self.data.glat, "frame": "galactic"} if self.is_pointlike: tag = "PointSpatialModel" else: tag = "DiskSpatialModel" pars["r_0"] = self.data["spec0_radius"] errs = { "lat_0": self.data.pos_err, "lon_0": self.data.pos_err / np.cos(self.data.glat), } model = Model.create(tag, "spatial", **pars) for name, value in errs.items(): model.parameters[name].error = value return model
[docs] def sky_model(self): """Sky model as a `~gammapy.modeling.models.SkyModel` object.""" return SkyModel( spatial_model=self.spatial_model(), spectral_model=self.spectral_model(), name=self.name, )
[docs]class SourceCatalog3HWC(SourceCatalog): """HAWC 3HWC catalog. One source is represented by `~gammapy.catalog.SourceCatalogObject3HWC`. The data is from tables 2 and 3 in the paper [1]_. The catalog table contains 65 rows / sources. References ---------- .. [1] 3HWC: The Third HAWC Catalog of Very-High-Energy Gamma-ray Sources", https://data.hawc-observatory.org/datasets/3hwc-survey/index.php """ tag = "3hwc" """Catalog name""" description = "3HWC catalog from the HAWC observatory" """Catalog description""" source_object_class = SourceCatalogObject3HWC def __init__(self, filename="$GAMMAPY_DATA/catalogs/3HWC.ecsv"): table = Table.read(make_path(filename), format="ascii.ecsv") source_name_key = "source_name" super().__init__(table=table, source_name_key=source_name_key)