##// END OF EJS Templates
largefiles: preserve the exit status of the log command
Matt Harbison -
r17577:0f39e935 stable
parent child Browse files
Show More
@@ -1,1087 +1,1087 b''
1 # Copyright 2009-2010 Gregory P. Ward
1 # Copyright 2009-2010 Gregory P. Ward
2 # Copyright 2009-2010 Intelerad Medical Systems Incorporated
2 # Copyright 2009-2010 Intelerad Medical Systems Incorporated
3 # Copyright 2010-2011 Fog Creek Software
3 # Copyright 2010-2011 Fog Creek Software
4 # Copyright 2010-2011 Unity Technologies
4 # Copyright 2010-2011 Unity Technologies
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 '''Overridden Mercurial commands and functions for the largefiles extension'''
9 '''Overridden Mercurial commands and functions for the largefiles extension'''
10
10
11 import os
11 import os
12 import copy
12 import copy
13
13
14 from mercurial import hg, commands, util, cmdutil, scmutil, match as match_, \
14 from mercurial import hg, commands, util, cmdutil, scmutil, match as match_, \
15 node, archival, error, merge
15 node, archival, error, merge
16 from mercurial.i18n import _
16 from mercurial.i18n import _
17 from mercurial.node import hex
17 from mercurial.node import hex
18 from hgext import rebase
18 from hgext import rebase
19
19
20 import lfutil
20 import lfutil
21 import lfcommands
21 import lfcommands
22
22
23 # -- Utility functions: commonly/repeatedly needed functionality ---------------
23 # -- Utility functions: commonly/repeatedly needed functionality ---------------
24
24
25 def installnormalfilesmatchfn(manifest):
25 def installnormalfilesmatchfn(manifest):
26 '''overrides scmutil.match so that the matcher it returns will ignore all
26 '''overrides scmutil.match so that the matcher it returns will ignore all
27 largefiles'''
27 largefiles'''
28 oldmatch = None # for the closure
28 oldmatch = None # for the closure
29 def overridematch(ctx, pats=[], opts={}, globbed=False,
29 def overridematch(ctx, pats=[], opts={}, globbed=False,
30 default='relpath'):
30 default='relpath'):
31 match = oldmatch(ctx, pats, opts, globbed, default)
31 match = oldmatch(ctx, pats, opts, globbed, default)
32 m = copy.copy(match)
32 m = copy.copy(match)
33 notlfile = lambda f: not (lfutil.isstandin(f) or lfutil.standin(f) in
33 notlfile = lambda f: not (lfutil.isstandin(f) or lfutil.standin(f) in
34 manifest)
34 manifest)
35 m._files = filter(notlfile, m._files)
35 m._files = filter(notlfile, m._files)
36 m._fmap = set(m._files)
36 m._fmap = set(m._files)
37 origmatchfn = m.matchfn
37 origmatchfn = m.matchfn
38 m.matchfn = lambda f: notlfile(f) and origmatchfn(f) or None
38 m.matchfn = lambda f: notlfile(f) and origmatchfn(f) or None
39 return m
39 return m
40 oldmatch = installmatchfn(overridematch)
40 oldmatch = installmatchfn(overridematch)
41
41
42 def installmatchfn(f):
42 def installmatchfn(f):
43 oldmatch = scmutil.match
43 oldmatch = scmutil.match
44 setattr(f, 'oldmatch', oldmatch)
44 setattr(f, 'oldmatch', oldmatch)
45 scmutil.match = f
45 scmutil.match = f
46 return oldmatch
46 return oldmatch
47
47
48 def restorematchfn():
48 def restorematchfn():
49 '''restores scmutil.match to what it was before installnormalfilesmatchfn
49 '''restores scmutil.match to what it was before installnormalfilesmatchfn
50 was called. no-op if scmutil.match is its original function.
50 was called. no-op if scmutil.match is its original function.
51
51
52 Note that n calls to installnormalfilesmatchfn will require n calls to
52 Note that n calls to installnormalfilesmatchfn will require n calls to
53 restore matchfn to reverse'''
53 restore matchfn to reverse'''
54 scmutil.match = getattr(scmutil.match, 'oldmatch', scmutil.match)
54 scmutil.match = getattr(scmutil.match, 'oldmatch', scmutil.match)
55
55
56 def addlargefiles(ui, repo, *pats, **opts):
56 def addlargefiles(ui, repo, *pats, **opts):
57 large = opts.pop('large', None)
57 large = opts.pop('large', None)
58 lfsize = lfutil.getminsize(
58 lfsize = lfutil.getminsize(
59 ui, lfutil.islfilesrepo(repo), opts.pop('lfsize', None))
59 ui, lfutil.islfilesrepo(repo), opts.pop('lfsize', None))
60
60
61 lfmatcher = None
61 lfmatcher = None
62 if lfutil.islfilesrepo(repo):
62 if lfutil.islfilesrepo(repo):
63 lfpats = ui.configlist(lfutil.longname, 'patterns', default=[])
63 lfpats = ui.configlist(lfutil.longname, 'patterns', default=[])
64 if lfpats:
64 if lfpats:
65 lfmatcher = match_.match(repo.root, '', list(lfpats))
65 lfmatcher = match_.match(repo.root, '', list(lfpats))
66
66
67 lfnames = []
67 lfnames = []
68 m = scmutil.match(repo[None], pats, opts)
68 m = scmutil.match(repo[None], pats, opts)
69 m.bad = lambda x, y: None
69 m.bad = lambda x, y: None
70 wctx = repo[None]
70 wctx = repo[None]
71 for f in repo.walk(m):
71 for f in repo.walk(m):
72 exact = m.exact(f)
72 exact = m.exact(f)
73 lfile = lfutil.standin(f) in wctx
73 lfile = lfutil.standin(f) in wctx
74 nfile = f in wctx
74 nfile = f in wctx
75 exists = lfile or nfile
75 exists = lfile or nfile
76
76
77 # Don't warn the user when they attempt to add a normal tracked file.
77 # Don't warn the user when they attempt to add a normal tracked file.
78 # The normal add code will do that for us.
78 # The normal add code will do that for us.
79 if exact and exists:
79 if exact and exists:
80 if lfile:
80 if lfile:
81 ui.warn(_('%s already a largefile\n') % f)
81 ui.warn(_('%s already a largefile\n') % f)
82 continue
82 continue
83
83
84 if (exact or not exists) and not lfutil.isstandin(f):
84 if (exact or not exists) and not lfutil.isstandin(f):
85 wfile = repo.wjoin(f)
85 wfile = repo.wjoin(f)
86
86
87 # In case the file was removed previously, but not committed
87 # In case the file was removed previously, but not committed
88 # (issue3507)
88 # (issue3507)
89 if not os.path.exists(wfile):
89 if not os.path.exists(wfile):
90 continue
90 continue
91
91
92 abovemin = (lfsize and
92 abovemin = (lfsize and
93 os.lstat(wfile).st_size >= lfsize * 1024 * 1024)
93 os.lstat(wfile).st_size >= lfsize * 1024 * 1024)
94 if large or abovemin or (lfmatcher and lfmatcher(f)):
94 if large or abovemin or (lfmatcher and lfmatcher(f)):
95 lfnames.append(f)
95 lfnames.append(f)
96 if ui.verbose or not exact:
96 if ui.verbose or not exact:
97 ui.status(_('adding %s as a largefile\n') % m.rel(f))
97 ui.status(_('adding %s as a largefile\n') % m.rel(f))
98
98
99 bad = []
99 bad = []
100 standins = []
100 standins = []
101
101
102 # Need to lock, otherwise there could be a race condition between
102 # Need to lock, otherwise there could be a race condition between
103 # when standins are created and added to the repo.
103 # when standins are created and added to the repo.
104 wlock = repo.wlock()
104 wlock = repo.wlock()
105 try:
105 try:
106 if not opts.get('dry_run'):
106 if not opts.get('dry_run'):
107 lfdirstate = lfutil.openlfdirstate(ui, repo)
107 lfdirstate = lfutil.openlfdirstate(ui, repo)
108 for f in lfnames:
108 for f in lfnames:
109 standinname = lfutil.standin(f)
109 standinname = lfutil.standin(f)
110 lfutil.writestandin(repo, standinname, hash='',
110 lfutil.writestandin(repo, standinname, hash='',
111 executable=lfutil.getexecutable(repo.wjoin(f)))
111 executable=lfutil.getexecutable(repo.wjoin(f)))
112 standins.append(standinname)
112 standins.append(standinname)
113 if lfdirstate[f] == 'r':
113 if lfdirstate[f] == 'r':
114 lfdirstate.normallookup(f)
114 lfdirstate.normallookup(f)
115 else:
115 else:
116 lfdirstate.add(f)
116 lfdirstate.add(f)
117 lfdirstate.write()
117 lfdirstate.write()
118 bad += [lfutil.splitstandin(f)
118 bad += [lfutil.splitstandin(f)
119 for f in lfutil.repoadd(repo, standins)
119 for f in lfutil.repoadd(repo, standins)
120 if f in m.files()]
120 if f in m.files()]
121 finally:
121 finally:
122 wlock.release()
122 wlock.release()
123 return bad
123 return bad
124
124
125 def removelargefiles(ui, repo, *pats, **opts):
125 def removelargefiles(ui, repo, *pats, **opts):
126 after = opts.get('after')
126 after = opts.get('after')
127 if not pats and not after:
127 if not pats and not after:
128 raise util.Abort(_('no files specified'))
128 raise util.Abort(_('no files specified'))
129 m = scmutil.match(repo[None], pats, opts)
129 m = scmutil.match(repo[None], pats, opts)
130 try:
130 try:
131 repo.lfstatus = True
131 repo.lfstatus = True
132 s = repo.status(match=m, clean=True)
132 s = repo.status(match=m, clean=True)
133 finally:
133 finally:
134 repo.lfstatus = False
134 repo.lfstatus = False
135 manifest = repo[None].manifest()
135 manifest = repo[None].manifest()
136 modified, added, deleted, clean = [[f for f in list
136 modified, added, deleted, clean = [[f for f in list
137 if lfutil.standin(f) in manifest]
137 if lfutil.standin(f) in manifest]
138 for list in [s[0], s[1], s[3], s[6]]]
138 for list in [s[0], s[1], s[3], s[6]]]
139
139
140 def warn(files, reason):
140 def warn(files, reason):
141 for f in files:
141 for f in files:
142 ui.warn(_('not removing %s: %s (use forget to undo)\n')
142 ui.warn(_('not removing %s: %s (use forget to undo)\n')
143 % (m.rel(f), reason))
143 % (m.rel(f), reason))
144 return int(len(files) > 0)
144 return int(len(files) > 0)
145
145
146 result = 0
146 result = 0
147
147
148 if after:
148 if after:
149 remove, forget = deleted, []
149 remove, forget = deleted, []
150 result = warn(modified + added + clean, _('file still exists'))
150 result = warn(modified + added + clean, _('file still exists'))
151 else:
151 else:
152 remove, forget = deleted + clean, []
152 remove, forget = deleted + clean, []
153 result = warn(modified, _('file is modified'))
153 result = warn(modified, _('file is modified'))
154 result = warn(added, _('file has been marked for add')) or result
154 result = warn(added, _('file has been marked for add')) or result
155
155
156 for f in sorted(remove + forget):
156 for f in sorted(remove + forget):
157 if ui.verbose or not m.exact(f):
157 if ui.verbose or not m.exact(f):
158 ui.status(_('removing %s\n') % m.rel(f))
158 ui.status(_('removing %s\n') % m.rel(f))
159
159
160 # Need to lock because standin files are deleted then removed from the
160 # Need to lock because standin files are deleted then removed from the
161 # repository and we could race inbetween.
161 # repository and we could race inbetween.
162 wlock = repo.wlock()
162 wlock = repo.wlock()
163 try:
163 try:
164 lfdirstate = lfutil.openlfdirstate(ui, repo)
164 lfdirstate = lfutil.openlfdirstate(ui, repo)
165 for f in remove:
165 for f in remove:
166 if not after:
166 if not after:
167 # If this is being called by addremove, notify the user that we
167 # If this is being called by addremove, notify the user that we
168 # are removing the file.
168 # are removing the file.
169 if getattr(repo, "_isaddremove", False):
169 if getattr(repo, "_isaddremove", False):
170 ui.status(_('removing %s\n') % f)
170 ui.status(_('removing %s\n') % f)
171 if os.path.exists(repo.wjoin(f)):
171 if os.path.exists(repo.wjoin(f)):
172 util.unlinkpath(repo.wjoin(f))
172 util.unlinkpath(repo.wjoin(f))
173 lfdirstate.remove(f)
173 lfdirstate.remove(f)
174 lfdirstate.write()
174 lfdirstate.write()
175 forget = [lfutil.standin(f) for f in forget]
175 forget = [lfutil.standin(f) for f in forget]
176 remove = [lfutil.standin(f) for f in remove]
176 remove = [lfutil.standin(f) for f in remove]
177 lfutil.repoforget(repo, forget)
177 lfutil.repoforget(repo, forget)
178 # If this is being called by addremove, let the original addremove
178 # If this is being called by addremove, let the original addremove
179 # function handle this.
179 # function handle this.
180 if not getattr(repo, "_isaddremove", False):
180 if not getattr(repo, "_isaddremove", False):
181 lfutil.reporemove(repo, remove, unlink=True)
181 lfutil.reporemove(repo, remove, unlink=True)
182 else:
182 else:
183 lfutil.reporemove(repo, remove, unlink=False)
183 lfutil.reporemove(repo, remove, unlink=False)
184 finally:
184 finally:
185 wlock.release()
185 wlock.release()
186
186
187 return result
187 return result
188
188
189 # For overriding mercurial.hgweb.webcommands so that largefiles will
189 # For overriding mercurial.hgweb.webcommands so that largefiles will
190 # appear at their right place in the manifests.
190 # appear at their right place in the manifests.
191 def decodepath(orig, path):
191 def decodepath(orig, path):
192 return lfutil.splitstandin(path) or path
192 return lfutil.splitstandin(path) or path
193
193
194 # -- Wrappers: modify existing commands --------------------------------
194 # -- Wrappers: modify existing commands --------------------------------
195
195
196 # Add works by going through the files that the user wanted to add and
196 # Add works by going through the files that the user wanted to add and
197 # checking if they should be added as largefiles. Then it makes a new
197 # checking if they should be added as largefiles. Then it makes a new
198 # matcher which matches only the normal files and runs the original
198 # matcher which matches only the normal files and runs the original
199 # version of add.
199 # version of add.
200 def overrideadd(orig, ui, repo, *pats, **opts):
200 def overrideadd(orig, ui, repo, *pats, **opts):
201 normal = opts.pop('normal')
201 normal = opts.pop('normal')
202 if normal:
202 if normal:
203 if opts.get('large'):
203 if opts.get('large'):
204 raise util.Abort(_('--normal cannot be used with --large'))
204 raise util.Abort(_('--normal cannot be used with --large'))
205 return orig(ui, repo, *pats, **opts)
205 return orig(ui, repo, *pats, **opts)
206 bad = addlargefiles(ui, repo, *pats, **opts)
206 bad = addlargefiles(ui, repo, *pats, **opts)
207 installnormalfilesmatchfn(repo[None].manifest())
207 installnormalfilesmatchfn(repo[None].manifest())
208 result = orig(ui, repo, *pats, **opts)
208 result = orig(ui, repo, *pats, **opts)
209 restorematchfn()
209 restorematchfn()
210
210
211 return (result == 1 or bad) and 1 or 0
211 return (result == 1 or bad) and 1 or 0
212
212
213 def overrideremove(orig, ui, repo, *pats, **opts):
213 def overrideremove(orig, ui, repo, *pats, **opts):
214 installnormalfilesmatchfn(repo[None].manifest())
214 installnormalfilesmatchfn(repo[None].manifest())
215 result = orig(ui, repo, *pats, **opts)
215 result = orig(ui, repo, *pats, **opts)
216 restorematchfn()
216 restorematchfn()
217 return removelargefiles(ui, repo, *pats, **opts) or result
217 return removelargefiles(ui, repo, *pats, **opts) or result
218
218
219 def overridestatusfn(orig, repo, rev2, **opts):
219 def overridestatusfn(orig, repo, rev2, **opts):
220 try:
220 try:
221 repo._repo.lfstatus = True
221 repo._repo.lfstatus = True
222 return orig(repo, rev2, **opts)
222 return orig(repo, rev2, **opts)
223 finally:
223 finally:
224 repo._repo.lfstatus = False
224 repo._repo.lfstatus = False
225
225
226 def overridestatus(orig, ui, repo, *pats, **opts):
226 def overridestatus(orig, ui, repo, *pats, **opts):
227 try:
227 try:
228 repo.lfstatus = True
228 repo.lfstatus = True
229 return orig(ui, repo, *pats, **opts)
229 return orig(ui, repo, *pats, **opts)
230 finally:
230 finally:
231 repo.lfstatus = False
231 repo.lfstatus = False
232
232
233 def overridedirty(orig, repo, ignoreupdate=False):
233 def overridedirty(orig, repo, ignoreupdate=False):
234 try:
234 try:
235 repo._repo.lfstatus = True
235 repo._repo.lfstatus = True
236 return orig(repo, ignoreupdate)
236 return orig(repo, ignoreupdate)
237 finally:
237 finally:
238 repo._repo.lfstatus = False
238 repo._repo.lfstatus = False
239
239
240 def overridelog(orig, ui, repo, *pats, **opts):
240 def overridelog(orig, ui, repo, *pats, **opts):
241 try:
241 try:
242 repo.lfstatus = True
242 repo.lfstatus = True
243 orig(ui, repo, *pats, **opts)
243 return orig(ui, repo, *pats, **opts)
244 finally:
244 finally:
245 repo.lfstatus = False
245 repo.lfstatus = False
246
246
247 def overrideverify(orig, ui, repo, *pats, **opts):
247 def overrideverify(orig, ui, repo, *pats, **opts):
248 large = opts.pop('large', False)
248 large = opts.pop('large', False)
249 all = opts.pop('lfa', False)
249 all = opts.pop('lfa', False)
250 contents = opts.pop('lfc', False)
250 contents = opts.pop('lfc', False)
251
251
252 result = orig(ui, repo, *pats, **opts)
252 result = orig(ui, repo, *pats, **opts)
253 if large:
253 if large:
254 result = result or lfcommands.verifylfiles(ui, repo, all, contents)
254 result = result or lfcommands.verifylfiles(ui, repo, all, contents)
255 return result
255 return result
256
256
257 # Override needs to refresh standins so that update's normal merge
257 # Override needs to refresh standins so that update's normal merge
258 # will go through properly. Then the other update hook (overriding repo.update)
258 # will go through properly. Then the other update hook (overriding repo.update)
259 # will get the new files. Filemerge is also overriden so that the merge
259 # will get the new files. Filemerge is also overriden so that the merge
260 # will merge standins correctly.
260 # will merge standins correctly.
261 def overrideupdate(orig, ui, repo, *pats, **opts):
261 def overrideupdate(orig, ui, repo, *pats, **opts):
262 lfdirstate = lfutil.openlfdirstate(ui, repo)
262 lfdirstate = lfutil.openlfdirstate(ui, repo)
263 s = lfdirstate.status(match_.always(repo.root, repo.getcwd()), [], False,
263 s = lfdirstate.status(match_.always(repo.root, repo.getcwd()), [], False,
264 False, False)
264 False, False)
265 (unsure, modified, added, removed, missing, unknown, ignored, clean) = s
265 (unsure, modified, added, removed, missing, unknown, ignored, clean) = s
266
266
267 # Need to lock between the standins getting updated and their
267 # Need to lock between the standins getting updated and their
268 # largefiles getting updated
268 # largefiles getting updated
269 wlock = repo.wlock()
269 wlock = repo.wlock()
270 try:
270 try:
271 if opts['check']:
271 if opts['check']:
272 mod = len(modified) > 0
272 mod = len(modified) > 0
273 for lfile in unsure:
273 for lfile in unsure:
274 standin = lfutil.standin(lfile)
274 standin = lfutil.standin(lfile)
275 if repo['.'][standin].data().strip() != \
275 if repo['.'][standin].data().strip() != \
276 lfutil.hashfile(repo.wjoin(lfile)):
276 lfutil.hashfile(repo.wjoin(lfile)):
277 mod = True
277 mod = True
278 else:
278 else:
279 lfdirstate.normal(lfile)
279 lfdirstate.normal(lfile)
280 lfdirstate.write()
280 lfdirstate.write()
281 if mod:
281 if mod:
282 raise util.Abort(_('uncommitted local changes'))
282 raise util.Abort(_('uncommitted local changes'))
283 # XXX handle removed differently
283 # XXX handle removed differently
284 if not opts['clean']:
284 if not opts['clean']:
285 for lfile in unsure + modified + added:
285 for lfile in unsure + modified + added:
286 lfutil.updatestandin(repo, lfutil.standin(lfile))
286 lfutil.updatestandin(repo, lfutil.standin(lfile))
287 finally:
287 finally:
288 wlock.release()
288 wlock.release()
289 return orig(ui, repo, *pats, **opts)
289 return orig(ui, repo, *pats, **opts)
290
290
291 # Before starting the manifest merge, merge.updates will call
291 # Before starting the manifest merge, merge.updates will call
292 # _checkunknown to check if there are any files in the merged-in
292 # _checkunknown to check if there are any files in the merged-in
293 # changeset that collide with unknown files in the working copy.
293 # changeset that collide with unknown files in the working copy.
294 #
294 #
295 # The largefiles are seen as unknown, so this prevents us from merging
295 # The largefiles are seen as unknown, so this prevents us from merging
296 # in a file 'foo' if we already have a largefile with the same name.
296 # in a file 'foo' if we already have a largefile with the same name.
297 #
297 #
298 # The overridden function filters the unknown files by removing any
298 # The overridden function filters the unknown files by removing any
299 # largefiles. This makes the merge proceed and we can then handle this
299 # largefiles. This makes the merge proceed and we can then handle this
300 # case further in the overridden manifestmerge function below.
300 # case further in the overridden manifestmerge function below.
301 def overridecheckunknownfile(origfn, repo, wctx, mctx, f):
301 def overridecheckunknownfile(origfn, repo, wctx, mctx, f):
302 if lfutil.standin(f) in wctx:
302 if lfutil.standin(f) in wctx:
303 return False
303 return False
304 return origfn(repo, wctx, mctx, f)
304 return origfn(repo, wctx, mctx, f)
305
305
306 # The manifest merge handles conflicts on the manifest level. We want
306 # The manifest merge handles conflicts on the manifest level. We want
307 # to handle changes in largefile-ness of files at this level too.
307 # to handle changes in largefile-ness of files at this level too.
308 #
308 #
309 # The strategy is to run the original manifestmerge and then process
309 # The strategy is to run the original manifestmerge and then process
310 # the action list it outputs. There are two cases we need to deal with:
310 # the action list it outputs. There are two cases we need to deal with:
311 #
311 #
312 # 1. Normal file in p1, largefile in p2. Here the largefile is
312 # 1. Normal file in p1, largefile in p2. Here the largefile is
313 # detected via its standin file, which will enter the working copy
313 # detected via its standin file, which will enter the working copy
314 # with a "get" action. It is not "merge" since the standin is all
314 # with a "get" action. It is not "merge" since the standin is all
315 # Mercurial is concerned with at this level -- the link to the
315 # Mercurial is concerned with at this level -- the link to the
316 # existing normal file is not relevant here.
316 # existing normal file is not relevant here.
317 #
317 #
318 # 2. Largefile in p1, normal file in p2. Here we get a "merge" action
318 # 2. Largefile in p1, normal file in p2. Here we get a "merge" action
319 # since the largefile will be present in the working copy and
319 # since the largefile will be present in the working copy and
320 # different from the normal file in p2. Mercurial therefore
320 # different from the normal file in p2. Mercurial therefore
321 # triggers a merge action.
321 # triggers a merge action.
322 #
322 #
323 # In both cases, we prompt the user and emit new actions to either
323 # In both cases, we prompt the user and emit new actions to either
324 # remove the standin (if the normal file was kept) or to remove the
324 # remove the standin (if the normal file was kept) or to remove the
325 # normal file and get the standin (if the largefile was kept). The
325 # normal file and get the standin (if the largefile was kept). The
326 # default prompt answer is to use the largefile version since it was
326 # default prompt answer is to use the largefile version since it was
327 # presumably changed on purpose.
327 # presumably changed on purpose.
328 #
328 #
329 # Finally, the merge.applyupdates function will then take care of
329 # Finally, the merge.applyupdates function will then take care of
330 # writing the files into the working copy and lfcommands.updatelfiles
330 # writing the files into the working copy and lfcommands.updatelfiles
331 # will update the largefiles.
331 # will update the largefiles.
332 def overridemanifestmerge(origfn, repo, p1, p2, pa, overwrite, partial):
332 def overridemanifestmerge(origfn, repo, p1, p2, pa, overwrite, partial):
333 actions = origfn(repo, p1, p2, pa, overwrite, partial)
333 actions = origfn(repo, p1, p2, pa, overwrite, partial)
334 processed = []
334 processed = []
335
335
336 for action in actions:
336 for action in actions:
337 if overwrite:
337 if overwrite:
338 processed.append(action)
338 processed.append(action)
339 continue
339 continue
340 f, m = action[:2]
340 f, m = action[:2]
341
341
342 choices = (_('&Largefile'), _('&Normal file'))
342 choices = (_('&Largefile'), _('&Normal file'))
343 if m == "g" and lfutil.splitstandin(f) in p1 and f in p2:
343 if m == "g" and lfutil.splitstandin(f) in p1 and f in p2:
344 # Case 1: normal file in the working copy, largefile in
344 # Case 1: normal file in the working copy, largefile in
345 # the second parent
345 # the second parent
346 lfile = lfutil.splitstandin(f)
346 lfile = lfutil.splitstandin(f)
347 standin = f
347 standin = f
348 msg = _('%s has been turned into a largefile\n'
348 msg = _('%s has been turned into a largefile\n'
349 'use (l)argefile or keep as (n)ormal file?') % lfile
349 'use (l)argefile or keep as (n)ormal file?') % lfile
350 if repo.ui.promptchoice(msg, choices, 0) == 0:
350 if repo.ui.promptchoice(msg, choices, 0) == 0:
351 processed.append((lfile, "r"))
351 processed.append((lfile, "r"))
352 processed.append((standin, "g", p2.flags(standin)))
352 processed.append((standin, "g", p2.flags(standin)))
353 else:
353 else:
354 processed.append((standin, "r"))
354 processed.append((standin, "r"))
355 elif m == "g" and lfutil.standin(f) in p1 and f in p2:
355 elif m == "g" and lfutil.standin(f) in p1 and f in p2:
356 # Case 2: largefile in the working copy, normal file in
356 # Case 2: largefile in the working copy, normal file in
357 # the second parent
357 # the second parent
358 standin = lfutil.standin(f)
358 standin = lfutil.standin(f)
359 lfile = f
359 lfile = f
360 msg = _('%s has been turned into a normal file\n'
360 msg = _('%s has been turned into a normal file\n'
361 'keep as (l)argefile or use (n)ormal file?') % lfile
361 'keep as (l)argefile or use (n)ormal file?') % lfile
362 if repo.ui.promptchoice(msg, choices, 0) == 0:
362 if repo.ui.promptchoice(msg, choices, 0) == 0:
363 processed.append((lfile, "r"))
363 processed.append((lfile, "r"))
364 else:
364 else:
365 processed.append((standin, "r"))
365 processed.append((standin, "r"))
366 processed.append((lfile, "g", p2.flags(lfile)))
366 processed.append((lfile, "g", p2.flags(lfile)))
367 else:
367 else:
368 processed.append(action)
368 processed.append(action)
369
369
370 return processed
370 return processed
371
371
372 # Override filemerge to prompt the user about how they wish to merge
372 # Override filemerge to prompt the user about how they wish to merge
373 # largefiles. This will handle identical edits, and copy/rename +
373 # largefiles. This will handle identical edits, and copy/rename +
374 # edit without prompting the user.
374 # edit without prompting the user.
375 def overridefilemerge(origfn, repo, mynode, orig, fcd, fco, fca):
375 def overridefilemerge(origfn, repo, mynode, orig, fcd, fco, fca):
376 # Use better variable names here. Because this is a wrapper we cannot
376 # Use better variable names here. Because this is a wrapper we cannot
377 # change the variable names in the function declaration.
377 # change the variable names in the function declaration.
378 fcdest, fcother, fcancestor = fcd, fco, fca
378 fcdest, fcother, fcancestor = fcd, fco, fca
379 if not lfutil.isstandin(orig):
379 if not lfutil.isstandin(orig):
380 return origfn(repo, mynode, orig, fcdest, fcother, fcancestor)
380 return origfn(repo, mynode, orig, fcdest, fcother, fcancestor)
381 else:
381 else:
382 if not fcother.cmp(fcdest): # files identical?
382 if not fcother.cmp(fcdest): # files identical?
383 return None
383 return None
384
384
385 # backwards, use working dir parent as ancestor
385 # backwards, use working dir parent as ancestor
386 if fcancestor == fcother:
386 if fcancestor == fcother:
387 fcancestor = fcdest.parents()[0]
387 fcancestor = fcdest.parents()[0]
388
388
389 if orig != fcother.path():
389 if orig != fcother.path():
390 repo.ui.status(_('merging %s and %s to %s\n')
390 repo.ui.status(_('merging %s and %s to %s\n')
391 % (lfutil.splitstandin(orig),
391 % (lfutil.splitstandin(orig),
392 lfutil.splitstandin(fcother.path()),
392 lfutil.splitstandin(fcother.path()),
393 lfutil.splitstandin(fcdest.path())))
393 lfutil.splitstandin(fcdest.path())))
394 else:
394 else:
395 repo.ui.status(_('merging %s\n')
395 repo.ui.status(_('merging %s\n')
396 % lfutil.splitstandin(fcdest.path()))
396 % lfutil.splitstandin(fcdest.path()))
397
397
398 if fcancestor.path() != fcother.path() and fcother.data() == \
398 if fcancestor.path() != fcother.path() and fcother.data() == \
399 fcancestor.data():
399 fcancestor.data():
400 return 0
400 return 0
401 if fcancestor.path() != fcdest.path() and fcdest.data() == \
401 if fcancestor.path() != fcdest.path() and fcdest.data() == \
402 fcancestor.data():
402 fcancestor.data():
403 repo.wwrite(fcdest.path(), fcother.data(), fcother.flags())
403 repo.wwrite(fcdest.path(), fcother.data(), fcother.flags())
404 return 0
404 return 0
405
405
406 if repo.ui.promptchoice(_('largefile %s has a merge conflict\n'
406 if repo.ui.promptchoice(_('largefile %s has a merge conflict\n'
407 'keep (l)ocal or take (o)ther?') %
407 'keep (l)ocal or take (o)ther?') %
408 lfutil.splitstandin(orig),
408 lfutil.splitstandin(orig),
409 (_('&Local'), _('&Other')), 0) == 0:
409 (_('&Local'), _('&Other')), 0) == 0:
410 return 0
410 return 0
411 else:
411 else:
412 repo.wwrite(fcdest.path(), fcother.data(), fcother.flags())
412 repo.wwrite(fcdest.path(), fcother.data(), fcother.flags())
413 return 0
413 return 0
414
414
415 # Copy first changes the matchers to match standins instead of
415 # Copy first changes the matchers to match standins instead of
416 # largefiles. Then it overrides util.copyfile in that function it
416 # largefiles. Then it overrides util.copyfile in that function it
417 # checks if the destination largefile already exists. It also keeps a
417 # checks if the destination largefile already exists. It also keeps a
418 # list of copied files so that the largefiles can be copied and the
418 # list of copied files so that the largefiles can be copied and the
419 # dirstate updated.
419 # dirstate updated.
420 def overridecopy(orig, ui, repo, pats, opts, rename=False):
420 def overridecopy(orig, ui, repo, pats, opts, rename=False):
421 # doesn't remove largefile on rename
421 # doesn't remove largefile on rename
422 if len(pats) < 2:
422 if len(pats) < 2:
423 # this isn't legal, let the original function deal with it
423 # this isn't legal, let the original function deal with it
424 return orig(ui, repo, pats, opts, rename)
424 return orig(ui, repo, pats, opts, rename)
425
425
426 def makestandin(relpath):
426 def makestandin(relpath):
427 path = scmutil.canonpath(repo.root, repo.getcwd(), relpath)
427 path = scmutil.canonpath(repo.root, repo.getcwd(), relpath)
428 return os.path.join(repo.wjoin(lfutil.standin(path)))
428 return os.path.join(repo.wjoin(lfutil.standin(path)))
429
429
430 fullpats = scmutil.expandpats(pats)
430 fullpats = scmutil.expandpats(pats)
431 dest = fullpats[-1]
431 dest = fullpats[-1]
432
432
433 if os.path.isdir(dest):
433 if os.path.isdir(dest):
434 if not os.path.isdir(makestandin(dest)):
434 if not os.path.isdir(makestandin(dest)):
435 os.makedirs(makestandin(dest))
435 os.makedirs(makestandin(dest))
436 # This could copy both lfiles and normal files in one command,
436 # This could copy both lfiles and normal files in one command,
437 # but we don't want to do that. First replace their matcher to
437 # but we don't want to do that. First replace their matcher to
438 # only match normal files and run it, then replace it to just
438 # only match normal files and run it, then replace it to just
439 # match largefiles and run it again.
439 # match largefiles and run it again.
440 nonormalfiles = False
440 nonormalfiles = False
441 nolfiles = False
441 nolfiles = False
442 try:
442 try:
443 try:
443 try:
444 installnormalfilesmatchfn(repo[None].manifest())
444 installnormalfilesmatchfn(repo[None].manifest())
445 result = orig(ui, repo, pats, opts, rename)
445 result = orig(ui, repo, pats, opts, rename)
446 except util.Abort, e:
446 except util.Abort, e:
447 if str(e) != _('no files to copy'):
447 if str(e) != _('no files to copy'):
448 raise e
448 raise e
449 else:
449 else:
450 nonormalfiles = True
450 nonormalfiles = True
451 result = 0
451 result = 0
452 finally:
452 finally:
453 restorematchfn()
453 restorematchfn()
454
454
455 # The first rename can cause our current working directory to be removed.
455 # The first rename can cause our current working directory to be removed.
456 # In that case there is nothing left to copy/rename so just quit.
456 # In that case there is nothing left to copy/rename so just quit.
457 try:
457 try:
458 repo.getcwd()
458 repo.getcwd()
459 except OSError:
459 except OSError:
460 return result
460 return result
461
461
462 try:
462 try:
463 try:
463 try:
464 # When we call orig below it creates the standins but we don't add
464 # When we call orig below it creates the standins but we don't add
465 # them to the dir state until later so lock during that time.
465 # them to the dir state until later so lock during that time.
466 wlock = repo.wlock()
466 wlock = repo.wlock()
467
467
468 manifest = repo[None].manifest()
468 manifest = repo[None].manifest()
469 oldmatch = None # for the closure
469 oldmatch = None # for the closure
470 def overridematch(ctx, pats=[], opts={}, globbed=False,
470 def overridematch(ctx, pats=[], opts={}, globbed=False,
471 default='relpath'):
471 default='relpath'):
472 newpats = []
472 newpats = []
473 # The patterns were previously mangled to add the standin
473 # The patterns were previously mangled to add the standin
474 # directory; we need to remove that now
474 # directory; we need to remove that now
475 for pat in pats:
475 for pat in pats:
476 if match_.patkind(pat) is None and lfutil.shortname in pat:
476 if match_.patkind(pat) is None and lfutil.shortname in pat:
477 newpats.append(pat.replace(lfutil.shortname, ''))
477 newpats.append(pat.replace(lfutil.shortname, ''))
478 else:
478 else:
479 newpats.append(pat)
479 newpats.append(pat)
480 match = oldmatch(ctx, newpats, opts, globbed, default)
480 match = oldmatch(ctx, newpats, opts, globbed, default)
481 m = copy.copy(match)
481 m = copy.copy(match)
482 lfile = lambda f: lfutil.standin(f) in manifest
482 lfile = lambda f: lfutil.standin(f) in manifest
483 m._files = [lfutil.standin(f) for f in m._files if lfile(f)]
483 m._files = [lfutil.standin(f) for f in m._files if lfile(f)]
484 m._fmap = set(m._files)
484 m._fmap = set(m._files)
485 origmatchfn = m.matchfn
485 origmatchfn = m.matchfn
486 m.matchfn = lambda f: (lfutil.isstandin(f) and
486 m.matchfn = lambda f: (lfutil.isstandin(f) and
487 (f in manifest) and
487 (f in manifest) and
488 origmatchfn(lfutil.splitstandin(f)) or
488 origmatchfn(lfutil.splitstandin(f)) or
489 None)
489 None)
490 return m
490 return m
491 oldmatch = installmatchfn(overridematch)
491 oldmatch = installmatchfn(overridematch)
492 listpats = []
492 listpats = []
493 for pat in pats:
493 for pat in pats:
494 if match_.patkind(pat) is not None:
494 if match_.patkind(pat) is not None:
495 listpats.append(pat)
495 listpats.append(pat)
496 else:
496 else:
497 listpats.append(makestandin(pat))
497 listpats.append(makestandin(pat))
498
498
499 try:
499 try:
500 origcopyfile = util.copyfile
500 origcopyfile = util.copyfile
501 copiedfiles = []
501 copiedfiles = []
502 def overridecopyfile(src, dest):
502 def overridecopyfile(src, dest):
503 if (lfutil.shortname in src and
503 if (lfutil.shortname in src and
504 dest.startswith(repo.wjoin(lfutil.shortname))):
504 dest.startswith(repo.wjoin(lfutil.shortname))):
505 destlfile = dest.replace(lfutil.shortname, '')
505 destlfile = dest.replace(lfutil.shortname, '')
506 if not opts['force'] and os.path.exists(destlfile):
506 if not opts['force'] and os.path.exists(destlfile):
507 raise IOError('',
507 raise IOError('',
508 _('destination largefile already exists'))
508 _('destination largefile already exists'))
509 copiedfiles.append((src, dest))
509 copiedfiles.append((src, dest))
510 origcopyfile(src, dest)
510 origcopyfile(src, dest)
511
511
512 util.copyfile = overridecopyfile
512 util.copyfile = overridecopyfile
513 result += orig(ui, repo, listpats, opts, rename)
513 result += orig(ui, repo, listpats, opts, rename)
514 finally:
514 finally:
515 util.copyfile = origcopyfile
515 util.copyfile = origcopyfile
516
516
517 lfdirstate = lfutil.openlfdirstate(ui, repo)
517 lfdirstate = lfutil.openlfdirstate(ui, repo)
518 for (src, dest) in copiedfiles:
518 for (src, dest) in copiedfiles:
519 if (lfutil.shortname in src and
519 if (lfutil.shortname in src and
520 dest.startswith(repo.wjoin(lfutil.shortname))):
520 dest.startswith(repo.wjoin(lfutil.shortname))):
521 srclfile = src.replace(repo.wjoin(lfutil.standin('')), '')
521 srclfile = src.replace(repo.wjoin(lfutil.standin('')), '')
522 destlfile = dest.replace(repo.wjoin(lfutil.standin('')), '')
522 destlfile = dest.replace(repo.wjoin(lfutil.standin('')), '')
523 destlfiledir = os.path.dirname(repo.wjoin(destlfile)) or '.'
523 destlfiledir = os.path.dirname(repo.wjoin(destlfile)) or '.'
524 if not os.path.isdir(destlfiledir):
524 if not os.path.isdir(destlfiledir):
525 os.makedirs(destlfiledir)
525 os.makedirs(destlfiledir)
526 if rename:
526 if rename:
527 os.rename(repo.wjoin(srclfile), repo.wjoin(destlfile))
527 os.rename(repo.wjoin(srclfile), repo.wjoin(destlfile))
528 lfdirstate.remove(srclfile)
528 lfdirstate.remove(srclfile)
529 else:
529 else:
530 util.copyfile(repo.wjoin(srclfile),
530 util.copyfile(repo.wjoin(srclfile),
531 repo.wjoin(destlfile))
531 repo.wjoin(destlfile))
532
532
533 lfdirstate.add(destlfile)
533 lfdirstate.add(destlfile)
534 lfdirstate.write()
534 lfdirstate.write()
535 except util.Abort, e:
535 except util.Abort, e:
536 if str(e) != _('no files to copy'):
536 if str(e) != _('no files to copy'):
537 raise e
537 raise e
538 else:
538 else:
539 nolfiles = True
539 nolfiles = True
540 finally:
540 finally:
541 restorematchfn()
541 restorematchfn()
542 wlock.release()
542 wlock.release()
543
543
544 if nolfiles and nonormalfiles:
544 if nolfiles and nonormalfiles:
545 raise util.Abort(_('no files to copy'))
545 raise util.Abort(_('no files to copy'))
546
546
547 return result
547 return result
548
548
549 # When the user calls revert, we have to be careful to not revert any
549 # When the user calls revert, we have to be careful to not revert any
550 # changes to other largefiles accidentally. This means we have to keep
550 # changes to other largefiles accidentally. This means we have to keep
551 # track of the largefiles that are being reverted so we only pull down
551 # track of the largefiles that are being reverted so we only pull down
552 # the necessary largefiles.
552 # the necessary largefiles.
553 #
553 #
554 # Standins are only updated (to match the hash of largefiles) before
554 # Standins are only updated (to match the hash of largefiles) before
555 # commits. Update the standins then run the original revert, changing
555 # commits. Update the standins then run the original revert, changing
556 # the matcher to hit standins instead of largefiles. Based on the
556 # the matcher to hit standins instead of largefiles. Based on the
557 # resulting standins update the largefiles. Then return the standins
557 # resulting standins update the largefiles. Then return the standins
558 # to their proper state
558 # to their proper state
559 def overriderevert(orig, ui, repo, *pats, **opts):
559 def overriderevert(orig, ui, repo, *pats, **opts):
560 # Because we put the standins in a bad state (by updating them)
560 # Because we put the standins in a bad state (by updating them)
561 # and then return them to a correct state we need to lock to
561 # and then return them to a correct state we need to lock to
562 # prevent others from changing them in their incorrect state.
562 # prevent others from changing them in their incorrect state.
563 wlock = repo.wlock()
563 wlock = repo.wlock()
564 try:
564 try:
565 lfdirstate = lfutil.openlfdirstate(ui, repo)
565 lfdirstate = lfutil.openlfdirstate(ui, repo)
566 (modified, added, removed, missing, unknown, ignored, clean) = \
566 (modified, added, removed, missing, unknown, ignored, clean) = \
567 lfutil.lfdirstatestatus(lfdirstate, repo, repo['.'].rev())
567 lfutil.lfdirstatestatus(lfdirstate, repo, repo['.'].rev())
568 for lfile in modified:
568 for lfile in modified:
569 lfutil.updatestandin(repo, lfutil.standin(lfile))
569 lfutil.updatestandin(repo, lfutil.standin(lfile))
570 for lfile in missing:
570 for lfile in missing:
571 if (os.path.exists(repo.wjoin(lfutil.standin(lfile)))):
571 if (os.path.exists(repo.wjoin(lfutil.standin(lfile)))):
572 os.unlink(repo.wjoin(lfutil.standin(lfile)))
572 os.unlink(repo.wjoin(lfutil.standin(lfile)))
573
573
574 try:
574 try:
575 ctx = scmutil.revsingle(repo, opts.get('rev'))
575 ctx = scmutil.revsingle(repo, opts.get('rev'))
576 oldmatch = None # for the closure
576 oldmatch = None # for the closure
577 def overridematch(ctx, pats=[], opts={}, globbed=False,
577 def overridematch(ctx, pats=[], opts={}, globbed=False,
578 default='relpath'):
578 default='relpath'):
579 match = oldmatch(ctx, pats, opts, globbed, default)
579 match = oldmatch(ctx, pats, opts, globbed, default)
580 m = copy.copy(match)
580 m = copy.copy(match)
581 def tostandin(f):
581 def tostandin(f):
582 if lfutil.standin(f) in ctx:
582 if lfutil.standin(f) in ctx:
583 return lfutil.standin(f)
583 return lfutil.standin(f)
584 elif lfutil.standin(f) in repo[None]:
584 elif lfutil.standin(f) in repo[None]:
585 return None
585 return None
586 return f
586 return f
587 m._files = [tostandin(f) for f in m._files]
587 m._files = [tostandin(f) for f in m._files]
588 m._files = [f for f in m._files if f is not None]
588 m._files = [f for f in m._files if f is not None]
589 m._fmap = set(m._files)
589 m._fmap = set(m._files)
590 origmatchfn = m.matchfn
590 origmatchfn = m.matchfn
591 def matchfn(f):
591 def matchfn(f):
592 if lfutil.isstandin(f):
592 if lfutil.isstandin(f):
593 # We need to keep track of what largefiles are being
593 # We need to keep track of what largefiles are being
594 # matched so we know which ones to update later --
594 # matched so we know which ones to update later --
595 # otherwise we accidentally revert changes to other
595 # otherwise we accidentally revert changes to other
596 # largefiles. This is repo-specific, so duckpunch the
596 # largefiles. This is repo-specific, so duckpunch the
597 # repo object to keep the list of largefiles for us
597 # repo object to keep the list of largefiles for us
598 # later.
598 # later.
599 if origmatchfn(lfutil.splitstandin(f)) and \
599 if origmatchfn(lfutil.splitstandin(f)) and \
600 (f in repo[None] or f in ctx):
600 (f in repo[None] or f in ctx):
601 lfileslist = getattr(repo, '_lfilestoupdate', [])
601 lfileslist = getattr(repo, '_lfilestoupdate', [])
602 lfileslist.append(lfutil.splitstandin(f))
602 lfileslist.append(lfutil.splitstandin(f))
603 repo._lfilestoupdate = lfileslist
603 repo._lfilestoupdate = lfileslist
604 return True
604 return True
605 else:
605 else:
606 return False
606 return False
607 return origmatchfn(f)
607 return origmatchfn(f)
608 m.matchfn = matchfn
608 m.matchfn = matchfn
609 return m
609 return m
610 oldmatch = installmatchfn(overridematch)
610 oldmatch = installmatchfn(overridematch)
611 scmutil.match
611 scmutil.match
612 matches = overridematch(repo[None], pats, opts)
612 matches = overridematch(repo[None], pats, opts)
613 orig(ui, repo, *pats, **opts)
613 orig(ui, repo, *pats, **opts)
614 finally:
614 finally:
615 restorematchfn()
615 restorematchfn()
616 lfileslist = getattr(repo, '_lfilestoupdate', [])
616 lfileslist = getattr(repo, '_lfilestoupdate', [])
617 lfcommands.updatelfiles(ui, repo, filelist=lfileslist,
617 lfcommands.updatelfiles(ui, repo, filelist=lfileslist,
618 printmessage=False)
618 printmessage=False)
619
619
620 # empty out the largefiles list so we start fresh next time
620 # empty out the largefiles list so we start fresh next time
621 repo._lfilestoupdate = []
621 repo._lfilestoupdate = []
622 for lfile in modified:
622 for lfile in modified:
623 if lfile in lfileslist:
623 if lfile in lfileslist:
624 if os.path.exists(repo.wjoin(lfutil.standin(lfile))) and lfile\
624 if os.path.exists(repo.wjoin(lfutil.standin(lfile))) and lfile\
625 in repo['.']:
625 in repo['.']:
626 lfutil.writestandin(repo, lfutil.standin(lfile),
626 lfutil.writestandin(repo, lfutil.standin(lfile),
627 repo['.'][lfile].data().strip(),
627 repo['.'][lfile].data().strip(),
628 'x' in repo['.'][lfile].flags())
628 'x' in repo['.'][lfile].flags())
629 lfdirstate = lfutil.openlfdirstate(ui, repo)
629 lfdirstate = lfutil.openlfdirstate(ui, repo)
630 for lfile in added:
630 for lfile in added:
631 standin = lfutil.standin(lfile)
631 standin = lfutil.standin(lfile)
632 if standin not in ctx and (standin in matches or opts.get('all')):
632 if standin not in ctx and (standin in matches or opts.get('all')):
633 if lfile in lfdirstate:
633 if lfile in lfdirstate:
634 lfdirstate.drop(lfile)
634 lfdirstate.drop(lfile)
635 util.unlinkpath(repo.wjoin(standin))
635 util.unlinkpath(repo.wjoin(standin))
636 lfdirstate.write()
636 lfdirstate.write()
637 finally:
637 finally:
638 wlock.release()
638 wlock.release()
639
639
640 def hgupdate(orig, repo, node):
640 def hgupdate(orig, repo, node):
641 # Only call updatelfiles the standins that have changed to save time
641 # Only call updatelfiles the standins that have changed to save time
642 oldstandins = lfutil.getstandinsstate(repo)
642 oldstandins = lfutil.getstandinsstate(repo)
643 result = orig(repo, node)
643 result = orig(repo, node)
644 newstandins = lfutil.getstandinsstate(repo)
644 newstandins = lfutil.getstandinsstate(repo)
645 filelist = lfutil.getlfilestoupdate(oldstandins, newstandins)
645 filelist = lfutil.getlfilestoupdate(oldstandins, newstandins)
646 lfcommands.updatelfiles(repo.ui, repo, filelist=filelist, printmessage=True)
646 lfcommands.updatelfiles(repo.ui, repo, filelist=filelist, printmessage=True)
647 return result
647 return result
648
648
649 def hgclean(orig, repo, node, show_stats=True):
649 def hgclean(orig, repo, node, show_stats=True):
650 result = orig(repo, node, show_stats)
650 result = orig(repo, node, show_stats)
651 lfcommands.updatelfiles(repo.ui, repo)
651 lfcommands.updatelfiles(repo.ui, repo)
652 return result
652 return result
653
653
654 def hgmerge(orig, repo, node, force=None, remind=True):
654 def hgmerge(orig, repo, node, force=None, remind=True):
655 # Mark the repo as being in the middle of a merge, so that
655 # Mark the repo as being in the middle of a merge, so that
656 # updatelfiles() will know that it needs to trust the standins in
656 # updatelfiles() will know that it needs to trust the standins in
657 # the working copy, not in the standins in the current node
657 # the working copy, not in the standins in the current node
658 repo._ismerging = True
658 repo._ismerging = True
659 try:
659 try:
660 result = orig(repo, node, force, remind)
660 result = orig(repo, node, force, remind)
661 lfcommands.updatelfiles(repo.ui, repo)
661 lfcommands.updatelfiles(repo.ui, repo)
662 finally:
662 finally:
663 repo._ismerging = False
663 repo._ismerging = False
664 return result
664 return result
665
665
666 # When we rebase a repository with remotely changed largefiles, we need to
666 # When we rebase a repository with remotely changed largefiles, we need to
667 # take some extra care so that the largefiles are correctly updated in the
667 # take some extra care so that the largefiles are correctly updated in the
668 # working copy
668 # working copy
669 def overridepull(orig, ui, repo, source=None, **opts):
669 def overridepull(orig, ui, repo, source=None, **opts):
670 revsprepull = len(repo)
670 revsprepull = len(repo)
671 if opts.get('rebase', False):
671 if opts.get('rebase', False):
672 repo._isrebasing = True
672 repo._isrebasing = True
673 try:
673 try:
674 if opts.get('update'):
674 if opts.get('update'):
675 del opts['update']
675 del opts['update']
676 ui.debug('--update and --rebase are not compatible, ignoring '
676 ui.debug('--update and --rebase are not compatible, ignoring '
677 'the update flag\n')
677 'the update flag\n')
678 del opts['rebase']
678 del opts['rebase']
679 cmdutil.bailifchanged(repo)
679 cmdutil.bailifchanged(repo)
680 origpostincoming = commands.postincoming
680 origpostincoming = commands.postincoming
681 def _dummy(*args, **kwargs):
681 def _dummy(*args, **kwargs):
682 pass
682 pass
683 commands.postincoming = _dummy
683 commands.postincoming = _dummy
684 repo.lfpullsource = source
684 repo.lfpullsource = source
685 if not source:
685 if not source:
686 source = 'default'
686 source = 'default'
687 try:
687 try:
688 result = commands.pull(ui, repo, source, **opts)
688 result = commands.pull(ui, repo, source, **opts)
689 finally:
689 finally:
690 commands.postincoming = origpostincoming
690 commands.postincoming = origpostincoming
691 revspostpull = len(repo)
691 revspostpull = len(repo)
692 if revspostpull > revsprepull:
692 if revspostpull > revsprepull:
693 result = result or rebase.rebase(ui, repo)
693 result = result or rebase.rebase(ui, repo)
694 finally:
694 finally:
695 repo._isrebasing = False
695 repo._isrebasing = False
696 else:
696 else:
697 repo.lfpullsource = source
697 repo.lfpullsource = source
698 if not source:
698 if not source:
699 source = 'default'
699 source = 'default'
700 oldheads = lfutil.getcurrentheads(repo)
700 oldheads = lfutil.getcurrentheads(repo)
701 result = orig(ui, repo, source, **opts)
701 result = orig(ui, repo, source, **opts)
702 # If we do not have the new largefiles for any new heads we pulled, we
702 # If we do not have the new largefiles for any new heads we pulled, we
703 # will run into a problem later if we try to merge or rebase with one of
703 # will run into a problem later if we try to merge or rebase with one of
704 # these heads, so cache the largefiles now direclty into the system
704 # these heads, so cache the largefiles now direclty into the system
705 # cache.
705 # cache.
706 ui.status(_("caching new largefiles\n"))
706 ui.status(_("caching new largefiles\n"))
707 numcached = 0
707 numcached = 0
708 heads = lfutil.getcurrentheads(repo)
708 heads = lfutil.getcurrentheads(repo)
709 newheads = set(heads).difference(set(oldheads))
709 newheads = set(heads).difference(set(oldheads))
710 for head in newheads:
710 for head in newheads:
711 (cached, missing) = lfcommands.cachelfiles(ui, repo, head)
711 (cached, missing) = lfcommands.cachelfiles(ui, repo, head)
712 numcached += len(cached)
712 numcached += len(cached)
713 ui.status(_("%d largefiles cached\n") % numcached)
713 ui.status(_("%d largefiles cached\n") % numcached)
714 if opts.get('all_largefiles'):
714 if opts.get('all_largefiles'):
715 revspostpull = len(repo)
715 revspostpull = len(repo)
716 revs = []
716 revs = []
717 for rev in xrange(revsprepull + 1, revspostpull):
717 for rev in xrange(revsprepull + 1, revspostpull):
718 revs.append(repo[rev].rev())
718 revs.append(repo[rev].rev())
719 lfcommands.downloadlfiles(ui, repo, revs)
719 lfcommands.downloadlfiles(ui, repo, revs)
720 return result
720 return result
721
721
722 def overrideclone(orig, ui, source, dest=None, **opts):
722 def overrideclone(orig, ui, source, dest=None, **opts):
723 if dest is None:
723 if dest is None:
724 dest = hg.defaultdest(source)
724 dest = hg.defaultdest(source)
725 if opts.get('all_largefiles') and not hg.islocal(dest):
725 if opts.get('all_largefiles') and not hg.islocal(dest):
726 raise util.Abort(_(
726 raise util.Abort(_(
727 '--all-largefiles is incompatible with non-local destination %s' %
727 '--all-largefiles is incompatible with non-local destination %s' %
728 dest))
728 dest))
729 result = hg.clone(ui, opts, source, dest,
729 result = hg.clone(ui, opts, source, dest,
730 pull=opts.get('pull'),
730 pull=opts.get('pull'),
731 stream=opts.get('uncompressed'),
731 stream=opts.get('uncompressed'),
732 rev=opts.get('rev'),
732 rev=opts.get('rev'),
733 update=True, # required for successful walkchangerevs
733 update=True, # required for successful walkchangerevs
734 branch=opts.get('branch'))
734 branch=opts.get('branch'))
735 if result is None:
735 if result is None:
736 return True
736 return True
737 if opts.get('all_largefiles'):
737 if opts.get('all_largefiles'):
738 sourcerepo, destrepo = result
738 sourcerepo, destrepo = result
739 success, missing = lfcommands.downloadlfiles(ui, destrepo.local(), None)
739 success, missing = lfcommands.downloadlfiles(ui, destrepo.local(), None)
740 return missing != 0
740 return missing != 0
741 return result is None
741 return result is None
742
742
743 def overriderebase(orig, ui, repo, **opts):
743 def overriderebase(orig, ui, repo, **opts):
744 repo._isrebasing = True
744 repo._isrebasing = True
745 try:
745 try:
746 orig(ui, repo, **opts)
746 orig(ui, repo, **opts)
747 finally:
747 finally:
748 repo._isrebasing = False
748 repo._isrebasing = False
749
749
750 def overridearchive(orig, repo, dest, node, kind, decode=True, matchfn=None,
750 def overridearchive(orig, repo, dest, node, kind, decode=True, matchfn=None,
751 prefix=None, mtime=None, subrepos=None):
751 prefix=None, mtime=None, subrepos=None):
752 # No need to lock because we are only reading history and
752 # No need to lock because we are only reading history and
753 # largefile caches, neither of which are modified.
753 # largefile caches, neither of which are modified.
754 lfcommands.cachelfiles(repo.ui, repo, node)
754 lfcommands.cachelfiles(repo.ui, repo, node)
755
755
756 if kind not in archival.archivers:
756 if kind not in archival.archivers:
757 raise util.Abort(_("unknown archive type '%s'") % kind)
757 raise util.Abort(_("unknown archive type '%s'") % kind)
758
758
759 ctx = repo[node]
759 ctx = repo[node]
760
760
761 if kind == 'files':
761 if kind == 'files':
762 if prefix:
762 if prefix:
763 raise util.Abort(
763 raise util.Abort(
764 _('cannot give prefix when archiving to files'))
764 _('cannot give prefix when archiving to files'))
765 else:
765 else:
766 prefix = archival.tidyprefix(dest, kind, prefix)
766 prefix = archival.tidyprefix(dest, kind, prefix)
767
767
768 def write(name, mode, islink, getdata):
768 def write(name, mode, islink, getdata):
769 if matchfn and not matchfn(name):
769 if matchfn and not matchfn(name):
770 return
770 return
771 data = getdata()
771 data = getdata()
772 if decode:
772 if decode:
773 data = repo.wwritedata(name, data)
773 data = repo.wwritedata(name, data)
774 archiver.addfile(prefix + name, mode, islink, data)
774 archiver.addfile(prefix + name, mode, islink, data)
775
775
776 archiver = archival.archivers[kind](dest, mtime or ctx.date()[0])
776 archiver = archival.archivers[kind](dest, mtime or ctx.date()[0])
777
777
778 if repo.ui.configbool("ui", "archivemeta", True):
778 if repo.ui.configbool("ui", "archivemeta", True):
779 def metadata():
779 def metadata():
780 base = 'repo: %s\nnode: %s\nbranch: %s\n' % (
780 base = 'repo: %s\nnode: %s\nbranch: %s\n' % (
781 hex(repo.changelog.node(0)), hex(node), ctx.branch())
781 hex(repo.changelog.node(0)), hex(node), ctx.branch())
782
782
783 tags = ''.join('tag: %s\n' % t for t in ctx.tags()
783 tags = ''.join('tag: %s\n' % t for t in ctx.tags()
784 if repo.tagtype(t) == 'global')
784 if repo.tagtype(t) == 'global')
785 if not tags:
785 if not tags:
786 repo.ui.pushbuffer()
786 repo.ui.pushbuffer()
787 opts = {'template': '{latesttag}\n{latesttagdistance}',
787 opts = {'template': '{latesttag}\n{latesttagdistance}',
788 'style': '', 'patch': None, 'git': None}
788 'style': '', 'patch': None, 'git': None}
789 cmdutil.show_changeset(repo.ui, repo, opts).show(ctx)
789 cmdutil.show_changeset(repo.ui, repo, opts).show(ctx)
790 ltags, dist = repo.ui.popbuffer().split('\n')
790 ltags, dist = repo.ui.popbuffer().split('\n')
791 tags = ''.join('latesttag: %s\n' % t for t in ltags.split(':'))
791 tags = ''.join('latesttag: %s\n' % t for t in ltags.split(':'))
792 tags += 'latesttagdistance: %s\n' % dist
792 tags += 'latesttagdistance: %s\n' % dist
793
793
794 return base + tags
794 return base + tags
795
795
796 write('.hg_archival.txt', 0644, False, metadata)
796 write('.hg_archival.txt', 0644, False, metadata)
797
797
798 for f in ctx:
798 for f in ctx:
799 ff = ctx.flags(f)
799 ff = ctx.flags(f)
800 getdata = ctx[f].data
800 getdata = ctx[f].data
801 if lfutil.isstandin(f):
801 if lfutil.isstandin(f):
802 path = lfutil.findfile(repo, getdata().strip())
802 path = lfutil.findfile(repo, getdata().strip())
803 if path is None:
803 if path is None:
804 raise util.Abort(
804 raise util.Abort(
805 _('largefile %s not found in repo store or system cache')
805 _('largefile %s not found in repo store or system cache')
806 % lfutil.splitstandin(f))
806 % lfutil.splitstandin(f))
807 f = lfutil.splitstandin(f)
807 f = lfutil.splitstandin(f)
808
808
809 def getdatafn():
809 def getdatafn():
810 fd = None
810 fd = None
811 try:
811 try:
812 fd = open(path, 'rb')
812 fd = open(path, 'rb')
813 return fd.read()
813 return fd.read()
814 finally:
814 finally:
815 if fd:
815 if fd:
816 fd.close()
816 fd.close()
817
817
818 getdata = getdatafn
818 getdata = getdatafn
819 write(f, 'x' in ff and 0755 or 0644, 'l' in ff, getdata)
819 write(f, 'x' in ff and 0755 or 0644, 'l' in ff, getdata)
820
820
821 if subrepos:
821 if subrepos:
822 for subpath in ctx.substate:
822 for subpath in ctx.substate:
823 sub = ctx.sub(subpath)
823 sub = ctx.sub(subpath)
824 submatch = match_.narrowmatcher(subpath, matchfn)
824 submatch = match_.narrowmatcher(subpath, matchfn)
825 sub.archive(repo.ui, archiver, prefix, submatch)
825 sub.archive(repo.ui, archiver, prefix, submatch)
826
826
827 archiver.done()
827 archiver.done()
828
828
829 def hgsubrepoarchive(orig, repo, ui, archiver, prefix, match=None):
829 def hgsubrepoarchive(orig, repo, ui, archiver, prefix, match=None):
830 rev = repo._state[1]
830 rev = repo._state[1]
831 ctx = repo._repo[rev]
831 ctx = repo._repo[rev]
832
832
833 lfcommands.cachelfiles(ui, repo._repo, ctx.node())
833 lfcommands.cachelfiles(ui, repo._repo, ctx.node())
834
834
835 def write(name, mode, islink, getdata):
835 def write(name, mode, islink, getdata):
836 # At this point, the standin has been replaced with the largefile name,
836 # At this point, the standin has been replaced with the largefile name,
837 # so the normal matcher works here without the lfutil variants.
837 # so the normal matcher works here without the lfutil variants.
838 if match and not match(f):
838 if match and not match(f):
839 return
839 return
840 data = getdata()
840 data = getdata()
841
841
842 archiver.addfile(prefix + repo._path + '/' + name, mode, islink, data)
842 archiver.addfile(prefix + repo._path + '/' + name, mode, islink, data)
843
843
844 for f in ctx:
844 for f in ctx:
845 ff = ctx.flags(f)
845 ff = ctx.flags(f)
846 getdata = ctx[f].data
846 getdata = ctx[f].data
847 if lfutil.isstandin(f):
847 if lfutil.isstandin(f):
848 path = lfutil.findfile(repo._repo, getdata().strip())
848 path = lfutil.findfile(repo._repo, getdata().strip())
849 if path is None:
849 if path is None:
850 raise util.Abort(
850 raise util.Abort(
851 _('largefile %s not found in repo store or system cache')
851 _('largefile %s not found in repo store or system cache')
852 % lfutil.splitstandin(f))
852 % lfutil.splitstandin(f))
853 f = lfutil.splitstandin(f)
853 f = lfutil.splitstandin(f)
854
854
855 def getdatafn():
855 def getdatafn():
856 fd = None
856 fd = None
857 try:
857 try:
858 fd = open(os.path.join(prefix, path), 'rb')
858 fd = open(os.path.join(prefix, path), 'rb')
859 return fd.read()
859 return fd.read()
860 finally:
860 finally:
861 if fd:
861 if fd:
862 fd.close()
862 fd.close()
863
863
864 getdata = getdatafn
864 getdata = getdatafn
865
865
866 write(f, 'x' in ff and 0755 or 0644, 'l' in ff, getdata)
866 write(f, 'x' in ff and 0755 or 0644, 'l' in ff, getdata)
867
867
868 for subpath in ctx.substate:
868 for subpath in ctx.substate:
869 sub = ctx.sub(subpath)
869 sub = ctx.sub(subpath)
870 submatch = match_.narrowmatcher(subpath, match)
870 submatch = match_.narrowmatcher(subpath, match)
871 sub.archive(ui, archiver, os.path.join(prefix, repo._path) + '/',
871 sub.archive(ui, archiver, os.path.join(prefix, repo._path) + '/',
872 submatch)
872 submatch)
873
873
874 # If a largefile is modified, the change is not reflected in its
874 # If a largefile is modified, the change is not reflected in its
875 # standin until a commit. cmdutil.bailifchanged() raises an exception
875 # standin until a commit. cmdutil.bailifchanged() raises an exception
876 # if the repo has uncommitted changes. Wrap it to also check if
876 # if the repo has uncommitted changes. Wrap it to also check if
877 # largefiles were changed. This is used by bisect and backout.
877 # largefiles were changed. This is used by bisect and backout.
878 def overridebailifchanged(orig, repo):
878 def overridebailifchanged(orig, repo):
879 orig(repo)
879 orig(repo)
880 repo.lfstatus = True
880 repo.lfstatus = True
881 modified, added, removed, deleted = repo.status()[:4]
881 modified, added, removed, deleted = repo.status()[:4]
882 repo.lfstatus = False
882 repo.lfstatus = False
883 if modified or added or removed or deleted:
883 if modified or added or removed or deleted:
884 raise util.Abort(_('outstanding uncommitted changes'))
884 raise util.Abort(_('outstanding uncommitted changes'))
885
885
886 # Fetch doesn't use cmdutil.bailifchanged so override it to add the check
886 # Fetch doesn't use cmdutil.bailifchanged so override it to add the check
887 def overridefetch(orig, ui, repo, *pats, **opts):
887 def overridefetch(orig, ui, repo, *pats, **opts):
888 repo.lfstatus = True
888 repo.lfstatus = True
889 modified, added, removed, deleted = repo.status()[:4]
889 modified, added, removed, deleted = repo.status()[:4]
890 repo.lfstatus = False
890 repo.lfstatus = False
891 if modified or added or removed or deleted:
891 if modified or added or removed or deleted:
892 raise util.Abort(_('outstanding uncommitted changes'))
892 raise util.Abort(_('outstanding uncommitted changes'))
893 return orig(ui, repo, *pats, **opts)
893 return orig(ui, repo, *pats, **opts)
894
894
895 def overrideforget(orig, ui, repo, *pats, **opts):
895 def overrideforget(orig, ui, repo, *pats, **opts):
896 installnormalfilesmatchfn(repo[None].manifest())
896 installnormalfilesmatchfn(repo[None].manifest())
897 orig(ui, repo, *pats, **opts)
897 orig(ui, repo, *pats, **opts)
898 restorematchfn()
898 restorematchfn()
899 m = scmutil.match(repo[None], pats, opts)
899 m = scmutil.match(repo[None], pats, opts)
900
900
901 try:
901 try:
902 repo.lfstatus = True
902 repo.lfstatus = True
903 s = repo.status(match=m, clean=True)
903 s = repo.status(match=m, clean=True)
904 finally:
904 finally:
905 repo.lfstatus = False
905 repo.lfstatus = False
906 forget = sorted(s[0] + s[1] + s[3] + s[6])
906 forget = sorted(s[0] + s[1] + s[3] + s[6])
907 forget = [f for f in forget if lfutil.standin(f) in repo[None].manifest()]
907 forget = [f for f in forget if lfutil.standin(f) in repo[None].manifest()]
908
908
909 for f in forget:
909 for f in forget:
910 if lfutil.standin(f) not in repo.dirstate and not \
910 if lfutil.standin(f) not in repo.dirstate and not \
911 os.path.isdir(m.rel(lfutil.standin(f))):
911 os.path.isdir(m.rel(lfutil.standin(f))):
912 ui.warn(_('not removing %s: file is already untracked\n')
912 ui.warn(_('not removing %s: file is already untracked\n')
913 % m.rel(f))
913 % m.rel(f))
914
914
915 for f in forget:
915 for f in forget:
916 if ui.verbose or not m.exact(f):
916 if ui.verbose or not m.exact(f):
917 ui.status(_('removing %s\n') % m.rel(f))
917 ui.status(_('removing %s\n') % m.rel(f))
918
918
919 # Need to lock because standin files are deleted then removed from the
919 # Need to lock because standin files are deleted then removed from the
920 # repository and we could race inbetween.
920 # repository and we could race inbetween.
921 wlock = repo.wlock()
921 wlock = repo.wlock()
922 try:
922 try:
923 lfdirstate = lfutil.openlfdirstate(ui, repo)
923 lfdirstate = lfutil.openlfdirstate(ui, repo)
924 for f in forget:
924 for f in forget:
925 if lfdirstate[f] == 'a':
925 if lfdirstate[f] == 'a':
926 lfdirstate.drop(f)
926 lfdirstate.drop(f)
927 else:
927 else:
928 lfdirstate.remove(f)
928 lfdirstate.remove(f)
929 lfdirstate.write()
929 lfdirstate.write()
930 lfutil.reporemove(repo, [lfutil.standin(f) for f in forget],
930 lfutil.reporemove(repo, [lfutil.standin(f) for f in forget],
931 unlink=True)
931 unlink=True)
932 finally:
932 finally:
933 wlock.release()
933 wlock.release()
934
934
935 def getoutgoinglfiles(ui, repo, dest=None, **opts):
935 def getoutgoinglfiles(ui, repo, dest=None, **opts):
936 dest = ui.expandpath(dest or 'default-push', dest or 'default')
936 dest = ui.expandpath(dest or 'default-push', dest or 'default')
937 dest, branches = hg.parseurl(dest, opts.get('branch'))
937 dest, branches = hg.parseurl(dest, opts.get('branch'))
938 revs, checkout = hg.addbranchrevs(repo, repo, branches, opts.get('rev'))
938 revs, checkout = hg.addbranchrevs(repo, repo, branches, opts.get('rev'))
939 if revs:
939 if revs:
940 revs = [repo.lookup(rev) for rev in scmutil.revrange(repo, revs)]
940 revs = [repo.lookup(rev) for rev in scmutil.revrange(repo, revs)]
941
941
942 try:
942 try:
943 remote = hg.peer(repo, opts, dest)
943 remote = hg.peer(repo, opts, dest)
944 except error.RepoError:
944 except error.RepoError:
945 return None
945 return None
946 o = lfutil.findoutgoing(repo, remote, False)
946 o = lfutil.findoutgoing(repo, remote, False)
947 if not o:
947 if not o:
948 return None
948 return None
949 o = repo.changelog.nodesbetween(o, revs)[0]
949 o = repo.changelog.nodesbetween(o, revs)[0]
950 if opts.get('newest_first'):
950 if opts.get('newest_first'):
951 o.reverse()
951 o.reverse()
952
952
953 toupload = set()
953 toupload = set()
954 for n in o:
954 for n in o:
955 parents = [p for p in repo.changelog.parents(n) if p != node.nullid]
955 parents = [p for p in repo.changelog.parents(n) if p != node.nullid]
956 ctx = repo[n]
956 ctx = repo[n]
957 files = set(ctx.files())
957 files = set(ctx.files())
958 if len(parents) == 2:
958 if len(parents) == 2:
959 mc = ctx.manifest()
959 mc = ctx.manifest()
960 mp1 = ctx.parents()[0].manifest()
960 mp1 = ctx.parents()[0].manifest()
961 mp2 = ctx.parents()[1].manifest()
961 mp2 = ctx.parents()[1].manifest()
962 for f in mp1:
962 for f in mp1:
963 if f not in mc:
963 if f not in mc:
964 files.add(f)
964 files.add(f)
965 for f in mp2:
965 for f in mp2:
966 if f not in mc:
966 if f not in mc:
967 files.add(f)
967 files.add(f)
968 for f in mc:
968 for f in mc:
969 if mc[f] != mp1.get(f, None) or mc[f] != mp2.get(f, None):
969 if mc[f] != mp1.get(f, None) or mc[f] != mp2.get(f, None):
970 files.add(f)
970 files.add(f)
971 toupload = toupload.union(
971 toupload = toupload.union(
972 set([f for f in files if lfutil.isstandin(f) and f in ctx]))
972 set([f for f in files if lfutil.isstandin(f) and f in ctx]))
973 return toupload
973 return toupload
974
974
975 def overrideoutgoing(orig, ui, repo, dest=None, **opts):
975 def overrideoutgoing(orig, ui, repo, dest=None, **opts):
976 result = orig(ui, repo, dest, **opts)
976 result = orig(ui, repo, dest, **opts)
977
977
978 if opts.pop('large', None):
978 if opts.pop('large', None):
979 toupload = getoutgoinglfiles(ui, repo, dest, **opts)
979 toupload = getoutgoinglfiles(ui, repo, dest, **opts)
980 if toupload is None:
980 if toupload is None:
981 ui.status(_('largefiles: No remote repo\n'))
981 ui.status(_('largefiles: No remote repo\n'))
982 else:
982 else:
983 ui.status(_('largefiles to upload:\n'))
983 ui.status(_('largefiles to upload:\n'))
984 for file in toupload:
984 for file in toupload:
985 ui.status(lfutil.splitstandin(file) + '\n')
985 ui.status(lfutil.splitstandin(file) + '\n')
986 ui.status('\n')
986 ui.status('\n')
987
987
988 return result
988 return result
989
989
990 def overridesummary(orig, ui, repo, *pats, **opts):
990 def overridesummary(orig, ui, repo, *pats, **opts):
991 try:
991 try:
992 repo.lfstatus = True
992 repo.lfstatus = True
993 orig(ui, repo, *pats, **opts)
993 orig(ui, repo, *pats, **opts)
994 finally:
994 finally:
995 repo.lfstatus = False
995 repo.lfstatus = False
996
996
997 if opts.pop('large', None):
997 if opts.pop('large', None):
998 toupload = getoutgoinglfiles(ui, repo, None, **opts)
998 toupload = getoutgoinglfiles(ui, repo, None, **opts)
999 if toupload is None:
999 if toupload is None:
1000 ui.status(_('largefiles: No remote repo\n'))
1000 ui.status(_('largefiles: No remote repo\n'))
1001 else:
1001 else:
1002 ui.status(_('largefiles: %d to upload\n') % len(toupload))
1002 ui.status(_('largefiles: %d to upload\n') % len(toupload))
1003
1003
1004 def overrideaddremove(orig, ui, repo, *pats, **opts):
1004 def overrideaddremove(orig, ui, repo, *pats, **opts):
1005 if not lfutil.islfilesrepo(repo):
1005 if not lfutil.islfilesrepo(repo):
1006 return orig(ui, repo, *pats, **opts)
1006 return orig(ui, repo, *pats, **opts)
1007 # Get the list of missing largefiles so we can remove them
1007 # Get the list of missing largefiles so we can remove them
1008 lfdirstate = lfutil.openlfdirstate(ui, repo)
1008 lfdirstate = lfutil.openlfdirstate(ui, repo)
1009 s = lfdirstate.status(match_.always(repo.root, repo.getcwd()), [], False,
1009 s = lfdirstate.status(match_.always(repo.root, repo.getcwd()), [], False,
1010 False, False)
1010 False, False)
1011 (unsure, modified, added, removed, missing, unknown, ignored, clean) = s
1011 (unsure, modified, added, removed, missing, unknown, ignored, clean) = s
1012
1012
1013 # Call into the normal remove code, but the removing of the standin, we want
1013 # Call into the normal remove code, but the removing of the standin, we want
1014 # to have handled by original addremove. Monkey patching here makes sure
1014 # to have handled by original addremove. Monkey patching here makes sure
1015 # we don't remove the standin in the largefiles code, preventing a very
1015 # we don't remove the standin in the largefiles code, preventing a very
1016 # confused state later.
1016 # confused state later.
1017 if missing:
1017 if missing:
1018 m = [repo.wjoin(f) for f in missing]
1018 m = [repo.wjoin(f) for f in missing]
1019 repo._isaddremove = True
1019 repo._isaddremove = True
1020 removelargefiles(ui, repo, *m, **opts)
1020 removelargefiles(ui, repo, *m, **opts)
1021 repo._isaddremove = False
1021 repo._isaddremove = False
1022 # Call into the normal add code, and any files that *should* be added as
1022 # Call into the normal add code, and any files that *should* be added as
1023 # largefiles will be
1023 # largefiles will be
1024 addlargefiles(ui, repo, *pats, **opts)
1024 addlargefiles(ui, repo, *pats, **opts)
1025 # Now that we've handled largefiles, hand off to the original addremove
1025 # Now that we've handled largefiles, hand off to the original addremove
1026 # function to take care of the rest. Make sure it doesn't do anything with
1026 # function to take care of the rest. Make sure it doesn't do anything with
1027 # largefiles by installing a matcher that will ignore them.
1027 # largefiles by installing a matcher that will ignore them.
1028 installnormalfilesmatchfn(repo[None].manifest())
1028 installnormalfilesmatchfn(repo[None].manifest())
1029 result = orig(ui, repo, *pats, **opts)
1029 result = orig(ui, repo, *pats, **opts)
1030 restorematchfn()
1030 restorematchfn()
1031 return result
1031 return result
1032
1032
1033 # Calling purge with --all will cause the largefiles to be deleted.
1033 # Calling purge with --all will cause the largefiles to be deleted.
1034 # Override repo.status to prevent this from happening.
1034 # Override repo.status to prevent this from happening.
1035 def overridepurge(orig, ui, repo, *dirs, **opts):
1035 def overridepurge(orig, ui, repo, *dirs, **opts):
1036 oldstatus = repo.status
1036 oldstatus = repo.status
1037 def overridestatus(node1='.', node2=None, match=None, ignored=False,
1037 def overridestatus(node1='.', node2=None, match=None, ignored=False,
1038 clean=False, unknown=False, listsubrepos=False):
1038 clean=False, unknown=False, listsubrepos=False):
1039 r = oldstatus(node1, node2, match, ignored, clean, unknown,
1039 r = oldstatus(node1, node2, match, ignored, clean, unknown,
1040 listsubrepos)
1040 listsubrepos)
1041 lfdirstate = lfutil.openlfdirstate(ui, repo)
1041 lfdirstate = lfutil.openlfdirstate(ui, repo)
1042 modified, added, removed, deleted, unknown, ignored, clean = r
1042 modified, added, removed, deleted, unknown, ignored, clean = r
1043 unknown = [f for f in unknown if lfdirstate[f] == '?']
1043 unknown = [f for f in unknown if lfdirstate[f] == '?']
1044 ignored = [f for f in ignored if lfdirstate[f] == '?']
1044 ignored = [f for f in ignored if lfdirstate[f] == '?']
1045 return modified, added, removed, deleted, unknown, ignored, clean
1045 return modified, added, removed, deleted, unknown, ignored, clean
1046 repo.status = overridestatus
1046 repo.status = overridestatus
1047 orig(ui, repo, *dirs, **opts)
1047 orig(ui, repo, *dirs, **opts)
1048 repo.status = oldstatus
1048 repo.status = oldstatus
1049
1049
1050 def overriderollback(orig, ui, repo, **opts):
1050 def overriderollback(orig, ui, repo, **opts):
1051 result = orig(ui, repo, **opts)
1051 result = orig(ui, repo, **opts)
1052 merge.update(repo, node=None, branchmerge=False, force=True,
1052 merge.update(repo, node=None, branchmerge=False, force=True,
1053 partial=lfutil.isstandin)
1053 partial=lfutil.isstandin)
1054 wlock = repo.wlock()
1054 wlock = repo.wlock()
1055 try:
1055 try:
1056 lfdirstate = lfutil.openlfdirstate(ui, repo)
1056 lfdirstate = lfutil.openlfdirstate(ui, repo)
1057 lfiles = lfutil.listlfiles(repo)
1057 lfiles = lfutil.listlfiles(repo)
1058 oldlfiles = lfutil.listlfiles(repo, repo[None].parents()[0].rev())
1058 oldlfiles = lfutil.listlfiles(repo, repo[None].parents()[0].rev())
1059 for file in lfiles:
1059 for file in lfiles:
1060 if file in oldlfiles:
1060 if file in oldlfiles:
1061 lfdirstate.normallookup(file)
1061 lfdirstate.normallookup(file)
1062 else:
1062 else:
1063 lfdirstate.add(file)
1063 lfdirstate.add(file)
1064 lfdirstate.write()
1064 lfdirstate.write()
1065 finally:
1065 finally:
1066 wlock.release()
1066 wlock.release()
1067 return result
1067 return result
1068
1068
1069 def overridetransplant(orig, ui, repo, *revs, **opts):
1069 def overridetransplant(orig, ui, repo, *revs, **opts):
1070 try:
1070 try:
1071 oldstandins = lfutil.getstandinsstate(repo)
1071 oldstandins = lfutil.getstandinsstate(repo)
1072 repo._istransplanting = True
1072 repo._istransplanting = True
1073 result = orig(ui, repo, *revs, **opts)
1073 result = orig(ui, repo, *revs, **opts)
1074 newstandins = lfutil.getstandinsstate(repo)
1074 newstandins = lfutil.getstandinsstate(repo)
1075 filelist = lfutil.getlfilestoupdate(oldstandins, newstandins)
1075 filelist = lfutil.getlfilestoupdate(oldstandins, newstandins)
1076 lfcommands.updatelfiles(repo.ui, repo, filelist=filelist,
1076 lfcommands.updatelfiles(repo.ui, repo, filelist=filelist,
1077 printmessage=True)
1077 printmessage=True)
1078 finally:
1078 finally:
1079 repo._istransplanting = False
1079 repo._istransplanting = False
1080 return result
1080 return result
1081
1081
1082 def overridecat(orig, ui, repo, file1, *pats, **opts):
1082 def overridecat(orig, ui, repo, file1, *pats, **opts):
1083 ctx = scmutil.revsingle(repo, opts.get('rev'))
1083 ctx = scmutil.revsingle(repo, opts.get('rev'))
1084 if not lfutil.standin(file1) in ctx:
1084 if not lfutil.standin(file1) in ctx:
1085 result = orig(ui, repo, file1, *pats, **opts)
1085 result = orig(ui, repo, file1, *pats, **opts)
1086 return result
1086 return result
1087 return lfcommands.catlfile(repo, file1, ctx.rev(), opts.get('output'))
1087 return lfcommands.catlfile(repo, file1, ctx.rev(), opts.get('output'))
General Comments 0
You need to be logged in to leave comments. Login now