LSST Applications 27.0.0,g0265f82a02+469cd937ee,g02d81e74bb+21ad69e7e1,g1470d8bcf6+cbe83ee85a,g2079a07aa2+e67c6346a6,g212a7c68fe+04a9158687,g2305ad1205+94392ce272,g295015adf3+81dd352a9d,g2bbee38e9b+469cd937ee,g337abbeb29+469cd937ee,g3939d97d7f+72a9f7b576,g487adcacf7+71499e7cba,g50ff169b8f+5929b3527e,g52b1c1532d+a6fc98d2e7,g591dd9f2cf+df404f777f,g5a732f18d5+be83d3ecdb,g64a986408d+21ad69e7e1,g858d7b2824+21ad69e7e1,g8a8a8dda67+a6fc98d2e7,g99cad8db69+f62e5b0af5,g9ddcbc5298+d4bad12328,ga1e77700b3+9c366c4306,ga8c6da7877+71e4819109,gb0e22166c9+25ba2f69a1,gb6a65358fc+469cd937ee,gbb8dafda3b+69d3c0e320,gc07e1c2157+a98bf949bb,gc120e1dc64+615ec43309,gc28159a63d+469cd937ee,gcf0d15dbbd+72a9f7b576,gdaeeff99f8+a38ce5ea23,ge6526c86ff+3a7c1ac5f1,ge79ae78c31+469cd937ee,gee10cc3b42+a6fc98d2e7,gf1cff7945b+21ad69e7e1,gfbcc870c63+9a11dc8c8f
LSST Data Management Base Package
Loading...
Searching...
No Matches
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.fileDescriptor.location.path
119 try:
120 index = detector_to_hdu[detectorId]
121 metadata = lsst.afw.fits.readMetadata(filename, index)
122 if metadata['CCDNUM'] != detectorId:
123 # the 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, try scanning
131 return self._scanHdus(filename, detectorId)
132
133 def readMetadata(self):
134 index, metadata = self._determineHDU(self.dataId['detector'])
135 astro_metadata_translator.fix_header(metadata)
136 return metadata
137
138 def readImage(self):
139 index, metadata = self._determineHDU(self.dataId['detector'])
140 return lsst.afw.image.ImageI(self.fileDescriptor.location.path, index)
141
142
144 """DECam Community Pipeline calibrations (bias, dark, flat, fringe) are
145 multi-extension FITS files with detector=index+1.
146 """
147
148 def _determineHDU(self, detectorId):
149 """The HDU to read is the same as the detector number."""
150 filename = self.fileDescriptor.location.path
151 metadata = lsst.afw.fits.readMetadata(filename, detectorId)
152 if metadata['CCDNUM'] != detectorId:
153 msg = f"Found CCDNUM={metadata['CCDNUM']} instead of {detectorId} in {filename} HDU={detectorId}."
154 raise ValueError(msg)
155 return detectorId, metadata
156
157 def readImage(self):
158 index, metadata = self._determineHDU_determineHDU(self.dataId['detector'])
159 return lsst.afw.image.ImageF(self.fileDescriptor.location.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:1689