LSST Applications g180d380827+0f66a164bb,g2079a07aa2+86d27d4dc4,g2305ad1205+7d304bc7a0,g29320951ab+500695df56,g2bbee38e9b+0e5473021a,g337abbeb29+0e5473021a,g33d1c0ed96+0e5473021a,g3a166c0a6a+0e5473021a,g3ddfee87b4+e42ea45bea,g48712c4677+36a86eeaa5,g487adcacf7+2dd8f347ac,g50ff169b8f+96c6868917,g52b1c1532d+585e252eca,g591dd9f2cf+c70619cc9d,g5a732f18d5+53520f316c,g5ea96fc03c+341ea1ce94,g64a986408d+f7cd9c7162,g858d7b2824+f7cd9c7162,g8a8a8dda67+585e252eca,g99cad8db69+469ab8c039,g9ddcbc5298+9a081db1e4,ga1e77700b3+15fc3df1f7,gb0e22166c9+60f28cb32d,gba4ed39666+c2a2e4ac27,gbb8dafda3b+c92fc63c7e,gbd866b1f37+f7cd9c7162,gc120e1dc64+02c66aa596,gc28159a63d+0e5473021a,gc3e9b769f7+b0068a2d9f,gcf0d15dbbd+e42ea45bea,gdaeeff99f8+f9a426f77a,ge6526c86ff+84383d05b3,ge79ae78c31+0e5473021a,gee10cc3b42+585e252eca,gff1a9f87cc+f7cd9c7162,w.2024.17
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