##// END OF EJS Templates
index: use `index.get_rev` in `unionrepo.unionrevlog`...
marmoute -
r43872:10f53873 default draft
parent child Browse files
Show More
@@ -1,285 +1,286 b''
1 1 # unionrepo.py - repository class for viewing union of repository changesets
2 2 #
3 3 # Derived from bundlerepo.py
4 4 # Copyright 2006, 2007 Benoit Boissinot <bboissin@gmail.com>
5 5 # Copyright 2013 Unity Technologies, Mads Kiilerich <madski@unity3d.com>
6 6 #
7 7 # This software may be used and distributed according to the terms of the
8 8 # GNU General Public License version 2 or any later version.
9 9
10 10 """Repository class for "in-memory pull" of one local repository to another,
11 11 allowing operations like diff and log with revsets.
12 12 """
13 13
14 14 from __future__ import absolute_import
15 15
16 16 from .i18n import _
17 17 from .pycompat import getattr
18 18
19 19 from . import (
20 20 changelog,
21 21 cmdutil,
22 22 encoding,
23 23 error,
24 24 filelog,
25 25 localrepo,
26 26 manifest,
27 27 mdiff,
28 28 pathutil,
29 29 revlog,
30 30 util,
31 31 vfs as vfsmod,
32 32 )
33 33
34 34
35 35 class unionrevlog(revlog.revlog):
36 36 def __init__(self, opener, indexfile, revlog2, linkmapper):
37 37 # How it works:
38 38 # To retrieve a revision, we just need to know the node id so we can
39 39 # look it up in revlog2.
40 40 #
41 41 # To differentiate a rev in the second revlog from a rev in the revlog,
42 42 # we check revision against repotiprev.
43 43 opener = vfsmod.readonlyvfs(opener)
44 44 revlog.revlog.__init__(self, opener, indexfile)
45 45 self.revlog2 = revlog2
46 46
47 47 n = len(self)
48 48 self.repotiprev = n - 1
49 49 self.bundlerevs = set() # used by 'bundle()' revset expression
50 50 for rev2 in self.revlog2:
51 51 rev = self.revlog2.index[rev2]
52 52 # rev numbers - in revlog2, very different from self.rev
53 53 _start, _csize, rsize, base, linkrev, p1rev, p2rev, node = rev
54 54 flags = _start & 0xFFFF
55 55
56 56 if linkmapper is None: # link is to same revlog
57 57 assert linkrev == rev2 # we never link back
58 58 link = n
59 59 else: # rev must be mapped from repo2 cl to unified cl by linkmapper
60 60 link = linkmapper(linkrev)
61 61
62 62 if linkmapper is not None: # link is to same revlog
63 63 base = linkmapper(base)
64 64
65 if node in self.nodemap:
65 this_rev = self.index.get_rev(node)
66 if this_rev is not None:
66 67 # this happens for the common revlog revisions
67 self.bundlerevs.add(self.nodemap[node])
68 self.bundlerevs.add(this_rev)
68 69 continue
69 70
70 71 p1node = self.revlog2.node(p1rev)
71 72 p2node = self.revlog2.node(p2rev)
72 73
73 74 # TODO: it's probably wrong to set compressed length to None, but
74 75 # I have no idea if csize is valid in the base revlog context.
75 76 e = (
76 77 flags,
77 78 None,
78 79 rsize,
79 80 base,
80 81 link,
81 82 self.rev(p1node),
82 83 self.rev(p2node),
83 84 node,
84 85 )
85 86 self.index.append(e)
86 87 self.bundlerevs.add(n)
87 88 n += 1
88 89
89 90 def _chunk(self, rev):
90 91 if rev <= self.repotiprev:
91 92 return revlog.revlog._chunk(self, rev)
92 93 return self.revlog2._chunk(self.node(rev))
93 94
94 95 def revdiff(self, rev1, rev2):
95 96 """return or calculate a delta between two revisions"""
96 97 if rev1 > self.repotiprev and rev2 > self.repotiprev:
97 98 return self.revlog2.revdiff(
98 99 self.revlog2.rev(self.node(rev1)),
99 100 self.revlog2.rev(self.node(rev2)),
100 101 )
101 102 elif rev1 <= self.repotiprev and rev2 <= self.repotiprev:
102 103 return super(unionrevlog, self).revdiff(rev1, rev2)
103 104
104 105 return mdiff.textdiff(self.rawdata(rev1), self.rawdata(rev2))
105 106
106 107 def _revisiondata(self, nodeorrev, _df=None, raw=False):
107 108 if isinstance(nodeorrev, int):
108 109 rev = nodeorrev
109 110 node = self.node(rev)
110 111 else:
111 112 node = nodeorrev
112 113 rev = self.rev(node)
113 114
114 115 if rev > self.repotiprev:
115 116 # work around manifestrevlog NOT being a revlog
116 117 revlog2 = getattr(self.revlog2, '_revlog', self.revlog2)
117 118 func = revlog2._revisiondata
118 119 else:
119 120 func = super(unionrevlog, self)._revisiondata
120 121 return func(node, _df=_df, raw=raw)
121 122
122 123 def addrevision(self, text, transaction, link, p1=None, p2=None, d=None):
123 124 raise NotImplementedError
124 125
125 126 def addgroup(
126 127 self,
127 128 deltas,
128 129 linkmapper,
129 130 transaction,
130 131 addrevisioncb=None,
131 132 maybemissingparents=False,
132 133 ):
133 134 raise NotImplementedError
134 135
135 136 def strip(self, minlink, transaction):
136 137 raise NotImplementedError
137 138
138 139 def checksize(self):
139 140 raise NotImplementedError
140 141
141 142
142 143 class unionchangelog(unionrevlog, changelog.changelog):
143 144 def __init__(self, opener, opener2):
144 145 changelog.changelog.__init__(self, opener)
145 146 linkmapper = None
146 147 changelog2 = changelog.changelog(opener2)
147 148 unionrevlog.__init__(
148 149 self, opener, self.indexfile, changelog2, linkmapper
149 150 )
150 151
151 152
152 153 class unionmanifest(unionrevlog, manifest.manifestrevlog):
153 154 def __init__(self, opener, opener2, linkmapper):
154 155 manifest.manifestrevlog.__init__(self, opener)
155 156 manifest2 = manifest.manifestrevlog(opener2)
156 157 unionrevlog.__init__(
157 158 self, opener, self.indexfile, manifest2, linkmapper
158 159 )
159 160
160 161
161 162 class unionfilelog(filelog.filelog):
162 163 def __init__(self, opener, path, opener2, linkmapper, repo):
163 164 filelog.filelog.__init__(self, opener, path)
164 165 filelog2 = filelog.filelog(opener2, path)
165 166 self._revlog = unionrevlog(
166 167 opener, self.indexfile, filelog2._revlog, linkmapper
167 168 )
168 169 self._repo = repo
169 170 self.repotiprev = self._revlog.repotiprev
170 171 self.revlog2 = self._revlog.revlog2
171 172
172 173 def iscensored(self, rev):
173 174 """Check if a revision is censored."""
174 175 if rev <= self.repotiprev:
175 176 return filelog.filelog.iscensored(self, rev)
176 177 node = self.node(rev)
177 178 return self.revlog2.iscensored(self.revlog2.rev(node))
178 179
179 180
180 181 class unionpeer(localrepo.localpeer):
181 182 def canpush(self):
182 183 return False
183 184
184 185
185 186 class unionrepository(object):
186 187 """Represents the union of data in 2 repositories.
187 188
188 189 Instances are not usable if constructed directly. Use ``instance()``
189 190 or ``makeunionrepository()`` to create a usable instance.
190 191 """
191 192
192 193 def __init__(self, repo2, url):
193 194 self.repo2 = repo2
194 195 self._url = url
195 196
196 197 self.ui.setconfig(b'phases', b'publish', False, b'unionrepo')
197 198
198 199 @localrepo.unfilteredpropertycache
199 200 def changelog(self):
200 201 return unionchangelog(self.svfs, self.repo2.svfs)
201 202
202 203 @localrepo.unfilteredpropertycache
203 204 def manifestlog(self):
204 205 rootstore = unionmanifest(
205 206 self.svfs, self.repo2.svfs, self.unfiltered()._clrev
206 207 )
207 208 return manifest.manifestlog(
208 209 self.svfs, self, rootstore, self.narrowmatch()
209 210 )
210 211
211 212 def _clrev(self, rev2):
212 213 """map from repo2 changelog rev to temporary rev in self.changelog"""
213 214 node = self.repo2.changelog.node(rev2)
214 215 return self.changelog.rev(node)
215 216
216 217 def url(self):
217 218 return self._url
218 219
219 220 def file(self, f):
220 221 return unionfilelog(
221 222 self.svfs, f, self.repo2.svfs, self.unfiltered()._clrev, self
222 223 )
223 224
224 225 def close(self):
225 226 self.repo2.close()
226 227
227 228 def cancopy(self):
228 229 return False
229 230
230 231 def peer(self):
231 232 return unionpeer(self)
232 233
233 234 def getcwd(self):
234 235 return encoding.getcwd() # always outside the repo
235 236
236 237
237 238 def instance(ui, path, create, intents=None, createopts=None):
238 239 if create:
239 240 raise error.Abort(_(b'cannot create new union repository'))
240 241 parentpath = ui.config(b"bundle", b"mainreporoot")
241 242 if not parentpath:
242 243 # try to find the correct path to the working directory repo
243 244 parentpath = cmdutil.findrepo(encoding.getcwd())
244 245 if parentpath is None:
245 246 parentpath = b''
246 247 if parentpath:
247 248 # Try to make the full path relative so we get a nice, short URL.
248 249 # In particular, we don't want temp dir names in test outputs.
249 250 cwd = encoding.getcwd()
250 251 if parentpath == cwd:
251 252 parentpath = b''
252 253 else:
253 254 cwd = pathutil.normasprefix(cwd)
254 255 if parentpath.startswith(cwd):
255 256 parentpath = parentpath[len(cwd) :]
256 257 if path.startswith(b'union:'):
257 258 s = path.split(b":", 1)[1].split(b"+", 1)
258 259 if len(s) == 1:
259 260 repopath, repopath2 = parentpath, s[0]
260 261 else:
261 262 repopath, repopath2 = s
262 263 else:
263 264 repopath, repopath2 = parentpath, path
264 265
265 266 return makeunionrepository(ui, repopath, repopath2)
266 267
267 268
268 269 def makeunionrepository(ui, repopath1, repopath2):
269 270 """Make a union repository object from 2 local repo paths."""
270 271 repo1 = localrepo.instance(ui, repopath1, create=False)
271 272 repo2 = localrepo.instance(ui, repopath2, create=False)
272 273
273 274 url = b'union:%s+%s' % (
274 275 util.expandpath(repopath1),
275 276 util.expandpath(repopath2),
276 277 )
277 278
278 279 class derivedunionrepository(unionrepository, repo1.__class__):
279 280 pass
280 281
281 282 repo = repo1
282 283 repo.__class__ = derivedunionrepository
283 284 unionrepository.__init__(repo1, repo2, url)
284 285
285 286 return repo
General Comments 0
You need to be logged in to leave comments. Login now