##// END OF EJS Templates
repoview: use try/except/finally
Matt Mackall -
r25086:140c2d1e default
parent child Browse files
Show More
@@ -1,348 +1,347
1 # repoview.py - Filtered view of a localrepo object
1 # repoview.py - Filtered view of a localrepo object
2 #
2 #
3 # Copyright 2012 Pierre-Yves David <pierre-yves.david@ens-lyon.org>
3 # Copyright 2012 Pierre-Yves David <pierre-yves.david@ens-lyon.org>
4 # Logilab SA <contact@logilab.fr>
4 # Logilab SA <contact@logilab.fr>
5 #
5 #
6 # This software may be used and distributed according to the terms of the
6 # This software may be used and distributed according to the terms of the
7 # GNU General Public License version 2 or any later version.
7 # GNU General Public License version 2 or any later version.
8
8
9 import heapq
9 import heapq
10 import copy
10 import copy
11 import error
11 import error
12 import phases
12 import phases
13 import util
13 import util
14 import obsolete
14 import obsolete
15 import struct
15 import struct
16 import tags as tagsmod
16 import tags as tagsmod
17 from node import nullrev
17 from node import nullrev
18
18
19 def hideablerevs(repo):
19 def hideablerevs(repo):
20 """Revisions candidates to be hidden
20 """Revisions candidates to be hidden
21
21
22 This is a standalone function to help extensions to wrap it."""
22 This is a standalone function to help extensions to wrap it."""
23 return obsolete.getrevs(repo, 'obsolete')
23 return obsolete.getrevs(repo, 'obsolete')
24
24
25 def _getstatichidden(repo):
25 def _getstatichidden(repo):
26 """Revision to be hidden (disregarding dynamic blocker)
26 """Revision to be hidden (disregarding dynamic blocker)
27
27
28 To keep a consistent graph, we cannot hide any revisions with
28 To keep a consistent graph, we cannot hide any revisions with
29 non-hidden descendants. This function computes the set of
29 non-hidden descendants. This function computes the set of
30 revisions that could be hidden while keeping the graph consistent.
30 revisions that could be hidden while keeping the graph consistent.
31
31
32 A second pass will be done to apply "dynamic blocker" like bookmarks or
32 A second pass will be done to apply "dynamic blocker" like bookmarks or
33 working directory parents.
33 working directory parents.
34
34
35 """
35 """
36 assert not repo.changelog.filteredrevs
36 assert not repo.changelog.filteredrevs
37 hidden = set(hideablerevs(repo))
37 hidden = set(hideablerevs(repo))
38 if hidden:
38 if hidden:
39 getphase = repo._phasecache.phase
39 getphase = repo._phasecache.phase
40 getparentrevs = repo.changelog.parentrevs
40 getparentrevs = repo.changelog.parentrevs
41 # Skip heads which are public (guaranteed to not be hidden)
41 # Skip heads which are public (guaranteed to not be hidden)
42 heap = [-r for r in repo.changelog.headrevs() if getphase(repo, r)]
42 heap = [-r for r in repo.changelog.headrevs() if getphase(repo, r)]
43 heapq.heapify(heap)
43 heapq.heapify(heap)
44 heappop = heapq.heappop
44 heappop = heapq.heappop
45 heappush = heapq.heappush
45 heappush = heapq.heappush
46 seen = set() # no need to init it with heads, they have no children
46 seen = set() # no need to init it with heads, they have no children
47 while heap:
47 while heap:
48 rev = -heappop(heap)
48 rev = -heappop(heap)
49 # All children have been processed so at that point, if no children
49 # All children have been processed so at that point, if no children
50 # removed 'rev' from the 'hidden' set, 'rev' is going to be hidden.
50 # removed 'rev' from the 'hidden' set, 'rev' is going to be hidden.
51 blocker = rev not in hidden
51 blocker = rev not in hidden
52 for parent in getparentrevs(rev):
52 for parent in getparentrevs(rev):
53 if parent == nullrev:
53 if parent == nullrev:
54 continue
54 continue
55 if blocker:
55 if blocker:
56 # If visible, ensure parent will be visible too
56 # If visible, ensure parent will be visible too
57 hidden.discard(parent)
57 hidden.discard(parent)
58 # - Avoid adding the same revision twice
58 # - Avoid adding the same revision twice
59 # - Skip nodes which are public (guaranteed to not be hidden)
59 # - Skip nodes which are public (guaranteed to not be hidden)
60 pre = len(seen)
60 pre = len(seen)
61 seen.add(parent)
61 seen.add(parent)
62 if pre < len(seen) and getphase(repo, rev):
62 if pre < len(seen) and getphase(repo, rev):
63 heappush(heap, -parent)
63 heappush(heap, -parent)
64 return hidden
64 return hidden
65
65
66 def _getdynamicblockers(repo):
66 def _getdynamicblockers(repo):
67 """Non-cacheable revisions blocking hidden changesets from being filtered.
67 """Non-cacheable revisions blocking hidden changesets from being filtered.
68
68
69 Get revisions that will block hidden changesets and are likely to change,
69 Get revisions that will block hidden changesets and are likely to change,
70 but unlikely to create hidden blockers. They won't be cached, so be careful
70 but unlikely to create hidden blockers. They won't be cached, so be careful
71 with adding additional computation."""
71 with adding additional computation."""
72
72
73 cl = repo.changelog
73 cl = repo.changelog
74 blockers = set()
74 blockers = set()
75 blockers.update([par.rev() for par in repo[None].parents()])
75 blockers.update([par.rev() for par in repo[None].parents()])
76 blockers.update([cl.rev(bm) for bm in repo._bookmarks.values()])
76 blockers.update([cl.rev(bm) for bm in repo._bookmarks.values()])
77
77
78 tags = {}
78 tags = {}
79 tagsmod.readlocaltags(repo.ui, repo, tags, {})
79 tagsmod.readlocaltags(repo.ui, repo, tags, {})
80 if tags:
80 if tags:
81 rev, nodemap = cl.rev, cl.nodemap
81 rev, nodemap = cl.rev, cl.nodemap
82 blockers.update(rev(t[0]) for t in tags.values() if t[0] in nodemap)
82 blockers.update(rev(t[0]) for t in tags.values() if t[0] in nodemap)
83 return blockers
83 return blockers
84
84
85 cacheversion = 1
85 cacheversion = 1
86 cachefile = 'cache/hidden'
86 cachefile = 'cache/hidden'
87
87
88 def cachehash(repo, hideable):
88 def cachehash(repo, hideable):
89 """return sha1 hash of repository data to identify a valid cache.
89 """return sha1 hash of repository data to identify a valid cache.
90
90
91 We calculate a sha1 of repo heads and the content of the obsstore and write
91 We calculate a sha1 of repo heads and the content of the obsstore and write
92 it to the cache. Upon reading we can easily validate by checking the hash
92 it to the cache. Upon reading we can easily validate by checking the hash
93 against the stored one and discard the cache in case the hashes don't match.
93 against the stored one and discard the cache in case the hashes don't match.
94 """
94 """
95 h = util.sha1()
95 h = util.sha1()
96 h.update(''.join(repo.heads()))
96 h.update(''.join(repo.heads()))
97 h.update(str(hash(frozenset(hideable))))
97 h.update(str(hash(frozenset(hideable))))
98 return h.digest()
98 return h.digest()
99
99
100 def _writehiddencache(cachefile, cachehash, hidden):
100 def _writehiddencache(cachefile, cachehash, hidden):
101 """write hidden data to a cache file"""
101 """write hidden data to a cache file"""
102 data = struct.pack('>%ii' % len(hidden), *sorted(hidden))
102 data = struct.pack('>%ii' % len(hidden), *sorted(hidden))
103 cachefile.write(struct.pack(">H", cacheversion))
103 cachefile.write(struct.pack(">H", cacheversion))
104 cachefile.write(cachehash)
104 cachefile.write(cachehash)
105 cachefile.write(data)
105 cachefile.write(data)
106
106
107 def trywritehiddencache(repo, hideable, hidden):
107 def trywritehiddencache(repo, hideable, hidden):
108 """write cache of hidden changesets to disk
108 """write cache of hidden changesets to disk
109
109
110 Will not write the cache if a wlock cannot be obtained lazily.
110 Will not write the cache if a wlock cannot be obtained lazily.
111 The cache consists of a head of 22byte:
111 The cache consists of a head of 22byte:
112 2 byte version number of the cache
112 2 byte version number of the cache
113 20 byte sha1 to validate the cache
113 20 byte sha1 to validate the cache
114 n*4 byte hidden revs
114 n*4 byte hidden revs
115 """
115 """
116 wlock = fh = None
116 wlock = fh = None
117 try:
117 try:
118 try:
119 wlock = repo.wlock(wait=False)
118 wlock = repo.wlock(wait=False)
120 # write cache to file
119 # write cache to file
121 newhash = cachehash(repo, hideable)
120 newhash = cachehash(repo, hideable)
122 fh = repo.vfs.open(cachefile, 'w+b', atomictemp=True)
121 fh = repo.vfs.open(cachefile, 'w+b', atomictemp=True)
123 _writehiddencache(fh, newhash, hidden)
122 _writehiddencache(fh, newhash, hidden)
124 except (IOError, OSError):
123 except (IOError, OSError):
125 repo.ui.debug('error writing hidden changesets cache')
124 repo.ui.debug('error writing hidden changesets cache')
126 except error.LockHeld:
125 except error.LockHeld:
127 repo.ui.debug('cannot obtain lock to write hidden changesets cache')
126 repo.ui.debug('cannot obtain lock to write hidden changesets cache')
128 finally:
127 finally:
129 if fh:
128 if fh:
130 fh.close()
129 fh.close()
131 if wlock:
130 if wlock:
132 wlock.release()
131 wlock.release()
133
132
134 def tryreadcache(repo, hideable):
133 def tryreadcache(repo, hideable):
135 """read a cache if the cache exists and is valid, otherwise returns None."""
134 """read a cache if the cache exists and is valid, otherwise returns None."""
136 hidden = fh = None
135 hidden = fh = None
137 try:
136 try:
138 if repo.vfs.exists(cachefile):
137 if repo.vfs.exists(cachefile):
139 fh = repo.vfs.open(cachefile, 'rb')
138 fh = repo.vfs.open(cachefile, 'rb')
140 version, = struct.unpack(">H", fh.read(2))
139 version, = struct.unpack(">H", fh.read(2))
141 oldhash = fh.read(20)
140 oldhash = fh.read(20)
142 newhash = cachehash(repo, hideable)
141 newhash = cachehash(repo, hideable)
143 if (cacheversion, oldhash) == (version, newhash):
142 if (cacheversion, oldhash) == (version, newhash):
144 # cache is valid, so we can start reading the hidden revs
143 # cache is valid, so we can start reading the hidden revs
145 data = fh.read()
144 data = fh.read()
146 count = len(data) / 4
145 count = len(data) / 4
147 hidden = frozenset(struct.unpack('>%ii' % count, data))
146 hidden = frozenset(struct.unpack('>%ii' % count, data))
148 return hidden
147 return hidden
149 finally:
148 finally:
150 if fh:
149 if fh:
151 fh.close()
150 fh.close()
152
151
153 def computehidden(repo):
152 def computehidden(repo):
154 """compute the set of hidden revision to filter
153 """compute the set of hidden revision to filter
155
154
156 During most operation hidden should be filtered."""
155 During most operation hidden should be filtered."""
157 assert not repo.changelog.filteredrevs
156 assert not repo.changelog.filteredrevs
158
157
159 hidden = frozenset()
158 hidden = frozenset()
160 hideable = hideablerevs(repo)
159 hideable = hideablerevs(repo)
161 if hideable:
160 if hideable:
162 cl = repo.changelog
161 cl = repo.changelog
163 hidden = tryreadcache(repo, hideable)
162 hidden = tryreadcache(repo, hideable)
164 if hidden is None:
163 if hidden is None:
165 hidden = frozenset(_getstatichidden(repo))
164 hidden = frozenset(_getstatichidden(repo))
166 trywritehiddencache(repo, hideable, hidden)
165 trywritehiddencache(repo, hideable, hidden)
167
166
168 # check if we have wd parents, bookmarks or tags pointing to hidden
167 # check if we have wd parents, bookmarks or tags pointing to hidden
169 # changesets and remove those.
168 # changesets and remove those.
170 dynamic = hidden & _getdynamicblockers(repo)
169 dynamic = hidden & _getdynamicblockers(repo)
171 if dynamic:
170 if dynamic:
172 blocked = cl.ancestors(dynamic, inclusive=True)
171 blocked = cl.ancestors(dynamic, inclusive=True)
173 hidden = frozenset(r for r in hidden if r not in blocked)
172 hidden = frozenset(r for r in hidden if r not in blocked)
174 return hidden
173 return hidden
175
174
176 def computeunserved(repo):
175 def computeunserved(repo):
177 """compute the set of revision that should be filtered when used a server
176 """compute the set of revision that should be filtered when used a server
178
177
179 Secret and hidden changeset should not pretend to be here."""
178 Secret and hidden changeset should not pretend to be here."""
180 assert not repo.changelog.filteredrevs
179 assert not repo.changelog.filteredrevs
181 # fast path in simple case to avoid impact of non optimised code
180 # fast path in simple case to avoid impact of non optimised code
182 hiddens = filterrevs(repo, 'visible')
181 hiddens = filterrevs(repo, 'visible')
183 if phases.hassecret(repo):
182 if phases.hassecret(repo):
184 cl = repo.changelog
183 cl = repo.changelog
185 secret = phases.secret
184 secret = phases.secret
186 getphase = repo._phasecache.phase
185 getphase = repo._phasecache.phase
187 first = min(cl.rev(n) for n in repo._phasecache.phaseroots[secret])
186 first = min(cl.rev(n) for n in repo._phasecache.phaseroots[secret])
188 revs = cl.revs(start=first)
187 revs = cl.revs(start=first)
189 secrets = set(r for r in revs if getphase(repo, r) >= secret)
188 secrets = set(r for r in revs if getphase(repo, r) >= secret)
190 return frozenset(hiddens | secrets)
189 return frozenset(hiddens | secrets)
191 else:
190 else:
192 return hiddens
191 return hiddens
193
192
194 def computemutable(repo):
193 def computemutable(repo):
195 """compute the set of revision that should be filtered when used a server
194 """compute the set of revision that should be filtered when used a server
196
195
197 Secret and hidden changeset should not pretend to be here."""
196 Secret and hidden changeset should not pretend to be here."""
198 assert not repo.changelog.filteredrevs
197 assert not repo.changelog.filteredrevs
199 # fast check to avoid revset call on huge repo
198 # fast check to avoid revset call on huge repo
200 if util.any(repo._phasecache.phaseroots[1:]):
199 if util.any(repo._phasecache.phaseroots[1:]):
201 getphase = repo._phasecache.phase
200 getphase = repo._phasecache.phase
202 maymutable = filterrevs(repo, 'base')
201 maymutable = filterrevs(repo, 'base')
203 return frozenset(r for r in maymutable if getphase(repo, r))
202 return frozenset(r for r in maymutable if getphase(repo, r))
204 return frozenset()
203 return frozenset()
205
204
206 def computeimpactable(repo):
205 def computeimpactable(repo):
207 """Everything impactable by mutable revision
206 """Everything impactable by mutable revision
208
207
209 The immutable filter still have some chance to get invalidated. This will
208 The immutable filter still have some chance to get invalidated. This will
210 happen when:
209 happen when:
211
210
212 - you garbage collect hidden changeset,
211 - you garbage collect hidden changeset,
213 - public phase is moved backward,
212 - public phase is moved backward,
214 - something is changed in the filtering (this could be fixed)
213 - something is changed in the filtering (this could be fixed)
215
214
216 This filter out any mutable changeset and any public changeset that may be
215 This filter out any mutable changeset and any public changeset that may be
217 impacted by something happening to a mutable revision.
216 impacted by something happening to a mutable revision.
218
217
219 This is achieved by filtered everything with a revision number egal or
218 This is achieved by filtered everything with a revision number egal or
220 higher than the first mutable changeset is filtered."""
219 higher than the first mutable changeset is filtered."""
221 assert not repo.changelog.filteredrevs
220 assert not repo.changelog.filteredrevs
222 cl = repo.changelog
221 cl = repo.changelog
223 firstmutable = len(cl)
222 firstmutable = len(cl)
224 for roots in repo._phasecache.phaseroots[1:]:
223 for roots in repo._phasecache.phaseroots[1:]:
225 if roots:
224 if roots:
226 firstmutable = min(firstmutable, min(cl.rev(r) for r in roots))
225 firstmutable = min(firstmutable, min(cl.rev(r) for r in roots))
227 # protect from nullrev root
226 # protect from nullrev root
228 firstmutable = max(0, firstmutable)
227 firstmutable = max(0, firstmutable)
229 return frozenset(xrange(firstmutable, len(cl)))
228 return frozenset(xrange(firstmutable, len(cl)))
230
229
231 # function to compute filtered set
230 # function to compute filtered set
232 #
231 #
233 # When adding a new filter you MUST update the table at:
232 # When adding a new filter you MUST update the table at:
234 # mercurial.branchmap.subsettable
233 # mercurial.branchmap.subsettable
235 # Otherwise your filter will have to recompute all its branches cache
234 # Otherwise your filter will have to recompute all its branches cache
236 # from scratch (very slow).
235 # from scratch (very slow).
237 filtertable = {'visible': computehidden,
236 filtertable = {'visible': computehidden,
238 'served': computeunserved,
237 'served': computeunserved,
239 'immutable': computemutable,
238 'immutable': computemutable,
240 'base': computeimpactable}
239 'base': computeimpactable}
241
240
242 def filterrevs(repo, filtername):
241 def filterrevs(repo, filtername):
243 """returns set of filtered revision for this filter name"""
242 """returns set of filtered revision for this filter name"""
244 if filtername not in repo.filteredrevcache:
243 if filtername not in repo.filteredrevcache:
245 func = filtertable[filtername]
244 func = filtertable[filtername]
246 repo.filteredrevcache[filtername] = func(repo.unfiltered())
245 repo.filteredrevcache[filtername] = func(repo.unfiltered())
247 return repo.filteredrevcache[filtername]
246 return repo.filteredrevcache[filtername]
248
247
249 class repoview(object):
248 class repoview(object):
250 """Provide a read/write view of a repo through a filtered changelog
249 """Provide a read/write view of a repo through a filtered changelog
251
250
252 This object is used to access a filtered version of a repository without
251 This object is used to access a filtered version of a repository without
253 altering the original repository object itself. We can not alter the
252 altering the original repository object itself. We can not alter the
254 original object for two main reasons:
253 original object for two main reasons:
255 - It prevents the use of a repo with multiple filters at the same time. In
254 - It prevents the use of a repo with multiple filters at the same time. In
256 particular when multiple threads are involved.
255 particular when multiple threads are involved.
257 - It makes scope of the filtering harder to control.
256 - It makes scope of the filtering harder to control.
258
257
259 This object behaves very closely to the original repository. All attribute
258 This object behaves very closely to the original repository. All attribute
260 operations are done on the original repository:
259 operations are done on the original repository:
261 - An access to `repoview.someattr` actually returns `repo.someattr`,
260 - An access to `repoview.someattr` actually returns `repo.someattr`,
262 - A write to `repoview.someattr` actually sets value of `repo.someattr`,
261 - A write to `repoview.someattr` actually sets value of `repo.someattr`,
263 - A deletion of `repoview.someattr` actually drops `someattr`
262 - A deletion of `repoview.someattr` actually drops `someattr`
264 from `repo.__dict__`.
263 from `repo.__dict__`.
265
264
266 The only exception is the `changelog` property. It is overridden to return
265 The only exception is the `changelog` property. It is overridden to return
267 a (surface) copy of `repo.changelog` with some revisions filtered. The
266 a (surface) copy of `repo.changelog` with some revisions filtered. The
268 `filtername` attribute of the view control the revisions that need to be
267 `filtername` attribute of the view control the revisions that need to be
269 filtered. (the fact the changelog is copied is an implementation detail).
268 filtered. (the fact the changelog is copied is an implementation detail).
270
269
271 Unlike attributes, this object intercepts all method calls. This means that
270 Unlike attributes, this object intercepts all method calls. This means that
272 all methods are run on the `repoview` object with the filtered `changelog`
271 all methods are run on the `repoview` object with the filtered `changelog`
273 property. For this purpose the simple `repoview` class must be mixed with
272 property. For this purpose the simple `repoview` class must be mixed with
274 the actual class of the repository. This ensures that the resulting
273 the actual class of the repository. This ensures that the resulting
275 `repoview` object have the very same methods than the repo object. This
274 `repoview` object have the very same methods than the repo object. This
276 leads to the property below.
275 leads to the property below.
277
276
278 repoview.method() --> repo.__class__.method(repoview)
277 repoview.method() --> repo.__class__.method(repoview)
279
278
280 The inheritance has to be done dynamically because `repo` can be of any
279 The inheritance has to be done dynamically because `repo` can be of any
281 subclasses of `localrepo`. Eg: `bundlerepo` or `statichttprepo`.
280 subclasses of `localrepo`. Eg: `bundlerepo` or `statichttprepo`.
282 """
281 """
283
282
284 def __init__(self, repo, filtername):
283 def __init__(self, repo, filtername):
285 object.__setattr__(self, '_unfilteredrepo', repo)
284 object.__setattr__(self, '_unfilteredrepo', repo)
286 object.__setattr__(self, 'filtername', filtername)
285 object.__setattr__(self, 'filtername', filtername)
287 object.__setattr__(self, '_clcachekey', None)
286 object.__setattr__(self, '_clcachekey', None)
288 object.__setattr__(self, '_clcache', None)
287 object.__setattr__(self, '_clcache', None)
289
288
290 # not a propertycache on purpose we shall implement a proper cache later
289 # not a propertycache on purpose we shall implement a proper cache later
291 @property
290 @property
292 def changelog(self):
291 def changelog(self):
293 """return a filtered version of the changeset
292 """return a filtered version of the changeset
294
293
295 this changelog must not be used for writing"""
294 this changelog must not be used for writing"""
296 # some cache may be implemented later
295 # some cache may be implemented later
297 unfi = self._unfilteredrepo
296 unfi = self._unfilteredrepo
298 unfichangelog = unfi.changelog
297 unfichangelog = unfi.changelog
299 revs = filterrevs(unfi, self.filtername)
298 revs = filterrevs(unfi, self.filtername)
300 cl = self._clcache
299 cl = self._clcache
301 newkey = (len(unfichangelog), unfichangelog.tip(), hash(revs),
300 newkey = (len(unfichangelog), unfichangelog.tip(), hash(revs),
302 unfichangelog._delayed)
301 unfichangelog._delayed)
303 if cl is not None:
302 if cl is not None:
304 # we need to check curkey too for some obscure reason.
303 # we need to check curkey too for some obscure reason.
305 # MQ test show a corruption of the underlying repo (in _clcache)
304 # MQ test show a corruption of the underlying repo (in _clcache)
306 # without change in the cachekey.
305 # without change in the cachekey.
307 oldfilter = cl.filteredrevs
306 oldfilter = cl.filteredrevs
308 try:
307 try:
309 cl.filteredrevs = () # disable filtering for tip
308 cl.filteredrevs = () # disable filtering for tip
310 curkey = (len(cl), cl.tip(), hash(oldfilter), cl._delayed)
309 curkey = (len(cl), cl.tip(), hash(oldfilter), cl._delayed)
311 finally:
310 finally:
312 cl.filteredrevs = oldfilter
311 cl.filteredrevs = oldfilter
313 if newkey != self._clcachekey or newkey != curkey:
312 if newkey != self._clcachekey or newkey != curkey:
314 cl = None
313 cl = None
315 # could have been made None by the previous if
314 # could have been made None by the previous if
316 if cl is None:
315 if cl is None:
317 cl = copy.copy(unfichangelog)
316 cl = copy.copy(unfichangelog)
318 cl.filteredrevs = revs
317 cl.filteredrevs = revs
319 object.__setattr__(self, '_clcache', cl)
318 object.__setattr__(self, '_clcache', cl)
320 object.__setattr__(self, '_clcachekey', newkey)
319 object.__setattr__(self, '_clcachekey', newkey)
321 return cl
320 return cl
322
321
323 def unfiltered(self):
322 def unfiltered(self):
324 """Return an unfiltered version of a repo"""
323 """Return an unfiltered version of a repo"""
325 return self._unfilteredrepo
324 return self._unfilteredrepo
326
325
327 def filtered(self, name):
326 def filtered(self, name):
328 """Return a filtered version of a repository"""
327 """Return a filtered version of a repository"""
329 if name == self.filtername:
328 if name == self.filtername:
330 return self
329 return self
331 return self.unfiltered().filtered(name)
330 return self.unfiltered().filtered(name)
332
331
333 # everything access are forwarded to the proxied repo
332 # everything access are forwarded to the proxied repo
334 def __getattr__(self, attr):
333 def __getattr__(self, attr):
335 return getattr(self._unfilteredrepo, attr)
334 return getattr(self._unfilteredrepo, attr)
336
335
337 def __setattr__(self, attr, value):
336 def __setattr__(self, attr, value):
338 return setattr(self._unfilteredrepo, attr, value)
337 return setattr(self._unfilteredrepo, attr, value)
339
338
340 def __delattr__(self, attr):
339 def __delattr__(self, attr):
341 return delattr(self._unfilteredrepo, attr)
340 return delattr(self._unfilteredrepo, attr)
342
341
343 # The `requirements` attribute is initialized during __init__. But
342 # The `requirements` attribute is initialized during __init__. But
344 # __getattr__ won't be called as it also exists on the class. We need
343 # __getattr__ won't be called as it also exists on the class. We need
345 # explicit forwarding to main repo here
344 # explicit forwarding to main repo here
346 @property
345 @property
347 def requirements(self):
346 def requirements(self):
348 return self._unfilteredrepo.requirements
347 return self._unfilteredrepo.requirements
General Comments 0
You need to be logged in to leave comments. Login now