LSST Applications  21.0.0+04719a4bac,21.0.0-1-ga51b5d4+f5e6047307,21.0.0-11-g2b59f77+a9c1acf22d,21.0.0-11-ga42c5b2+86977b0b17,21.0.0-12-gf4ce030+76814010d2,21.0.0-13-g1721dae+760e7a6536,21.0.0-13-g3a573fe+768d78a30a,21.0.0-15-g5a7caf0+f21cbc5713,21.0.0-16-g0fb55c1+b60e2d390c,21.0.0-19-g4cded4ca+71a93a33c0,21.0.0-2-g103fe59+bb20972958,21.0.0-2-g45278ab+04719a4bac,21.0.0-2-g5242d73+3ad5d60fb1,21.0.0-2-g7f82c8f+8babb168e8,21.0.0-2-g8f08a60+06509c8b61,21.0.0-2-g8faa9b5+616205b9df,21.0.0-2-ga326454+8babb168e8,21.0.0-2-gde069b7+5e4aea9c2f,21.0.0-2-gecfae73+1d3a86e577,21.0.0-2-gfc62afb+3ad5d60fb1,21.0.0-25-g1d57be3cd+e73869a214,21.0.0-3-g357aad2+ed88757d29,21.0.0-3-g4a4ce7f+3ad5d60fb1,21.0.0-3-g4be5c26+3ad5d60fb1,21.0.0-3-g65f322c+e0b24896a3,21.0.0-3-g7d9da8d+616205b9df,21.0.0-3-ge02ed75+a9c1acf22d,21.0.0-4-g591bb35+a9c1acf22d,21.0.0-4-g65b4814+b60e2d390c,21.0.0-4-gccdca77+0de219a2bc,21.0.0-4-ge8a399c+6c55c39e83,21.0.0-5-gd00fb1e+05fce91b99,21.0.0-6-gc675373+3ad5d60fb1,21.0.0-64-g1122c245+4fb2b8f86e,21.0.0-7-g04766d7+cd19d05db2,21.0.0-7-gdf92d54+04719a4bac,21.0.0-8-g5674e7b+d1bd76f71f,master-gac4afde19b+a9c1acf22d,w.2021.13
LSST Data Management Base Package
shims.py
Go to the documentation of this file.
1 # This file is part of pipe_base.
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 """Shim classes that provide (limited) Gen2 Butler interfaces to Gen3
23 data repositories.
24 
25 All of the classes here still operate on Gen3 data IDs - the shim layer
26 essentially assumes that Gen2 Tasks treat the data ID as an opaque blob, which
27 is usually (but not always true). When it isn't, the best approach now is
28 probably to use the GENERATION class attribute on the Butler classes to
29 special-case code for each generation.
30 """
31 
32 __all__ = ("ShimButler", "ShimButlerSubset", "ShimDataRef")
33 
34 from lsst.daf.persistence import NoResults
35 from lsst.daf.butler import StorageClassFactory
36 
37 
38 class ShimButler:
39  """A shim for a Gen2 `~lsst.daf.persistence.Butler` with a Gen3
40  `~lsst.daf.butler.Butler` backend.
41 
42  Parameters
43  ----------
44  butler3 : `lsst.daf.butler.Butler`
45  Generation 3 Butler instance.
46  """
47 
48  GENERATION = 2.5
49  """This is a Generation 2 shim for a Generation3 Butler.
50  """
51 
52  def __init__(self, butler3):
53  self._butler3_butler3 = butler3
54 
55  def _makeDataId(self, dataId=None, **rest):
56  """Construct a full data ID by merging the given arguments with the
57  internal data ID.
58  """
59  fullDataId = dict()
60  if dataId is not None:
61  fullDataId.update(dataId)
62  fullDataId.update(rest)
63  return fullDataId
64 
65  def _translateDatasetType(self, datasetType):
66  if "_" in datasetType:
67  if datasetType.endswith("_md"):
68  return f"{datasetType[:-3]}.metadata"
69  for component in StorageClassFactory().getStorageClass("Exposure").components:
70  suffix = f"_{component}"
71  if datasetType.endswith(suffix):
72  return "{}.{}".format(datasetType[:-len(suffix)], component)
73  return datasetType
74 
75  def datasetExists(self, datasetType, dataId=None, write=False, **rest):
76  """Check whether a datataset exists in the repository.
77 
78  Parameters
79  ----------
80  datasetType : `str`
81  Name of the Gen2 dataset type.
82  dataId : `dict` or `~lsst.daf.butler.DataId`, optional
83  A Generation 3 data ID that identifies the dataset.
84  write : `bool`
85  This option is provided for compatibility with
86  `lsst.daf.persistence.Butler`, but must be `False`.
87  rest
88  Additional key-value pairs to augment the given data ID.
89 
90  Returns
91  -------
92  exists : `bool`
93  `True` if the dataset is present in the repository, `False`
94  otherwise.
95  """
96  if write:
97  raise NotImplementedError("ShimButler cannot implement datasetExists with 'write=True'")
98  datasetType = self._translateDatasetType_translateDatasetType(datasetType)
99  try:
100  return self._butler3_butler3.datasetExists(datasetType, self._makeDataId_makeDataId(dataId, **rest))
101  except LookupError:
102  # Gen3 datasetExists raises if Dataset is not present in Registry;
103  # Gen2 does not distinguish between present in Datastore and
104  # present in Registry.
105  return False
106 
107  def get(self, datasetType, dataId=None, immediate=True, **rest):
108  """Retrieve a dataset.
109 
110  Parameters
111  ----------
112  datasetType : `str`
113  Name of the Gen2 dataset type.
114  dataId : `dict` or `~lsst.daf.butler.DataId`, optional
115  A Generation 3 data ID that identifies the dataset.
116  immediate : `bool`
117  This option is provided for compatibility with
118  `lsst.daf.persistence.Butler`, but is ignored.
119  rest
120  Additional key-value pairs to augment the given data ID.
121 
122  Returns
123  -------
124  dataset
125  Retrieved object.
126 
127  Raises
128  ------
129  `~lsst.daf.persistence.NoResults`
130  Raised if the dataset does not exist.
131  """
132  datasetType = self._translateDatasetType_translateDatasetType(datasetType)
133  fullDataId = self._makeDataId_makeDataId(dataId, **rest)
134  if datasetType.endswith("_sub"):
135  import lsst.afw.image
136  datasetType = datasetType[:-len("_sub")]
137  parameters = dict(bbox=fullDataId.pop("bbox"))
138  origin = fullDataId.pop("imageOrigin", lsst.afw.image.PARENT)
139  parameters["origin"] = origin
140  else:
141  parameters = {}
142  try:
143  return self._butler3_butler3.get(datasetType, fullDataId, parameters=parameters)
144  except (FileNotFoundError, LookupError) as err:
145  raise NoResults(str(err), datasetType, fullDataId)
146 
147  def put(self, obj, datasetType, dataId=None, doBackup=False, **rest):
148  """Write a dataset.
149 
150  Parameters
151  ----------
152  obj
153  Object to write.
154  datasetType : `str`
155  Name of the Gen2 dataset type.
156  dataId : `dict` or `~lsst.daf.butler.DataId`, optional
157  A Generation 3 data ID that identifies the dataset.
158  doBackup : `bool`
159  This option is provided for compatibility with
160  `lsst.daf.persistence.Butler`, but must be `False`.
161  rest
162  Additional key-value pairs to augment the given data ID.
163  """
164  if doBackup:
165  raise NotImplementedError("ShimButler cannot implement put with 'doBackup=True'")
166  datasetType = self._translateDatasetType_translateDatasetType(datasetType)
167  self._butler3_butler3.put(obj, datasetType, self._makeDataId_makeDataId(dataId, **rest))
168 
169  def dataRef(self, datasetType, level=None, dataId=None, **rest):
170  """Return a DataRef associated with the given dataset type and data ID.
171 
172  Parameters
173  ----------
174  datasetType : `str`
175  Name of the dataset type.
176  dataId : `dict` or `~lsst.daf.butler.DataId`, optional
177  A Generation 3 data ID that identifies the dataset.
178  level
179  This option is provided for compatibility with
180  `lsst.daf.persistence.Butler`, but must be `None`.
181  rest
182  Additional key-value pairs to augment the given data ID.
183  """
184  if level is not None:
185  raise NotImplementedError("ShimButler cannot implement dataRef with 'level != None'")
186  fullDataId = {}
187  if dataId is not None:
188  fullDataId.update(dataId)
189  fullDataId.update(rest)
190  return next(iter(ShimButlerSubset(self, datasetType, [fullDataId])))
191 
192 
194  """A shim for a Gen2 `~lsst.daf.persistence.ButlerSubset` with a Gen3
195  `~lsst.daf.butler.Butler` backend.
196 
197  Parameters
198  ----------
199  butler : `ShimButler`
200  Butler shim instance.
201  datasetType : `str`
202  Name of the dataset type.
203  dataIds : iterable of `dict` or `~lsst.daf.butler.DataId`
204  Generation 3 data IDs that define the data in this subset.
205  """
206 
207  GENERATION = 2.5
208  """This is a Generation 2 shim for a Generation3 Butler.
209  """
210 
211  def __init__(self, butler, datasetType, dataIds):
212  self.butlerbutler = butler
213  self.datasetTypedatasetType = datasetType
214  self._dataIds_dataIds = tuple(dataIds)
215 
216  def __len__(self):
217  return len(self._dataIds_dataIds)
218 
219  def __iter__(self):
220  for dataId in self._dataIds_dataIds:
221  yield ShimDataRef(self, dataId)
222 
223 
225  """A shim for a Gen2 `~lsst.daf.persistence.ButlerDataRef` with a Gen3
226  `~lsst.daf.butler.Butler` backend.
227 
228  Parameters
229  ----------
230  butlerSubset : `ShimButlerSubset`
231  ButlerSubset shim instance. Sets the butler and default dataset type
232  used by the Dataref.
233  dataId : `dict` or `~lsst.daf.butler.DataId`
234  Generation 3 data ID associated with this reference.
235  """
236 
237  GENERATION = 2.5
238  """This is a Generation 2 shim for a Generation3 Butler.
239  """
240 
241  def __init__(self, butlerSubset, dataId):
242  self.butlerSubsetbutlerSubset = butlerSubset
243  self.dataIddataId = dataId
244 
245  def get(self, datasetType=None, **rest):
246  """Retrieve a dataset.
247 
248  Parameters
249  ----------
250  datasetType : `str`, optional.
251  Name of the dataset type. Defaults to the dataset type used to
252  construct the `ShimButlerSubset`.
253  rest
254  Additional arguments forwarded to `ShimButler.get`.
255 
256  Returns
257  -------
258  dataset
259  Retrieved object.
260 
261  Raises
262  ------
263  `~lsst.daf.persistence.NoResults`
264  Raised if the dataset does not exist.
265  """
266  if datasetType is None:
267  datasetType = self.butlerSubsetbutlerSubset.datasetType
268  return self.butlerSubsetbutlerSubset.butler.get(datasetType, self.dataIddataId, **rest)
269 
270  def put(self, obj, datasetType=None, doBackup=False, **rest):
271  """Write a dataset.
272 
273  Parameters
274  ----------
275  obj
276  Object to write.
277  datasetType : `str`, optional
278  Name of the dataset type. Defaults to the dataset type used to
279  construct the `ShimButlerSubset`.
280  doBackup : `bool`
281  This option is provided for compatibility with
282  `lsst.daf.persistence.ButlerDataRef`, but must be `False`.
283  rest
284  Additional arguments forwarded to `ShimButler.put`.
285  """
286  if datasetType is None:
287  datasetType = self.butlerSubsetbutlerSubset.datasetType
288  self.butlerSubsetbutlerSubset.butler.put(obj, datasetType, self.dataIddataId, doBackup=doBackup, **rest)
289 
290  def datasetExists(self, datasetType=None, write=False, **rest):
291  """Check whether a datataset exists in the repository.
292 
293  Parameters
294  ----------
295  datasetType : `str`, optional
296  Name of the dataset type. Defaults to the dataset type used to
297  construct the `ShimButlerSubset`.
298  write : `bool`
299  This option is provided for compatibility with
300  `lsst.daf.persistence.ButlerDataRef`, but must be `False`.
301  rest
302  Additional arguments forwarded to `ShimButler.datasetExists`.
303 
304  Returns
305  -------
306  exists : `bool`
307  `True` if the dataset is present in the repository, `False`
308  otherwise.
309  """
310  if datasetType is None:
311  datasetType = self.butlerSubsetbutlerSubset.datasetType
312  return self.butlerSubsetbutlerSubset.butler.datasetExists(
313  datasetType, self.dataIddataId, write=write, **rest)
314 
315  def getButler(self):
316  """Return the (shim) Butler used by this DataRef.
317  """
318  return self.butlerSubsetbutlerSubset.butler
def get(self, datasetType, dataId=None, immediate=True, **rest)
Definition: shims.py:107
def dataRef(self, datasetType, level=None, dataId=None, **rest)
Definition: shims.py:169
def put(self, obj, datasetType, dataId=None, doBackup=False, **rest)
Definition: shims.py:147
def _translateDatasetType(self, datasetType)
Definition: shims.py:65
def __init__(self, butler3)
Definition: shims.py:52
def _makeDataId(self, dataId=None, **rest)
Definition: shims.py:55
def datasetExists(self, datasetType, dataId=None, write=False, **rest)
Definition: shims.py:75
def __init__(self, butler, datasetType, dataIds)
Definition: shims.py:211
def datasetExists(self, datasetType=None, write=False, **rest)
Definition: shims.py:290
def __init__(self, butlerSubset, dataId)
Definition: shims.py:241
def put(self, obj, datasetType=None, doBackup=False, **rest)
Definition: shims.py:270
def get(self, datasetType=None, **rest)
Definition: shims.py:245
Backwards-compatibility support for depersisting the old Calib (FluxMag0/FluxMag0Err) objects.
def format(config, name=None, writeSourceLine=True, prefix="", verbose=False)
Definition: history.py:174