LSST Applications g063fba187b+cac8b7c890,g0f08755f38+6aee506743,g1653933729+a8ce1bb630,g168dd56ebc+a8ce1bb630,g1a2382251a+b4475c5878,g1dcb35cd9c+8f9bc1652e,g20f6ffc8e0+6aee506743,g217e2c1bcf+73dee94bd0,g28da252d5a+1f19c529b9,g2bbee38e9b+3f2625acfc,g2bc492864f+3f2625acfc,g3156d2b45e+6e55a43351,g32e5bea42b+1bb94961c2,g347aa1857d+3f2625acfc,g35bb328faa+a8ce1bb630,g3a166c0a6a+3f2625acfc,g3e281a1b8c+c5dd892a6c,g3e8969e208+a8ce1bb630,g414038480c+5927e1bc1e,g41af890bb2+8a9e676b2a,g7af13505b9+809c143d88,g80478fca09+6ef8b1810f,g82479be7b0+f568feb641,g858d7b2824+6aee506743,g89c8672015+f4add4ffd5,g9125e01d80+a8ce1bb630,ga5288a1d22+2903d499ea,gb58c049af0+d64f4d3760,gc28159a63d+3f2625acfc,gcab2d0539d+b12535109e,gcf0d15dbbd+46a3f46ba9,gda6a2b7d83+46a3f46ba9,gdaeeff99f8+1711a396fd,ge79ae78c31+3f2625acfc,gef2f8181fd+0a71e47438,gf0baf85859+c1f95f4921,gfa517265be+6aee506743,gfa999e8aa5+17cd334064,w.2024.51
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._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_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_determineHDU(self.data_id['detector'])
160 return lsst.afw.image.ImageF(self._reader_path_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