LSST Applications g0b6bd0c080+a72a5dd7e6,g1182afd7b4+2a019aa3bb,g17e5ecfddb+2b8207f7de,g1d67935e3f+06cf436103,g38293774b4+ac198e9f13,g396055baef+6a2097e274,g3b44f30a73+6611e0205b,g480783c3b1+98f8679e14,g48ccf36440+89c08d0516,g4b93dc025c+98f8679e14,g5c4744a4d9+a302e8c7f0,g613e996a0d+e1c447f2e0,g6c8d09e9e7+25247a063c,g7271f0639c+98f8679e14,g7a9cd813b8+124095ede6,g9d27549199+a302e8c7f0,ga1cf026fa3+ac198e9f13,ga32aa97882+7403ac30ac,ga786bb30fb+7a139211af,gaa63f70f4e+9994eb9896,gabf319e997+ade567573c,gba47b54d5d+94dc90c3ea,gbec6a3398f+06cf436103,gc6308e37c7+07dd123edb,gc655b1545f+ade567573c,gcc9029db3c+ab229f5caf,gd01420fc67+06cf436103,gd877ba84e5+06cf436103,gdb4cecd868+6f279b5b48,ge2d134c3d5+cc4dbb2e3f,ge448b5faa6+86d1ceac1d,gecc7e12556+98f8679e14,gf3ee170dca+25247a063c,gf4ac96e456+ade567573c,gf9f5ea5b4d+ac198e9f13,gff490e6085+8c2580be5c,w.2022.27
LSST Data Management Base Package
repositoryCfg.py
Go to the documentation of this file.
1#!/usr/bin/env python
2
3#
4# LSST Data Management System
5# Copyright 2016 LSST Corporation.
6#
7# This product includes software developed by the
8# LSST Project (http://www.lsst.org/).
9#
10# This program is free software: you can redistribute it and/or modify
11# it under the terms of the GNU General Public License as published by
12# the Free Software Foundation, either version 3 of the License, or
13# (at your option) any later version.
14#
15# This program is distributed in the hope that it will be useful,
16# but WITHOUT ANY WARRANTY; without even the implied warranty of
17# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18# GNU General Public License for more details.
19#
20# You should have received a copy of the LSST License Statement and
21# the GNU General Public License along with this program. If not,
22# see <http://www.lsstcorp.org/LegalNotices/>.
23#
24
25# -*- python -*-
26
27import copy
28import yaml
29from . import iterify, doImport, Storage, ParentsMismatch
30
31
32class RepositoryCfg(yaml.YAMLObject):
33 """RepositoryCfg stores the configuration of a repository. Its contents are persisted to the repository
34 when the repository is created in persistent storage. Thereafter the the RepositoryCfg should not change.
35
36 Parameters
37 ----------
38 mapper : string
39 The mapper associated with the repository. The string should be importable to a class object.
40 mapperArgs : dict
41 Arguments & values to pass to the mapper when initializing it.
42 parents : list of URI
43 URIs to the locaiton of the parent RepositoryCfgs of this repository.
44 policy : dict
45 Policy associated with this repository, overrides all other policy data (which may be loaded from
46 policies in derived packages).
47 deserializing : bool
48 Butler internal use only. This flag is used to indicate to the init funciton that the repository class
49 is being deserialized and should not perform certain operations that normally happen in other uses of
50 init.
51 """
52 yaml_tag = u"!RepositoryCfg_v1"
53
54 def __init__(self, root, mapper, mapperArgs, parents, policy):
55 self._root_root = root
56 self._mapper_mapper = mapper
57 self._mapperArgs_mapperArgs = {} if mapperArgs is None else mapperArgs
58 self._parents_parents = []
59 self.addParentsaddParents(iterify(parents))
60 self._policy_policy = policy
61 self.dirtydirty = True # if dirty, the parameters have been changed since the cfg was read or written.
62
63 @staticmethod
64 def v1Constructor(loader, node):
65 """Constructor for 'version 1' of the serlized RepositoryCfg.
66
67 If new parameters are added to RepositoryCfg they will have to be checked for in d; if they are there
68 then their value should be used and if they are not there a default value must be used in place.
69
70 In case the structure of the serialzed file must be changed in a way that invalidates some of the
71 keys:
72 1. Increment the version number (after _v1) in the yaml_tag of this class.
73 2. Add a new constructor (similar to this one) to deserialze new serializations of this class.
74 3. Registered the new constructor for the new version with yaml, the same way it is done at the bottom
75 of this file.
76 4. All constructors for the older version(s) of persisted RepositoryCfg must be changed to adapt
77 the old keys to their new uses and create the current (new) version of a repository cfg, or raise a
78 RuntimeError in the case that older versions of serialized RepositoryCfgs can not be adapted.
79 There is an example of migrating from a fictitious v0 to v1 in tests/repositoryCfg.py
80 """
81 d = loader.construct_mapping(node)
82 cfg = RepositoryCfg(root=d['_root'], mapper=d['_mapper'], mapperArgs=d['_mapperArgs'],
83 parents=[], policy=d.get('_policy', None))
84 # Where possible we mangle the parents so that they are relative to root, for example if the root and
85 # the parents are both in the same PosixStorage. The parents are serialized in mangled form; when
86 # deserializing the parents we do not re-mangle them.
87 cfg._parents = d['_parents']
88 if cfg._parents is None:
89 cfg._parents = []
90 cfg.dirty = False
91 return cfg
92
93 def __eq__(self, other):
94 if not other:
95 return False
96 return self.rootrootrootroot == other.root and \
97 self.mappermappermappermapper == other.mapper and \
98 self.mapperArgsmapperArgsmapperArgsmapperArgs == other.mapperArgs and \
99 self.parentsparentsparents == other.parents and \
100 self.policypolicypolicy == other.policy
101
102 def __ne__(self, other):
103 return not self.__eq____eq__(other)
104
105 def extend(self, other):
106 """Extend this RepositoryCfg with extendable values from the other RepositoryCfg.
107
108 Currently the only extendable value is parents; see `extendParents` for more detials about extending
109 the parents list.
110
111 Parameters
112 ----------
113 other : RepositoryCfg
114 A RepositoryCfg instance to update values from.
115
116 Raises
117 ------
118 RuntimeError
119 If non-extendable parameters do not match a RuntimeError will be raised.
120 (If this RepositoryCfg's parents can not be extended with the parents of the other repository,
121 extendParents will raise).
122 """
123 if (self.rootrootrootroot != other.root
124 or self.mappermappermappermapper != other.mapper
125 or self.mapperArgsmapperArgsmapperArgsmapperArgs != other.mapperArgs
126 or self.policypolicypolicy != other.policy):
127 raise RuntimeError("{} can not be extended with cfg:{}".format(self, other))
128 self.extendParentsextendParents(other.parents)
129
130 def _extendsParents(self, newParents):
131 """Query if a list of parents starts with the same list of parents as this RepositoryCfg's parents,
132 with new parents at the end.
133
134 Parameters
135 ----------
136 newParents : list of string and/or RepositoryCfg
137 A list of parents that contains all the parents that would be in this RepositoryCfg.
138 This must include parents that may already be in this RepositoryCfg's parents list. Paths must be
139 in absolute form (not relative).
140
141 Returns
142 -------
143 bool
144 True if the beginning of the new list matches this RepositoryCfg's parents list, False if not.
145 """
146 doesExtendParents = False
147 return doesExtendParents
148
149 def extendParents(self, newParents):
150 """Determine if a parents list matches our parents list, with extra items at the end. If a list of
151 parents does not match but the mismatch is because of new parents at the end of the list, then they
152 can be added to the cfg.
153
154 Parameters
155 ----------
156 newParents : list of string
157 A list of parents that contains all the parents that are to be recorded into this RepositoryCfg.
158 This must include parents that may already be in this RepositoryCfg's parents list
159
160 Raises
161 ------
162 ParentsListMismatch
163 Description
164 """
165 newParents = self._normalizeParents_normalizeParents(self.rootrootrootroot, newParents)
166 doRaise = False
167 if self._parents_parents != newParents:
168 if all(x == y for (x, y) in zip(self._parents_parents, newParents)):
169 if len(self._parents_parents) < len(newParents):
170 self._parents_parents = newParents
171 self.dirtydirty = True
172 elif len(self._parents_parents) == len(newParents):
173 pass
174 else:
175 doRaise = True
176 else:
177 doRaise = True
178 if doRaise:
179 raise ParentsMismatch(("The beginning of the passed-in parents list: {} does not match the "
180 "existing parents list in this RepositoryCfg: {}").format(
181 newParents, self._parents_parents))
182
183 @property
184 def root(self):
185 return self._root_root
186
187 @root.setter
188 def root(self, root):
189 if root is not None and self._root_root is not None:
190 raise RuntimeError("Explicity clear root (set to None) before changing the value of root.")
191 self._root_root = root
192
193 @property
194 def mapper(self):
195 return self._mapper_mapper
196
197 @mapper.setter
198 def mapper(self, mapper):
199 if self._mapper_mapper is not None:
200 raise RuntimeError("Should not set mapper over previous not-None value.")
201 self.dirtydirty = True
202 self._mapper_mapper = mapper
203
204 @property
205 def mapperArgs(self):
206 return self._mapperArgs_mapperArgs
207
208 @mapperArgs.setter
209 def mapperArgs(self, newDict):
210 self.dirtydirty = True
211 self._mapperArgs_mapperArgs = newDict
212
213 @property
214 def parents(self):
215 return self._denormalizeParents_denormalizeParents(self.rootrootrootroot, self._parents_parents)
216
217 @staticmethod
218 def _normalizeParents(root, newParents):
219 """Eliminate symlinks in newParents and get the relative path (if one exists) from root to each parent
220 root.
221
222 Parameters
223 ----------
224 newParents : list containing strings and RepoistoryCfg instances
225 Same as in `addParents`.
226
227 Returns
228 -------
229 list of strings and RepositoryCfg instances.
230 Normalized list of parents
231 """
232 newParents = iterify(newParents)
233 for i in range(len(newParents)):
234 if isinstance(newParents[i], RepositoryCfg):
235 newParents[i] = copy.copy(newParents[i])
236 parentRoot = newParents[i].root
237 newParents[i].root = None
238 newParents[i].root = Storage.relativePath(root, parentRoot)
239 else:
240 newParents[i] = Storage.relativePath(root, newParents[i])
241 return newParents
242
243 @staticmethod
244 def _denormalizeParents(root, parents):
245 def getAbs(root, parent):
246 if isinstance(parent, RepositoryCfg):
247 parentRoot = parent.root
248 parent.root = None
249 parent.root = Storage.absolutePath(root, parentRoot)
250 else:
251 parent = Storage.absolutePath(root, parent)
252 return parent
253 return [getAbs(root, parent) for parent in parents]
254
255 def addParents(self, newParents):
256 """Add a parent or list of parents to this RepositoryCfg
257
258 Parameters
259 ----------
260 newParents : string or RepositoryCfg instance, or list of these.
261 If string, newParents should be a path or URI to the parent
262 repository. If RepositoryCfg, newParents should be a RepositoryCfg
263 that describes the parent repository in part or whole.
264 """
265 newParents = self._normalizeParents_normalizeParents(self.rootrootrootroot, newParents)
266 for newParent in newParents:
267 if newParent not in self._parents_parents:
268 self.dirtydirty = True
269 self._parents_parents.append(newParent)
270
271 @property
272 def policy(self):
273 return self._policy_policy
274
275 @staticmethod
276 def makeFromArgs(repositoryArgs):
277 cfg = RepositoryCfg(root=repositoryArgs.root,
278 mapper=repositoryArgs.mapper,
279 mapperArgs=repositoryArgs.mapperArgs,
280 parents=None,
281 policy=repositoryArgs.policy)
282 return cfg
283
284 def matchesArgs(self, repositoryArgs):
285 """Checks that a repositoryArgs instance will work with this repositoryCfg. This is useful
286 when loading an already-existing repository that has a persisted cfg, to ensure that the args that are
287 passed into butler do not conflict with the persisted cfg."""
288 if repositoryArgs.root is not None and self._root_root != repositoryArgs.root:
289 return False
290
291 repoArgsMapper = repositoryArgs.mapper
292 cfgMapper = self._mapper_mapper
293 if isinstance(repoArgsMapper, str):
294 repoArgsMapper = doImport(repoArgsMapper)
295 if isinstance(cfgMapper, str):
296 cfgMapper = doImport(cfgMapper)
297 if repoArgsMapper is not None and repoArgsMapper != cfgMapper:
298 return False
299 # check mapperArgs for any keys in common and if their value does not match then return false.
300 if self._mapperArgs_mapperArgs is not None and repositoryArgs.mapperArgs is not None:
301 for key in set(self._mapperArgs_mapperArgs.keys()) & set(repositoryArgs.mapperArgs):
302 if self._mapperArgs_mapperArgs[key] != repositoryArgs.mapperArgs[key]:
303 return False
304 if repositoryArgs.policy and repositoryArgs.policy != self._policy_policy:
305 return False
306
307 return True
308
309 def __repr__(self):
310 return "%s(root=%r, mapper=%r, mapperArgs=%r, parents=%s, policy=%s)" % (
311 self.__class__.__name__,
312 self._root_root,
313 self._mapper_mapper,
314 self._mapperArgs_mapperArgs,
315 self._parents_parents,
316 self._policy_policy)
317
318
319loaderList = [yaml.Loader, ]
320try:
321 loaderList.append(yaml.UnsafeLoader)
322except AttributeError:
323 pass
324
325for loader in loaderList:
326 yaml.add_constructor(u"!RepositoryCfg_v1", RepositoryCfg.v1Constructor, Loader=loader)
def __init__(self, root, mapper, mapperArgs, parents, policy)
daf::base::PropertySet * set
Definition: fits.cc:912
std::shared_ptr< FrameSet > append(FrameSet const &first, FrameSet const &second)
Construct a FrameSet that performs two transformations in series.
Definition: functional.cc:33
bool all(CoordinateExpr< N > const &expr) noexcept
Return true if all elements are true.
def format(config, name=None, writeSourceLine=True, prefix="", verbose=False)
Definition: history.py:174