From 3bba90ca0b07134d830329284620421d6e4190e0 Mon Sep 17 00:00:00 2001 From: teald Date: Thu, 15 Aug 2024 16:40:54 -0700 Subject: [PATCH 1/6] =?UTF-8?q?docs(quickstart):=20=F0=9F=9A=A7=20Add=20fi?= =?UTF-8?q?rst=20draft=20of=20tags=20exmaple.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- docs/quickstart.rst | 103 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 103 insertions(+) diff --git a/docs/quickstart.rst b/docs/quickstart.rst index 6de49ce1..17749d40 100644 --- a/docs/quickstart.rst +++ b/docs/quickstart.rst @@ -336,6 +336,109 @@ get all descriptors from a class using the ``.descriptors`` attribute. You'll see that our ``airmass`` descriptor is available for the ``GMOSScienceAstroData`` class, but not for the ``GMOSAstroData`` class. +Tags +==== + +Tags are a way to categorize data in |astrodata|. They are meant to be +used to identify the type of data in a file, and can be used to filter +data when opening files. + +Tags are stored in the ``.tags`` attribute of an |AstroData| object. You +can add tags to a class by using the ``@astro_data_tag`` decorator on methods +that output tags. For example, we want to tag our ``GMOSScienceAstroData`` +class as ``'GMOS'`` and ``'SCIENCE'``: + +.. code-block:: python + + class GMOSScienceAstroDataTagged(GMOSScienceAstroData): + """A class for GMOS science data with tags. + + Note: This still has all the methods from GMOSScienceAstroData! It is a + subset of the tags used for the GMOS instrument |AstroData| class. + """ + + @astro_data_tag + def _tag_instrument(self): + # tags = ['GMOS', self.instrument().upper().replace('-', '_')] + return TagSet(["GMOS"]) + + @astro_data_tag + def _tag_dark(self): + if self.phu.get("OBSTYPE") == "DARK": + return TagSet(["DARK", "CAL"], blocks=["IMAGE", "SPECT"]) + + @astro_data_tag + def _tag_arc(self): + if self.phu.get("OBSTYPE") == "ARC": + return TagSet(["ARC", "CAL"]) + + def _tag_is_bias(self): + if self.phu.get("OBSTYPE") == "BIAS": + return True + else: + return False + + def _tag_is_bpm(self): + if self.phu.get("OBSTYPE") == "BPM" or "BPMASK" in self.phu: + return True + else: + return False + + @astro_data_tag + def _tag_bias(self): + if self._tag_is_bias(): + return TagSet(["BIAS", "CAL"], blocks=["IMAGE", "SPECT"]) + + @astro_data_tag + def _tag_flat(self): + if self.phu.get("OBSTYPE") == "FLAT": + if self.phu.get("GRATING") == "MIRROR": + f1, f2 = self.phu.get("FILTER1"), self.phu.get("FILTER2") + # This kind of filter prevents imaging to be classified as FLAT + if any(("Hartmann" in f) for f in (f1, f2)): + return None + return TagSet(["GCALFLAT", "FLAT", "CAL"]) + + @astro_data_tag + def _tag_twilight(self): + if self.phu.get("OBJECT", "").upper() == "TWILIGHT": + # Twilight flats are of OBSTYPE == OBJECT, meaning that the generic + # FLAT tag won't be triggered. Add it explicitly. + return TagSet( + [ + "TWILIGHT", + "CAL", + "SLITILLUM" if self._tag_is_spect() else "FLAT", + ], + ) + + @astro_data_tag + def _tag_image_or_spect(self): + if self.phu.get('GRATING') == 'MIRROR': + return TagSet(['IMAGE']) + else: + return TagSet(['SPECT']) + + factory.add_class(GMOSScienceAstroDataTagged) + + +These tags were taken from the |DRAGONS| GMOS package, and exemplify some +basic and more complex tag usage in |astrodata|. For example, the ``_tag_dark`` +method will tag the data as ``'DARK'`` and ``'CAL'`` if the ``OBSTYPE`` is +``'DARK'``. The ``blocks`` argument is used to specify that the tags will +"block" other tags from being applied to the data, in this case the ``'IMAGE'`` +and ``'SPECT'`` tags. + +Let's make new |AstroData| objects for our files and see what tags they have: + +.. code-block:: python + + for f in files: + ad = astrodata.from_file(f'quickstart_data/{f}') + print(f"Opened {ad.filename} with class {ad.__class__}") + print(f"Tags: {ad.tags}") + + Advanced Usage ============== From 5a8ad41a7a9db5d68ac6c546ac6c17de6fd6184f Mon Sep 17 00:00:00 2001 From: teald Date: Thu, 15 Aug 2024 16:52:34 -0700 Subject: [PATCH 2/6] =?UTF-8?q?docs(quickstart):=20=F0=9F=9A=A7=20Tweak=20?= =?UTF-8?q?example=20classes.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- docs/quickstart.rst | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/docs/quickstart.rst b/docs/quickstart.rst index 17749d40..da5395fb 100644 --- a/docs/quickstart.rst +++ b/docs/quickstart.rst @@ -350,7 +350,10 @@ class as ``'GMOS'`` and ``'SCIENCE'``: .. code-block:: python - class GMOSScienceAstroDataTagged(GMOSScienceAstroData): + # Let's remove the GMOSScienceAstroData class from the factory to avoid conflicts. + factory.remove_class(GMOSScienceAstroData) + + class GMOSAstroDataTagged(GMOSAstroData): """A class for GMOS science data with tags. Note: This still has all the methods from GMOSScienceAstroData! It is a @@ -419,7 +422,7 @@ class as ``'GMOS'`` and ``'SCIENCE'``: else: return TagSet(['SPECT']) - factory.add_class(GMOSScienceAstroDataTagged) + factory.add_class(GMOSAstroDataTagged) These tags were taken from the |DRAGONS| GMOS package, and exemplify some From 0434145b8ec5b1252b29807b516affbac3b02ef0 Mon Sep 17 00:00:00 2001 From: teald Date: Fri, 16 Aug 2024 17:05:33 -0700 Subject: [PATCH 3/6] =?UTF-8?q?docs(quickstart):=20=E2=9C=8D=EF=B8=8F=20Fi?= =?UTF-8?q?rst=20full=20draft=20of=20quickstart.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- docs/quickstart.rst | 105 ++++++++++++++++++++++++++++++++++++++------ 1 file changed, 92 insertions(+), 13 deletions(-) diff --git a/docs/quickstart.rst b/docs/quickstart.rst index da5395fb..6608bd40 100644 --- a/docs/quickstart.rst +++ b/docs/quickstart.rst @@ -353,6 +353,8 @@ class as ``'GMOS'`` and ``'SCIENCE'``: # Let's remove the GMOSScienceAstroData class from the factory to avoid conflicts. factory.remove_class(GMOSScienceAstroData) + from astrodata import astro_data_tag, TagSet + class GMOSAstroDataTagged(GMOSAstroData): """A class for GMOS science data with tags. @@ -375,18 +377,6 @@ class as ``'GMOS'`` and ``'SCIENCE'``: if self.phu.get("OBSTYPE") == "ARC": return TagSet(["ARC", "CAL"]) - def _tag_is_bias(self): - if self.phu.get("OBSTYPE") == "BIAS": - return True - else: - return False - - def _tag_is_bpm(self): - if self.phu.get("OBSTYPE") == "BPM" or "BPMASK" in self.phu: - return True - else: - return False - @astro_data_tag def _tag_bias(self): if self._tag_is_bias(): @@ -422,6 +412,30 @@ class as ``'GMOS'`` and ``'SCIENCE'``: else: return TagSet(['SPECT']) + def _tag_is_bias(self): + if self.phu.get("OBSTYPE") == "BIAS": + return True + else: + return False + + def _tag_is_bpm(self): + if self.phu.get("OBSTYPE") == "BPM" or "BPMASK" in self.phu: + return True + else: + return False + + def _tag_is_spect(self): + pairs = ( + ('MASKTYP', 0), + ('MASKNAME', 'None'), + ('GRATING', 'MIRROR') + ) + + matches = (self.phu.get(kw) == value for (kw, value) in pairs) + if any(matches): + return False + return True + factory.add_class(GMOSAstroDataTagged) @@ -436,11 +450,76 @@ Let's make new |AstroData| objects for our files and see what tags they have: .. code-block:: python + all_ad_data = [] + for f in files: - ad = astrodata.from_file(f'quickstart_data/{f}') + location = f"quickstart_data/{f}" + ad = astrodata.from_file(location) print(f"Opened {ad.filename} with class {ad.__class__}") print(f"Tags: {ad.tags}") + all_ad_data.append(ad) + +The result: + +.. code-block:: text + + Opened N20170614S0201.fits with class + Tags: {'IMAGE', 'GMOS'} + Opened N20170614S0202.fits with class + Tags: {'IMAGE', 'GMOS'} + Opened N20170614S0203.fits with class + Tags: {'IMAGE', 'GMOS'} + Opened N20170614S0204.fits with class + Tags: {'IMAGE', 'GMOS'} + Opened N20170614S0205.fits with class + Tags: {'IMAGE', 'GMOS'} + Opened N20170613S0180.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170613S0181.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170613S0182.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170613S0183.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170613S0184.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170615S0534.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170615S0535.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170615S0536.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170615S0537.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170615S0538.fits with class + Tags: {'BIAS', 'CAL', 'GMOS'} + Opened N20170702S0178.fits with class + Tags: {'FLAT', 'GMOS', 'TWILIGHT', 'IMAGE', 'CAL'} + Opened N20170702S0179.fits with class + Tags: {'FLAT', 'GMOS', 'TWILIGHT', 'IMAGE', 'CAL'} + Opened N20170702S0180.fits with class + Tags: {'FLAT', 'GMOS', 'TWILIGHT', 'IMAGE', 'CAL'} + Opened N20170702S0181.fits with class + Tags: {'FLAT', 'GMOS', 'TWILIGHT', 'IMAGE', 'CAL'} + Opened N20170702S0182.fits with class + Tags: {'FLAT', 'GMOS', 'TWILIGHT', 'IMAGE', 'CAL'} + Opened bpm_20170306_gmos-n_Ham_22_full_12amp.fits with class + Tags: {'SPECT', 'GMOS'} + + +Now, our data is automatically tagged with the appropriate tags when we open +the files, and we can use these tags to filter data when manipulating files. + +.. code-block:: python + + # Filter out all the bias frames. + for ad in all_ad_data: + if 'BIAS' in ad.tags: + print(f"{ad.filename} is a bias frame.") + +There are many more features available in |astrodata|, but this should give +you a good starting point for working with your own data. Advanced Usage ============== From 1d038270341d699c81abd2bcd9cb817d45dadde7 Mon Sep 17 00:00:00 2001 From: teald Date: Mon, 19 Aug 2024 09:50:30 -0700 Subject: [PATCH 4/6] =?UTF-8?q?docs(quickstart):=20=E2=9C=8D=EF=B8=8F=20Ad?= =?UTF-8?q?d=20table=20with=20tag=20args?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- docs/quickstart.rst | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/docs/quickstart.rst b/docs/quickstart.rst index 6608bd40..c8dee4d7 100644 --- a/docs/quickstart.rst +++ b/docs/quickstart.rst @@ -446,6 +446,23 @@ method will tag the data as ``'DARK'`` and ``'CAL'`` if the ``OBSTYPE`` is "block" other tags from being applied to the data, in this case the ``'IMAGE'`` and ``'SPECT'`` tags. ++------------------+----------------------------------------------------+ +| Tag group | Description | ++==================+====================================================+ +| ``add`` | Add the tag to the global set. | ++------------------+----------------------------------------------------+ +| ``remove`` | Remove the tag from the global set. | ++------------------+----------------------------------------------------+ +| ``blocked_by`` | Tags that, if present, will prevent the TagSet | +| | from being added. | ++------------------+----------------------------------------------------+ +| ``blocks`` | *Other* ``TagSet`` s containing these tags will | +| | not be added to the global set. | ++------------------+----------------------------------------------------+ +| ``if_present`` | This ``TagSet`` will only be added if the | +| | specified tags are present. | ++------------------+----------------------------------------------------+ + Let's make new |AstroData| objects for our files and see what tags they have: .. code-block:: python From 73f64dc5935378b1106f5010fa6e77a0eae5e7f5 Mon Sep 17 00:00:00 2001 From: teald Date: Mon, 19 Aug 2024 10:00:26 -0700 Subject: [PATCH 5/6] =?UTF-8?q?docs(nox):=20=E2=9C=A8=20Add=20link=20to=20?= =?UTF-8?q?end=20of=20docs=20session.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- noxfile.py | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/noxfile.py b/noxfile.py index 7021273f..daeb8daf 100644 --- a/noxfile.py +++ b/noxfile.py @@ -671,9 +671,17 @@ def docs(session: nox.Session) -> None: # Build the documentation. # TODO(teald): Add nitpicky flag to fix warnings. -- Issue #42 - target = Path("_build") + target = Path("_build").absolute() session.run("rm", "-rf", target, external=True) session.run("sphinx-build", "-W", "docs", target) + session.log(f"You can find the documentation at: {str(target)}") + + index_loc = target / "index.html" + + session.log( + f"Your index can be opened in a browser with:\n" + f"{index_loc.absolute().as_uri()}" + ) def use_devpi_server(func): From 008210568c42c3f7a66092e636fa54fc9713d4f2 Mon Sep 17 00:00:00 2001 From: teald Date: Mon, 19 Aug 2024 10:01:06 -0700 Subject: [PATCH 6/6] =?UTF-8?q?docs:=20=E2=9C=8D=EF=B8=8F=20Add=20clarifyi?= =?UTF-8?q?ng=20comments=20to=20examples.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- astrodata/utils.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/astrodata/utils.py b/astrodata/utils.py index 49d59f8f..cd4d5194 100644 --- a/astrodata/utils.py +++ b/astrodata/utils.py @@ -142,7 +142,7 @@ class TagSet(namedtuple("TagSet", "add remove blocked_by blocks if_present")): ) >>> TagSet({'BIAS', 'CAL'}) # doctest: +SKIP TagSet( - add={'BIAS', 'CAL'}, + add={'BIAS', 'CAL'}, # These tags are added to the global set remove=set(), blocked_by=set(), blocks=set(), @@ -151,7 +151,7 @@ class TagSet(namedtuple("TagSet", "add remove blocked_by blocks if_present")): >>> TagSet(remove={'BIAS', 'CAL'}) # doctest: +SKIP TagSet( add=set(), - remove={'BIAS', 'CAL'}, + remove={'BIAS', 'CAL'}, # These tags are removed from the global set blocked_by=set(), blocks=set(), if_present=set()