Mercurial > hg
annotate hgext/largefiles/lfutil.py @ 15316:c65f5b6e26d4 stable
largefiles: rename functions and methods to match desired behavior
The original intent was that the largefiles would primarily be in the
repository, with the global cache being only that--a cache. The naming
conventions and actual intent have both strayed. In this first patch, the
naming conventions are switched to match the actual intent, as are the
configuration options.
author | Benjamin Pollack <benjamin@bitquabit.com> |
---|---|
date | Thu, 20 Oct 2011 13:24:09 -0400 |
parents | 9aa9d4bb3d88 |
children | 41f371150ccb |
rev | line source |
---|---|
15168 | 1 # Copyright 2009-2010 Gregory P. Ward |
2 # Copyright 2009-2010 Intelerad Medical Systems Incorporated | |
3 # Copyright 2010-2011 Fog Creek Software | |
4 # Copyright 2010-2011 Unity Technologies | |
5 # | |
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. | |
8 | |
9 '''largefiles utility code: must not import other modules in this package.''' | |
10 | |
11 import os | |
12 import errno | |
13 import shutil | |
14 import stat | |
15 import hashlib | |
16 | |
15226
2223ea21c98f
largefiles: cleanup import, now that we can assume > 1.9 for bundled extension
Na'Tosha Bard <natosha@unity3d.com>
parents:
15224
diff
changeset
|
17 from mercurial import dirstate, httpconnection, match as match_, util, scmutil |
15168 | 18 from mercurial.i18n import _ |
19 | |
20 shortname = '.hglf' | |
21 longname = 'largefiles' | |
22 | |
23 | |
24 # -- Portability wrappers ---------------------------------------------- | |
25 | |
15224
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
26 def dirstate_walk(dirstate, matcher, unknown=False, ignored=False): |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
27 return dirstate.walk(matcher, [], unknown, ignored) |
15168 | 28 |
29 def repo_add(repo, list): | |
15224
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
30 add = repo[None].add |
15168 | 31 return add(list) |
32 | |
33 def repo_remove(repo, list, unlink=False): | |
15224
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
34 def remove(list, unlink): |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
35 wlock = repo.wlock() |
15168 | 36 try: |
15224
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
37 if unlink: |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
38 for f in list: |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
39 try: |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
40 util.unlinkpath(repo.wjoin(f)) |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
41 except OSError, inst: |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
42 if inst.errno != errno.ENOENT: |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
43 raise |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
44 repo[None].forget(list) |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
45 finally: |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
46 wlock.release() |
15168 | 47 return remove(list, unlink=unlink) |
48 | |
49 def repo_forget(repo, list): | |
15224
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
50 forget = repo[None].forget |
15168 | 51 return forget(list) |
52 | |
53 def findoutgoing(repo, remote, force): | |
15224
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
54 from mercurial import discovery |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
55 common, _anyinc, _heads = discovery.findcommonincoming(repo, |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
56 remote, force=force) |
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
57 return repo.changelog.findmissing(common) |
15168 | 58 |
59 # -- Private worker functions ------------------------------------------ | |
60 | |
15227
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
61 def getminsize(ui, assumelfiles, opt, default=10): |
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
62 lfsize = opt |
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
63 if not lfsize and assumelfiles: |
15304
9aa9d4bb3d88
largefiles: rename config setting 'size' to 'minsize'
Greg Ward <greg@gerg.ca>
parents:
15255
diff
changeset
|
64 lfsize = ui.config(longname, 'minsize', default=default) |
15227
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
65 if lfsize: |
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
66 try: |
15228
ee625de3541e
largefiles: allow minimum size to be a float
Greg Ward <greg@gerg.ca>
parents:
15227
diff
changeset
|
67 lfsize = float(lfsize) |
15227
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
68 except ValueError: |
15228
ee625de3541e
largefiles: allow minimum size to be a float
Greg Ward <greg@gerg.ca>
parents:
15227
diff
changeset
|
69 raise util.Abort(_('largefiles: size must be number (not %s)\n') |
15227
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
70 % lfsize) |
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
71 if lfsize is None: |
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
72 raise util.Abort(_('minimum size for largefiles must be specified')) |
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
73 return lfsize |
a7686abf73a6
largefiles: factor out lfutil.getminsize()
Greg Ward <greg@gerg.ca>
parents:
15226
diff
changeset
|
74 |
15168 | 75 def link(src, dest): |
76 try: | |
15206
f85c76b16f27
largefiles: fix commit of specified file on non-windows
Na'Tosha Bard <natosha@unity3d.com>
parents:
15188
diff
changeset
|
77 util.oslink(src, dest) |
15168 | 78 except OSError: |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
79 # if hardlinks fail, fallback on copy |
15168 | 80 shutil.copyfile(src, dest) |
81 os.chmod(dest, os.stat(src).st_mode) | |
82 | |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
83 def usercachepath(ui, hash): |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
84 path = ui.config(longname, 'usercache', None) |
15168 | 85 if path: |
86 path = os.path.join(path, hash) | |
87 else: | |
88 if os.name == 'nt': | |
15255
7ab05d752405
largefiles: cosmetics, whitespace, code style
Greg Ward <greg@gerg.ca>
parents:
15253
diff
changeset
|
89 appdata = os.getenv('LOCALAPPDATA', os.getenv('APPDATA')) |
7ab05d752405
largefiles: cosmetics, whitespace, code style
Greg Ward <greg@gerg.ca>
parents:
15253
diff
changeset
|
90 path = os.path.join(appdata, longname, hash) |
15168 | 91 elif os.name == 'posix': |
92 path = os.path.join(os.getenv('HOME'), '.' + longname, hash) | |
93 else: | |
15253
67d010779907
largefiles: improve error reporting
Greg Ward <greg@gerg.ca>
parents:
15252
diff
changeset
|
94 raise util.Abort(_('unknown operating system: %s\n') % os.name) |
15168 | 95 return path |
96 | |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
97 def inusercache(ui, hash): |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
98 return os.path.exists(usercachepath(ui, hash)) |
15168 | 99 |
100 def findfile(repo, hash): | |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
101 if instore(repo, hash): |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
102 repo.ui.note(_('Found %s in store\n') % hash) |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
103 return storepath(repo, hash) |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
104 if inusercache(repo.ui, hash): |
15168 | 105 repo.ui.note(_('Found %s in system cache\n') % hash) |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
106 return usercachepath(repo.ui, hash) |
15168 | 107 return None |
108 | |
109 class largefiles_dirstate(dirstate.dirstate): | |
110 def __getitem__(self, key): | |
111 return super(largefiles_dirstate, self).__getitem__(unixpath(key)) | |
112 def normal(self, f): | |
113 return super(largefiles_dirstate, self).normal(unixpath(f)) | |
114 def remove(self, f): | |
115 return super(largefiles_dirstate, self).remove(unixpath(f)) | |
116 def add(self, f): | |
117 return super(largefiles_dirstate, self).add(unixpath(f)) | |
118 def drop(self, f): | |
119 return super(largefiles_dirstate, self).drop(unixpath(f)) | |
120 def forget(self, f): | |
121 return super(largefiles_dirstate, self).forget(unixpath(f)) | |
122 | |
123 def openlfdirstate(ui, repo): | |
124 ''' | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
125 Return a dirstate object that tracks largefiles: i.e. its root is |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
126 the repo root, but it is saved in .hg/largefiles/dirstate. |
15168 | 127 ''' |
128 admin = repo.join(longname) | |
15224
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
129 opener = scmutil.opener(admin) |
15169
aa262fff87ac
largefile: fix up hasattr usage
Matt Mackall <mpm@selenic.com>
parents:
15168
diff
changeset
|
130 if util.safehasattr(repo.dirstate, '_validate'): |
15168 | 131 lfdirstate = largefiles_dirstate(opener, ui, repo.root, |
132 repo.dirstate._validate) | |
133 else: | |
134 lfdirstate = largefiles_dirstate(opener, ui, repo.root) | |
135 | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
136 # If the largefiles dirstate does not exist, populate and create |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
137 # it. This ensures that we create it on the first meaningful |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
138 # largefiles operation in a new clone. It also gives us an easy |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
139 # way to forcibly rebuild largefiles state: |
15168 | 140 # rm .hg/largefiles/dirstate && hg status |
141 # Or even, if things are really messed up: | |
142 # rm -rf .hg/largefiles && hg status | |
143 if not os.path.exists(os.path.join(admin, 'dirstate')): | |
144 util.makedirs(admin) | |
145 matcher = getstandinmatcher(repo) | |
146 for standin in dirstate_walk(repo.dirstate, matcher): | |
147 lfile = splitstandin(standin) | |
148 hash = readstandin(repo, lfile) | |
149 lfdirstate.normallookup(lfile) | |
150 try: | |
151 if hash == hashfile(lfile): | |
152 lfdirstate.normal(lfile) | |
153 except IOError, err: | |
154 if err.errno != errno.ENOENT: | |
155 raise | |
156 | |
157 lfdirstate.write() | |
158 | |
159 return lfdirstate | |
160 | |
161 def lfdirstate_status(lfdirstate, repo, rev): | |
162 wlock = repo.wlock() | |
163 try: | |
164 match = match_.always(repo.root, repo.getcwd()) | |
165 s = lfdirstate.status(match, [], False, False, False) | |
166 unsure, modified, added, removed, missing, unknown, ignored, clean = s | |
167 for lfile in unsure: | |
168 if repo[rev][standin(lfile)].data().strip() != \ | |
169 hashfile(repo.wjoin(lfile)): | |
170 modified.append(lfile) | |
171 else: | |
172 clean.append(lfile) | |
173 lfdirstate.normal(lfile) | |
174 lfdirstate.write() | |
175 finally: | |
176 wlock.release() | |
177 return (modified, added, removed, missing, unknown, ignored, clean) | |
178 | |
179 def listlfiles(repo, rev=None, matcher=None): | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
180 '''return a list of largefiles in the working copy or the |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
181 specified changeset''' |
15168 | 182 |
183 if matcher is None: | |
184 matcher = getstandinmatcher(repo) | |
185 | |
186 # ignore unknown files in working directory | |
15255
7ab05d752405
largefiles: cosmetics, whitespace, code style
Greg Ward <greg@gerg.ca>
parents:
15253
diff
changeset
|
187 return [splitstandin(f) |
7ab05d752405
largefiles: cosmetics, whitespace, code style
Greg Ward <greg@gerg.ca>
parents:
15253
diff
changeset
|
188 for f in repo[rev].walk(matcher) |
15168 | 189 if rev is not None or repo.dirstate[f] != '?'] |
190 | |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
191 def instore(repo, hash): |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
192 return os.path.exists(storepath(repo, hash)) |
15168 | 193 |
194 def createdir(dir): | |
195 if not os.path.exists(dir): | |
196 os.makedirs(dir) | |
197 | |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
198 def storepath(repo, hash): |
15168 | 199 return repo.join(os.path.join(longname, hash)) |
200 | |
201 def copyfromcache(repo, hash, filename): | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
202 '''Copy the specified largefile from the repo or system cache to |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
203 filename in the repository. Return true on success or false if the |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
204 file was not found in either cache (which should not happened: |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
205 this is meant to be called only after ensuring that the needed |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
206 largefile exists in the cache).''' |
15168 | 207 path = findfile(repo, hash) |
208 if path is None: | |
209 return False | |
210 util.makedirs(os.path.dirname(repo.wjoin(filename))) | |
211 shutil.copy(path, repo.wjoin(filename)) | |
212 return True | |
213 | |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
214 def copytostore(repo, rev, file, uploaded=False): |
15168 | 215 hash = readstandin(repo, file) |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
216 if instore(repo, hash): |
15168 | 217 return |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
218 copytostoreabsolute(repo, repo.wjoin(file), hash) |
15168 | 219 |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
220 def copytostoreabsolute(repo, file, hash): |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
221 createdir(os.path.dirname(storepath(repo, hash))) |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
222 if inusercache(repo.ui, hash): |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
223 link(usercachepath(repo.ui, hash), storepath(repo, hash)) |
15168 | 224 else: |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
225 shutil.copyfile(file, storepath(repo, hash)) |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
226 os.chmod(storepath(repo, hash), os.stat(file).st_mode) |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
227 linktousercache(repo, hash) |
15168 | 228 |
15316
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
229 def linktousercache(repo, hash): |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
230 createdir(os.path.dirname(usercachepath(repo.ui, hash))) |
c65f5b6e26d4
largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents:
15304
diff
changeset
|
231 link(storepath(repo, hash), usercachepath(repo.ui, hash)) |
15168 | 232 |
233 def getstandinmatcher(repo, pats=[], opts={}): | |
234 '''Return a match object that applies pats to the standin directory''' | |
235 standindir = repo.pathto(shortname) | |
236 if pats: | |
237 # patterns supplied: search standin directory relative to current dir | |
238 cwd = repo.getcwd() | |
239 if os.path.isabs(cwd): | |
240 # cwd is an absolute path for hg -R <reponame> | |
241 # work relative to the repository root in this case | |
242 cwd = '' | |
243 pats = [os.path.join(standindir, cwd, pat) for pat in pats] | |
244 elif os.path.isdir(standindir): | |
245 # no patterns: relative to repo root | |
246 pats = [standindir] | |
247 else: | |
248 # no patterns and no standin dir: return matcher that matches nothing | |
249 match = match_.match(repo.root, None, [], exact=True) | |
250 match.matchfn = lambda f: False | |
251 return match | |
252 return getmatcher(repo, pats, opts, showbad=False) | |
253 | |
254 def getmatcher(repo, pats=[], opts={}, showbad=True): | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
255 '''Wrapper around scmutil.match() that adds showbad: if false, |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
256 neuter the match object's bad() method so it does not print any |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
257 warnings about missing files or directories.''' |
15224
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
258 match = scmutil.match(repo[None], pats, opts) |
15168 | 259 |
260 if not showbad: | |
261 match.bad = lambda f, msg: None | |
262 return match | |
263 | |
264 def composestandinmatcher(repo, rmatcher): | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
265 '''Return a matcher that accepts standins corresponding to the |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
266 files accepted by rmatcher. Pass the list of files in the matcher |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
267 as the paths specified by the user.''' |
15168 | 268 smatcher = getstandinmatcher(repo, rmatcher.files()) |
269 isstandin = smatcher.matchfn | |
270 def composed_matchfn(f): | |
271 return isstandin(f) and rmatcher.matchfn(splitstandin(f)) | |
272 smatcher.matchfn = composed_matchfn | |
273 | |
274 return smatcher | |
275 | |
276 def standin(filename): | |
277 '''Return the repo-relative path to the standin for the specified big | |
278 file.''' | |
279 # Notes: | |
280 # 1) Most callers want an absolute path, but _create_standin() needs | |
281 # it repo-relative so lfadd() can pass it to repo_add(). So leave | |
282 # it up to the caller to use repo.wjoin() to get an absolute path. | |
283 # 2) Join with '/' because that's what dirstate always uses, even on | |
284 # Windows. Change existing separator to '/' first in case we are | |
285 # passed filenames from an external source (like the command line). | |
286 return shortname + '/' + filename.replace(os.sep, '/') | |
287 | |
288 def isstandin(filename): | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
289 '''Return true if filename is a big file standin. filename must be |
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
290 in Mercurial's internal form (slash-separated).''' |
15168 | 291 return filename.startswith(shortname + '/') |
292 | |
293 def splitstandin(filename): | |
294 # Split on / because that's what dirstate always uses, even on Windows. | |
295 # Change local separator to / first just in case we are passed filenames | |
296 # from an external source (like the command line). | |
297 bits = filename.replace(os.sep, '/').split('/', 1) | |
298 if len(bits) == 2 and bits[0] == shortname: | |
299 return bits[1] | |
300 else: | |
301 return None | |
302 | |
303 def updatestandin(repo, standin): | |
304 file = repo.wjoin(splitstandin(standin)) | |
305 if os.path.exists(file): | |
306 hash = hashfile(file) | |
307 executable = getexecutable(file) | |
308 writestandin(repo, standin, hash, executable) | |
309 | |
310 def readstandin(repo, filename, node=None): | |
311 '''read hex hash from standin for filename at given node, or working | |
312 directory if no node is given''' | |
313 return repo[node][standin(filename)].data().strip() | |
314 | |
315 def writestandin(repo, standin, hash, executable): | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
316 '''write hash to <repo.root>/<standin>''' |
15168 | 317 writehash(hash, repo.wjoin(standin), executable) |
318 | |
319 def copyandhash(instream, outfile): | |
320 '''Read bytes from instream (iterable) and write them to outfile, | |
321 computing the SHA-1 hash of the data along the way. Close outfile | |
322 when done and return the binary hash.''' | |
323 hasher = util.sha1('') | |
324 for data in instream: | |
325 hasher.update(data) | |
326 outfile.write(data) | |
327 | |
328 # Blecch: closing a file that somebody else opened is rude and | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
329 # wrong. But it's so darn convenient and practical! After all, |
15168 | 330 # outfile was opened just to copy and hash. |
331 outfile.close() | |
332 | |
333 return hasher.digest() | |
334 | |
335 def hashrepofile(repo, file): | |
336 return hashfile(repo.wjoin(file)) | |
337 | |
338 def hashfile(file): | |
339 if not os.path.exists(file): | |
340 return '' | |
341 hasher = util.sha1('') | |
342 fd = open(file, 'rb') | |
343 for data in blockstream(fd): | |
344 hasher.update(data) | |
345 fd.close() | |
346 return hasher.hexdigest() | |
347 | |
348 class limitreader(object): | |
349 def __init__(self, f, limit): | |
350 self.f = f | |
351 self.limit = limit | |
352 | |
353 def read(self, length): | |
354 if self.limit == 0: | |
355 return '' | |
356 length = length > self.limit and self.limit or length | |
357 self.limit -= length | |
358 return self.f.read(length) | |
359 | |
360 def close(self): | |
361 pass | |
362 | |
363 def blockstream(infile, blocksize=128 * 1024): | |
364 """Generator that yields blocks of data from infile and closes infile.""" | |
365 while True: | |
366 data = infile.read(blocksize) | |
367 if not data: | |
368 break | |
369 yield data | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
370 # same blecch as copyandhash() above |
15168 | 371 infile.close() |
372 | |
373 def readhash(filename): | |
374 rfile = open(filename, 'rb') | |
375 hash = rfile.read(40) | |
376 rfile.close() | |
377 if len(hash) < 40: | |
378 raise util.Abort(_('bad hash in \'%s\' (only %d bytes long)') | |
379 % (filename, len(hash))) | |
380 return hash | |
381 | |
382 def writehash(hash, filename, executable): | |
383 util.makedirs(os.path.dirname(filename)) | |
384 if os.path.exists(filename): | |
385 os.unlink(filename) | |
386 wfile = open(filename, 'wb') | |
387 | |
388 try: | |
389 wfile.write(hash) | |
390 wfile.write('\n') | |
391 finally: | |
392 wfile.close() | |
393 if os.path.exists(filename): | |
394 os.chmod(filename, getmode(executable)) | |
395 | |
396 def getexecutable(filename): | |
397 mode = os.stat(filename).st_mode | |
15255
7ab05d752405
largefiles: cosmetics, whitespace, code style
Greg Ward <greg@gerg.ca>
parents:
15253
diff
changeset
|
398 return ((mode & stat.S_IXUSR) and |
7ab05d752405
largefiles: cosmetics, whitespace, code style
Greg Ward <greg@gerg.ca>
parents:
15253
diff
changeset
|
399 (mode & stat.S_IXGRP) and |
7ab05d752405
largefiles: cosmetics, whitespace, code style
Greg Ward <greg@gerg.ca>
parents:
15253
diff
changeset
|
400 (mode & stat.S_IXOTH)) |
15168 | 401 |
402 def getmode(executable): | |
403 if executable: | |
404 return 0755 | |
405 else: | |
406 return 0644 | |
407 | |
408 def urljoin(first, second, *arg): | |
409 def join(left, right): | |
410 if not left.endswith('/'): | |
411 left += '/' | |
412 if right.startswith('/'): | |
413 right = right[1:] | |
414 return left + right | |
415 | |
416 url = join(first, second) | |
417 for a in arg: | |
418 url = join(url, a) | |
419 return url | |
420 | |
421 def hexsha1(data): | |
422 """hexsha1 returns the hex-encoded sha1 sum of the data in the file-like | |
423 object data""" | |
424 h = hashlib.sha1() | |
425 for chunk in util.filechunkiter(data): | |
426 h.update(chunk) | |
427 return h.hexdigest() | |
428 | |
429 def httpsendfile(ui, filename): | |
15224
7c604d8c7e83
largefiles: remove pre-1.9 code from extension first bundled with 1.9
Na'Tosha Bard <natosha@unity3d.com>
parents:
15206
diff
changeset
|
430 return httpconnection.httpsendfile(ui, filename, 'rb') |
15168 | 431 |
432 def unixpath(path): | |
15252
6e809bb4f969
largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents:
15228
diff
changeset
|
433 '''Return a version of path normalized for use with the lfdirstate.''' |
15168 | 434 return os.path.normpath(path).replace(os.sep, '/') |
435 | |
436 def islfilesrepo(repo): | |
15170
c1a4a3220711
largefiles: fix over-long lines
Matt Mackall <mpm@selenic.com>
parents:
15169
diff
changeset
|
437 return ('largefiles' in repo.requirements and |
15188
8e115063950d
largefiles: don't break existing tests (syntax error, bad imports)
Greg Ward <greg@gerg.ca>
parents:
15171
diff
changeset
|
438 any_(shortname + '/' in f[0] for f in repo.store.datafiles())) |
15168 | 439 |
440 def any_(gen): | |
441 for x in gen: | |
442 if x: | |
443 return True | |
444 return False | |
445 | |
446 class storeprotonotcapable(BaseException): | |
447 def __init__(self, storetypes): | |
448 self.storetypes = storetypes |