Skip to content

Commit

Permalink
Add reference_location to SubarrayDescription
Browse files Browse the repository at this point in the history
  • Loading branch information
maxnoe committed Jun 30, 2022
1 parent 29e350c commit 12ea37c
Show file tree
Hide file tree
Showing 2 changed files with 70 additions and 22 deletions.
76 changes: 55 additions & 21 deletions ctapipe/instrument/subarray.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
from copy import copy
from itertools import groupby
from typing import Dict, List, Union
from astropy.coordinates.earth import EarthLocation

import numpy as np
import tables
Expand Down Expand Up @@ -51,16 +52,6 @@ class SubarrayDescription:
Collects the `~ctapipe.instrument.TelescopeDescription` of all telescopes
along with their positions on the ground.
Parameters
----------
name: str
name of this subarray
tel_positions: Dict[Array]
dict of x,y,z telescope positions on the ground by tel_id. These are
converted internally to a coordinate in the `~ctapipe.coordinates.GroundFrame`
tel_descriptions: Dict[TelescopeDescription]
dict of TelescopeDescriptions by tel_id
Attributes
----------
name: str
Expand All @@ -69,16 +60,35 @@ class SubarrayDescription:
coordinates of all telescopes
tels:
dict of TelescopeDescription for each telescope in the subarray
tel_ids: np.ndarray
array of tel_ids
tel_indices: dict
dict mapping tel_id to index in array attributes
"""

def __init__(self, name, tel_positions=None, tel_descriptions=None):
def __init__(
self,
name,
tel_positions=None,
tel_descriptions=None,
reference_location: EarthLocation = None,
):
"""
Initialize a new SubarrayDescription
Parameters
----------
name : str
name of this subarray
tel_positions : Dict[int, np.ndarray]
dict of x,y,z telescope positions on the ground by tel_id. These are
converted internally to a coordinate in the `~ctapipe.coordinates.GroundFrame`
tel_descriptions : Dict[TelescopeDescription]
dict of TelescopeDescriptions by tel_id
reference_location : `astropy.coordinates.EarthLocation`
EarthLocation of the array reference position, (0, 0, 0) of the
coordinate system used for `tel_positions`.
"""
self.name = name
self.positions = tel_positions or dict()
self.tels: Dict[int, TelescopeDescription] = tel_descriptions or dict()
self.reference_location = reference_location

if self.positions.keys() != self.tels.keys():
raise ValueError("Telescope ids in positions and descriptions do not match")
Expand All @@ -92,8 +102,8 @@ def __repr__(self):
)

@property
def tel(self):
"""for backward compatibility"""
def tel(self) -> Dict[int, TelescopeDescription]:
"""Dictionary mapping tel_ids to TelescopeDescriptions"""
return self.tels

@property
Expand Down Expand Up @@ -238,7 +248,7 @@ def to_table(self, kind="subarray"):
which table to generate (subarray or optics)
"""

if kind == 'joined':
if kind == "joined":
table = self.to_table()
optics = self.to_table(kind="optics")
optics["optics_index"] = np.arange(len(optics))
Expand All @@ -252,17 +262,22 @@ def to_table(self, kind="subarray"):
)

table.remove_columns(["optics_index", "camera_index"])
table.add_index('tel_id')
table.add_index("tel_id")
return table


meta = {
"ORIGIN": "ctapipe.instrument.SubarrayDescription",
"SUBARRAY": self.name,
"SOFT_VER": CTAPIPE_VERSION,
"TAB_TYPE": kind,
}

if self.reference_location is not None:
itrs = self.reference_location.itrs
meta["OBSGEO-X"] = itrs.x.to_value(u.m)
meta["OBSGEO-Y"] = itrs.y.to_value(u.m)
meta["OBSGEO-Z"] = itrs.z.to_value(u.m)

if kind == "subarray":

unique_types = self.telescope_types
Expand Down Expand Up @@ -447,13 +462,17 @@ def __eq__(self, other):
if self.positions.keys() != other.positions.keys():
return False

if self.reference_location != other.reference_location:
return False

for tel_id in self.tels.keys():
if self.tels[tel_id] != other.tels[tel_id]:
return False

for tel_id in self.tels.keys():
if np.any(self.positions[tel_id] != other.positions[tel_id]):
return False

return True

def to_hdf(self, h5file, overwrite=False, mode="a"):
Expand Down Expand Up @@ -510,7 +529,13 @@ def to_hdf(self, h5file, overwrite=False, mode="a"):
overwrite=overwrite,
)

h5file.root.configuration.instrument.subarray._v_attrs.name = self.name
meta = h5file.root.configuration.instrument.subarray._v_attrs
meta["name"] = self.name
if self.reference_location is not None:
itrs = self.reference_location.itrs
meta["reference_itrs_x"] = itrs.x.to_value(u.m)
meta["reference_itrs_y"] = itrs.y.to_value(u.m)
meta["reference_itrs_z"] = itrs.z.to_value(u.m)

@classmethod
def from_hdf(cls, path):
Expand Down Expand Up @@ -588,6 +613,7 @@ def from_hdf(cls, path):
positions = np.column_stack([layout[f"pos_{c}"] for c in "xyz"])

name = "Unknown"
reference_location = None
with ExitStack() as stack:
if not isinstance(path, tables.File):
path = stack.enter_context(tables.open_file(path, mode="r"))
Expand All @@ -596,10 +622,18 @@ def from_hdf(cls, path):
if "name" in attrs:
name = str(attrs.name)

if "reference_itrs_x" in attrs:
reference_location = EarthLocation(
x=attrs["reference_itrs_x"] * u.m,
y=attrs["reference_itrs_y"] * u.m,
z=attrs["reference_itrs_z"] * u.m,
)

return cls(
name=name,
tel_positions={
tel_id: pos for tel_id, pos in zip(layout["tel_id"], positions)
},
tel_descriptions=telescope_descriptions,
reference_location=reference_location,
)
16 changes: 15 additions & 1 deletion ctapipe/instrument/tests/test_subarray.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
""" Tests for SubarrayDescriptions """
from copy import deepcopy
from astropy.coordinates.earth import EarthLocation

import numpy as np
from astropy import units as u
Expand Down Expand Up @@ -28,7 +29,16 @@ def example_subarray(n_tels=10):
)
pos[tel_id] = rng.uniform(-100, 100, size=3) * u.m

return SubarrayDescription("test array", tel_positions=pos, tel_descriptions=tel)
return SubarrayDescription(
"test array",
tel_positions=pos,
tel_descriptions=tel,
reference_location=EarthLocation(
lon=-17 * u.deg,
lat=28 * u.deg,
height=2200 * u.m,
),
)


def test_subarray_description():
Expand Down Expand Up @@ -225,3 +235,7 @@ def test_unknown_telescopes(example_subarray):

with pytest.raises(UnknownTelescopeID):
example_subarray.select_subarray([300, 201])


def test_reference_location():
from ctapipe.instrument import SubarrayDescription

0 comments on commit 12ea37c

Please sign in to comment.