Source code for scopesim.optics.optical_element

from inspect import isclass
from typing import TextIO
from io import StringIO

from astropy.table import Table

from .. import effects as efs
from ..effects.effects_utils import (make_effect, get_all_effects,
                                     z_order_in_range)
from ..utils import write_report, get_logger
from ..reports.rst_utils import table_to_rst
from .. import rc


logger = get_logger(__name__)


[docs]class OpticalElement: """ Contains all information to describe a section of an optical system. There are 5 major section: ``location``, ``telescope``, ``relay optics`` ``instrument``, ``detector``. An OpticalElement describes how a certain section of the optical train changes the incoming photon distribution by specifying a list of ``Effects`` along with a set of local properties e.g. temperature, etc which are common to more than one Effect Parameters ---------- yaml_dict : dict Description of optical section properties, effects, and meta-data kwargs : dict Optical Element specific information which has no connection to the effects that are passed. Any global values, e.g. airmass (i.e. bang strings) are passed on to the individual effect which can extract the relevant bang_string from the UserCommands object held in self.cmds Attributes ---------- meta : dict Contains meta data from the yaml, and is updated with the OBS_DICT key-value pairs properties : dict Contains any properties that is "global" to the optical element, e.g. instrument temperature, atmospheric pressure. Any OBS_DICT keywords are cleaned with from the ``meta`` dict during initialisation effects : list of dicts Contains the a list of dict descriptions of the effects that the optical element generates. Any OBS_DICT keywords are cleaned with from the ``meta`` dict during initialisation. """ def __init__(self, yaml_dict=None, cmds=None, **kwargs): self.meta = {"name": "<empty>"} self.meta.update(kwargs) self.properties = {} self.effects = [] self.cmds = cmds if isinstance(yaml_dict, dict): self.meta.update({key: yaml_dict[key] for key in yaml_dict if key not in {"properties", "effects"}}) if "properties" in yaml_dict: self.properties = yaml_dict["properties"] or {} if "name" in yaml_dict: self.properties["element_name"] = yaml_dict["name"] if "effects" in yaml_dict and len(yaml_dict["effects"]) > 0: for eff_dic in yaml_dict["effects"]: if "name" in eff_dic and hasattr(self.cmds, "ignore_effects"): if eff_dic["name"] in self.cmds.ignore_effects: eff_dic["include"] = False self.effects.append(make_effect(eff_dic, self.cmds, **self.properties))
[docs] def add_effect(self, effect): if isinstance(effect, efs.Effect): self.effects.append(effect) else: logger.warning("%s is not an Effect object and was not added", effect)
[docs] def get_all(self, effect_class): return get_all_effects(self.effects, effect_class)
[docs] def get_z_order_effects(self, z_level: int, z_max: int = None): """ Yield all effects in the given 100-range of `z_level`. E.g., ``z_level=200`` will yield all effect with a z_order between 200 and 299. Optionally, the upper limit can be set manually with the optional argument `z_max`. Parameters ---------- z_level : int 100-range of z_orders. z_max : int, optional Optional upper bound. This is currently not used anywhere in ScopeSim, but the functionality is tested. If None (default), this will be set to ``z_level + 99``. Raises ------ TypeError Raised if either `z_level` or `z_max` is not of int type. ValueError Raised if `z_max` (if given) is less than `z_level`. Yields ------ eff : Iterator of effects Iterator containing all effect objects in the given z_order range. """ if not isinstance(z_level, int): raise TypeError(f"z_level must be int, got {type(z_level)=}") if z_max is not None and not isinstance(z_max, int): raise TypeError(f"If given, z_max must be int, got {type(z_max)=}") z_min = z_level if z_max is not None: if z_max < z_min: raise ValueError( "z_max must be greater (or equal to) z_level, but " f"{z_max=} < {z_level=}.") else: z_max = z_min + 100 # range doesn't include final element -> 100 z_range = range(z_min, z_max) for eff in self.effects: if not eff.include or "z_order" not in eff.meta: continue if z_order_in_range(eff.meta["z_order"], z_range): yield eff
def _get_matching_effects(self, effect_classes): return (eff for eff in self.effects if isinstance(eff, effect_classes)) @property def surfaces_list(self): effect_classes = (efs.SurfaceList, efs.FilterWheel, efs.TERCurve) return list(self._get_matching_effects(effect_classes)) @property def masks_list(self): effect_classes = (efs.ApertureList, efs.ApertureMask) return list(self._get_matching_effects(effect_classes))
[docs] def list_effects(self): elements = [self.meta["name"]] * len(self.effects) names = [eff.display_name for eff in self.effects] classes = [eff.__class__.__name__ for eff in self.effects] included = [eff.meta["include"] for eff in self.effects] z_orders = [eff.meta["z_order"] for eff in self.effects] colnames = ["element", "name", "class", "included", "z_orders"] data = [elements, names, classes, included, z_orders] tbl = Table(names=colnames, data=data, copy=False) return tbl
def __add__(self, other): self.add_effect(other) def __getitem__(self, item): """ Return Effects of Effect meta properties. Parameters ---------- item : str, int, Effect-Class Either the name, list index, or Class of an effect. It is possible to get meta entries from a named effect by using: ``#<effect_name>.<meta-key>`` Returns ------- obj : str, float, Effect, list - str, float : if #-string is used to get Effect.meta entries - Effect : if a unique Effect name is given - list : if an Effect-Class is given Examples -------- :: from scopesim.effect import TERCurve opt_el = opt_elem.OpticalElement(detector_yaml_dict) effect_list = opt_el[TERCurve] effect = opt_el[0] effect = opt_el["detector_qe_curve"] meta_value = opt_el["#detector_qe_curve.filename"] """ obj = None if isclass(item): obj = self.get_all(item) elif isinstance(item, int): obj = self.effects[item] elif isinstance(item, str): if item.startswith("#") and "." in item: eff, meta = item.replace("#", "").split(".") obj = self[eff][f"#{meta}"] else: obj = [eff for eff in self.effects if eff.meta["name"] == item] if isinstance(obj, list) and len(obj) == 1: obj = obj[0] # if obj is None or len(obj) == 0: # logger.warning( # "No result for key: '%s'. Did you mean '#%s'?", item, item) return obj
[docs] def write_string(self, stream: TextIO, list_effects: bool = True) -> None: """Write formatted string representation to I/O stream.""" stream.write(f"{self!s} contains {len(self.effects)} Effects\n") if list_effects: for i_eff, eff in enumerate(self.effects): stream.write(f"[{i_eff}] {eff!r}\n")
[docs] def pretty_str(self) -> str: """Return formatted string representation as str.""" with StringIO() as str_stream: self.write_string(str_stream) output = str_stream.getvalue() return output
@property def display_name(self): return self.meta.get("name", self.meta.get("filename", "<empty>")) def __repr__(self): return f"<{self.__class__.__name__}>" def __str__(self): return f"{self.__class__.__name__}: \"{self.display_name}\"" def _repr_pretty_(self, p, cycle): """For ipython.""" if cycle: p.text(f"{self.__class__.__name__}(...)") else: p.text(str(self)) @property def properties_str(self): # TODO: This seems to be used only in the report below. # Once the report uses stream writing, change this to a function # that simply write to that same stream... padlen = max(len(key) for key in self.properties) + 4 exclude = {"comments", "changes", "description", "history", "report"} with StringIO() as str_stream: for key in self.properties.keys() - exclude: str_stream.write(f"{key:>{padlen}} : {self.properties[key]}\n") output = str_stream.getvalue() return output
[docs] def report(self, filename=None, output="rst", rst_title_chars="^#*+", **kwargs): rst_str = f""" {str(self)} {rst_title_chars[0] * len(str(self))} **Element**: {self.meta.get("object", "<unknown optical element>")} **Alias**: {self.meta.get("alias", "<unknown alias>")} **Description**: {self.meta.get("description", "<no description>")} Global properties {rst_title_chars[1] * 17} :: {self.properties_str} """ if len(self.list_effects()) > 0: rst_str += f""" Effects {rst_title_chars[1] * 7} Summary of Effects included in this optical element: .. table:: :name: {"tbl:" + self.meta.get("name", "<unknown OpticalElement>")} {table_to_rst(self.list_effects(), indent=4)} """ reports = [eff.report(rst_title_chars=rst_title_chars[-2:], **kwargs) for eff in self.effects] rst_str += "\n\n" + "\n\n".join(reports) write_report(rst_str, filename, output) return rst_str