Loading [MathJax]/extensions/tex2jax.js
LSST Applications g04dff08e69+fafbcb10e2,g0d33ba9806+e09a96fa4e,g0fba68d861+cc01b48236,g1e78f5e6d3+fb95f9dda6,g1ec0fe41b4+f536777771,g1fd858c14a+ae46bc2a71,g35bb328faa+fcb1d3bbc8,g4af146b050+dd94f3aad7,g4d2262a081+7ee6f976aa,g53246c7159+fcb1d3bbc8,g5a012ec0e7+b20b785ecb,g60b5630c4e+e09a96fa4e,g6273192d42+bf8cfc5e62,g67b6fd64d1+4086c0989b,g78460c75b0+2f9a1b4bcd,g786e29fd12+cf7ec2a62a,g7b71ed6315+fcb1d3bbc8,g87b7deb4dc+831c06c8fc,g8852436030+54b48a5987,g89139ef638+4086c0989b,g9125e01d80+fcb1d3bbc8,g94187f82dc+e09a96fa4e,g989de1cb63+4086c0989b,g9f33ca652e+64be6d9d51,g9f7030ddb1+d11454dffd,ga2b97cdc51+e09a96fa4e,gabe3b4be73+1e0a283bba,gabf8522325+fa80ff7197,gb1101e3267+23605820ec,gb58c049af0+f03b321e39,gb89ab40317+4086c0989b,gcf25f946ba+54b48a5987,gd6cbbdb0b4+af3c3595f5,gd9a9a58781+fcb1d3bbc8,gde0f65d7ad+15f2daff9d,ge278dab8ac+d65b3c2b70,ge410e46f29+4086c0989b,gf67bdafdda+4086c0989b,v29.0.0.rc5
LSST Data Management Base Package
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
rawFormatter.py
Go to the documentation of this file.
1# This file is part of obs_decam.
2#
3# Developed for the LSST Data Management System.
4# This product includes software developed by the LSST Project
5# (http://www.lsst.org).
6# See the COPYRIGHT file at the top-level directory of this distribution
7# for details of code ownership.
8#
9# This program is free software: you can redistribute it and/or modify
10# it under the terms of the GNU General Public License as published by
11# the Free Software Foundation, either version 3 of the License, or
12# (at your option) any later version.
13#
14# This program is distributed in the hope that it will be useful,
15# but WITHOUT ANY WARRANTY; without even the implied warranty of
16# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17# GNU General Public License for more details.
18#
19# You should have received a copy of the GNU General Public License
20# along with this program. If not, see <http://www.gnu.org/licenses/>.
21
22"""Gen3 Butler Formatters for Dark Energy Camera raw data.
23"""
24
25import logging
26import astro_metadata_translator
27
28import lsst.afw.fits
29import lsst.afw.image
30from lsst.obs.base import FitsRawFormatterBase
31
32from . import DarkEnergyCamera
33
34__all__ = ("DarkEnergyCameraRawFormatter", "DarkEnergyCameraCPCalibFormatter")
35
36
37# The mapping of detector id to HDU in raw files for "most" DECam data.
38# We try this first before scaning the HDUs manually.
39detector_to_hdu = {25: 1, 26: 2, 27: 3, 32: 4, 33: 5, 34: 6, 19: 7, 20: 8, 13: 9,
40 14: 10, 8: 11, 4: 12, 39: 13, 40: 14, 45: 15, 46: 16, 51: 17, 56: 18, 21: 19,
41 22: 20, 23: 21, 24: 22, 17: 23, 18: 24, 15: 25, 16: 26, 9: 27, 10: 28, 11: 29,
42 12: 30, 5: 31, 6: 32, 7: 33, 1: 34, 2: 35, 3: 36, 35: 37, 36: 38, 37: 39,
43 38: 40, 28: 41, 29: 42, 30: 43, 31: 44, 41: 45, 42: 46, 43: 47, 44: 48, 49: 49,
44 50: 50, 47: 51, 48: 52, 52: 53, 53: 54, 54: 55, 55: 56, 57: 57, 58: 58, 59: 59,
45 60: 60, 61: 61, 62: 62, 72: 63, 71: 64, 64: 65, 63: 66, 73: 67, 74: 68, 70: 69,
46 69: 70
47 }
48
49
50class DarkEnergyCameraRawFormatter(FitsRawFormatterBase):
51 translatorClass = astro_metadata_translator.DecamTranslator
52 filterDefinitions = DarkEnergyCamera.filterDefinitions
53
54 # DECam has a coordinate system flipped on X with respect to our
55 # VisitInfo definition of the field angle orientation.
56 # We have to specify the flip explicitly until DM-20746 is implemented.
57 wcsFlipX = True
58
59 def getDetector(self, id):
60 return DarkEnergyCamera().getCamera()[id]
61
62 def _scanHdus(self, filename, detectorId):
63 """Scan through a file for the HDU containing data from one detector.
64
65 Parameters
66 ----------
67 filename : `str`
68 The file to search through.
69 detectorId : `int`
70 The detector id to search for.
71
72 Returns
73 -------
74 index : `int`
75 The index of the HDU with the requested data.
76 metadata: `lsst.daf.base.PropertyList`
77 The metadata read from the header for that detector id.
78
79 Raises
80 ------
81 ValueError
82 Raised if detectorId is not found in any of the file HDUs
83 """
84 log = logging.getLogger("lsst.obs.decam.DarkEnergyCameraRawFormatter")
85 log.debug("Did not find detector=%s at expected HDU=%s in %s: scanning through all HDUs.",
86 detectorId, detector_to_hdu[detectorId], filename)
87
88 fitsData = lsst.afw.fits.Fits(filename, 'r')
89 # NOTE: The primary header (HDU=0) does not contain detector data.
90 for i in range(1, fitsData.countHdus()):
91 fitsData.setHdu(i)
92 metadata = fitsData.readMetadata()
93 if metadata['CCDNUM'] == detectorId:
94 return i, metadata
95 else:
96 raise ValueError(f"Did not find detectorId={detectorId} as CCDNUM in any HDU of {filename}.")
97
98 def _determineHDU(self, detectorId):
99 """Determine the correct HDU number for a given detector id.
100
101 Parameters
102 ----------
103 detectorId : `int`
104 The detector id to search for.
105
106 Returns
107 -------
108 index : `int`
109 The index of the HDU with the requested data.
110 metadata : `lsst.daf.base.PropertyList`
111 The metadata read from the header for that detector id.
112
113 Raises
114 ------
115 ValueError
116 Raised if detectorId is not found in any of the file HDUs
117 """
118 filename = self._reader_path
119 try:
120 index = detector_to_hdu[detectorId]
121 metadata = lsst.afw.fits.readMetadata(filename, index)
122 if metadata['CCDNUM'] != detectorId:
123 # detector->HDU mapping is different in this file: try scanning
124 return self._scanHdus(filename, detectorId)
125 else:
126 fitsData = lsst.afw.fits.Fits(filename, 'r')
127 fitsData.setHdu(index)
128 return index, metadata
130 # If the file doesn't contain all the HDUs of "normal" files,
131 # try scanning.
132 return self._scanHdus(filename, detectorId)
133
134 def readMetadata(self):
135 index, metadata = self._determineHDU(self.data_id['detector'])
136 astro_metadata_translator.fix_header(metadata)
137 return metadata
138
139 def readImage(self):
140 index, metadata = self._determineHDU(self.data_id['detector'])
141 return lsst.afw.image.ImageI(self._reader_path, index)
142
143
145 """DECam Community Pipeline calibrations (bias, dark, flat, fringe) are
146 multi-extension FITS files with detector=index+1.
147 """
148
149 def _determineHDU(self, detectorId):
150 """The HDU to read is the same as the detector number."""
151 filename = self._reader_path
152 metadata = lsst.afw.fits.readMetadata(filename, detectorId)
153 if metadata['CCDNUM'] != detectorId:
154 msg = f"Found CCDNUM={metadata['CCDNUM']} instead of {detectorId} in {filename} HDU={detectorId}."
155 raise ValueError(msg)
156 return detectorId, metadata
157
158 def readImage(self):
159 index, metadata = self._determineHDU(self.data_id['detector'])
160 return lsst.afw.image.ImageF(self._reader_path, index)
An exception thrown when problems are found when reading or writing FITS files.
Definition fits.h:36
A simple struct that combines the two arguments that must be passed to most cfitsio routines and cont...
Definition fits.h:308
std::shared_ptr< daf::base::PropertyList > readMetadata(std::string const &fileName, int hdu=DEFAULT_HDU, bool strip=false)
Read FITS header.
Definition fits.cc:1696