annotate hgext/largefiles/basestore.py @ 16154:9b072a5f8f92 stable

largefiles: respect store.createmode in basestore.get This replaces another use of tempfile with atomictempfile. The problem with tempfile is that it creates files with 600 permissions instead of respecting repo.store.createmode.
author Martin Geisler <mg@aragost.com>
date Thu, 23 Feb 2012 13:37:10 +0100
parents f9efb325ea32
children d87d9d8a8e03
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
15168
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
1 # Copyright 2009-2010 Gregory P. Ward
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
2 # Copyright 2009-2010 Intelerad Medical Systems Incorporated
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
3 # Copyright 2010-2011 Fog Creek Software
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
4 # Copyright 2010-2011 Unity Technologies
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
5 #
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
6 # This software may be used and distributed according to the terms of the
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
7 # GNU General Public License version 2 or any later version.
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
8
15252
6e809bb4f969 largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents: 15169
diff changeset
9 '''base class for store implementations and store-related utility code'''
15168
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
10
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
11 import binascii
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
12 import re
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
13
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
14 from mercurial import util, node, hg
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
15 from mercurial.i18n import _
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
16
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
17 import lfutil
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
18
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
19 class StoreError(Exception):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
20 '''Raised when there is a problem getting files from or putting
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
21 files to a central store.'''
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
22 def __init__(self, filename, hash, url, detail):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
23 self.filename = filename
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
24 self.hash = hash
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
25 self.url = url
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
26 self.detail = detail
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
27
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
28 def longmessage(self):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
29 if self.url:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
30 return ('%s: %s\n'
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
31 '(failed URL: %s)\n'
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
32 % (self.filename, self.detail, self.url))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
33 else:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
34 return ('%s: %s\n'
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
35 '(no default or default-push path set in hgrc)\n'
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
36 % (self.filename, self.detail))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
37
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
38 def __str__(self):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
39 return "%s: %s" % (self.url, self.detail)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
40
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
41 class basestore(object):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
42 def __init__(self, ui, repo, url):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
43 self.ui = ui
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
44 self.repo = repo
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
45 self.url = url
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
46
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
47 def put(self, source, hash):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
48 '''Put source file into the store under <filename>/<hash>.'''
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
49 raise NotImplementedError('abstract method')
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
50
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
51 def exists(self, hash):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
52 '''Check to see if the store contains the given hash.'''
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
53 raise NotImplementedError('abstract method')
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
54
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
55 def get(self, files):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
56 '''Get the specified largefiles from the store and write to local
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
57 files under repo.root. files is a list of (filename, hash)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
58 tuples. Return (success, missing), lists of files successfuly
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
59 downloaded and those not found in the store. success is a list
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
60 of (filename, hash) tuples; missing is a list of filenames that
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
61 we could not get. (The detailed error message will already have
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
62 been presented to the user, so missing is just supplied as a
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
63 summary.)'''
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
64 success = []
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
65 missing = []
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
66 ui = self.ui
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
67
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
68 at = 0
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
69 for filename, hash in files:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
70 ui.progress(_('getting largefiles'), at, unit='lfile',
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
71 total=len(files))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
72 at += 1
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
73 ui.note(_('getting %s:%s\n') % (filename, hash))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
74
15316
c65f5b6e26d4 largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents: 15302
diff changeset
75 storefilename = lfutil.storepath(self.repo, hash)
16154
9b072a5f8f92 largefiles: respect store.createmode in basestore.get
Martin Geisler <mg@aragost.com>
parents: 15943
diff changeset
76 tmpfile = util.atomictempfile(storefilename,
9b072a5f8f92 largefiles: respect store.createmode in basestore.get
Martin Geisler <mg@aragost.com>
parents: 15943
diff changeset
77 createmode=self.repo.store.createmode)
15168
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
78
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
79 try:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
80 hhash = binascii.hexlify(self._getfile(tmpfile, filename, hash))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
81 except StoreError, err:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
82 ui.warn(err.longmessage())
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
83 hhash = ""
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
84
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
85 if hhash != hash:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
86 if hhash != "":
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
87 ui.warn(_('%s: data corruption (expected %s, got %s)\n')
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
88 % (filename, hash, hhash))
16154
9b072a5f8f92 largefiles: respect store.createmode in basestore.get
Martin Geisler <mg@aragost.com>
parents: 15943
diff changeset
89 tmpfile.discard() # no-op if it's already closed
15168
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
90 missing.append(filename)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
91 continue
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
92
16154
9b072a5f8f92 largefiles: respect store.createmode in basestore.get
Martin Geisler <mg@aragost.com>
parents: 15943
diff changeset
93 tmpfile.close()
15316
c65f5b6e26d4 largefiles: rename functions and methods to match desired behavior
Benjamin Pollack <benjamin@bitquabit.com>
parents: 15302
diff changeset
94 lfutil.linktousercache(self.repo, hash)
15168
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
95 success.append((filename, hhash))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
96
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
97 ui.progress(_('getting largefiles'), None)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
98 return (success, missing)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
99
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
100 def verify(self, revs, contents=False):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
101 '''Verify the existence (and, optionally, contents) of every big
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
102 file revision referenced by every changeset in revs.
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
103 Return 0 if all is well, non-zero on any errors.'''
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
104 write = self.ui.write
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
105 failed = False
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
106
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
107 write(_('searching %d changesets for largefiles\n') % len(revs))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
108 verified = set() # set of (filename, filenode) tuples
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
109
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
110 for rev in revs:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
111 cctx = self.repo[rev]
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
112 cset = "%d:%s" % (cctx.rev(), node.short(cctx.node()))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
113
15319
9da7e96cd5c2 largefiles: remove redundant any_ function
Benjamin Pollack <benjamin@bitquabit.com>
parents: 15316
diff changeset
114 failed = util.any(self._verifyfile(
15168
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
115 cctx, cset, contents, standin, verified) for standin in cctx)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
116
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
117 num_revs = len(verified)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
118 num_lfiles = len(set([fname for (fname, fnode) in verified]))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
119 if contents:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
120 write(_('verified contents of %d revisions of %d largefiles\n')
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
121 % (num_revs, num_lfiles))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
122 else:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
123 write(_('verified existence of %d revisions of %d largefiles\n')
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
124 % (num_revs, num_lfiles))
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
125
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
126 return int(failed)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
127
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
128 def _getfile(self, tmpfile, filename, hash):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
129 '''Fetch one revision of one file from the store and write it
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
130 to tmpfile. Compute the hash of the file on-the-fly as it
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
131 downloads and return the binary hash. Close tmpfile. Raise
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
132 StoreError if unable to download the file (e.g. it does not
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
133 exist in the store).'''
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
134 raise NotImplementedError('abstract method')
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
135
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
136 def _verifyfile(self, cctx, cset, contents, standin, verified):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
137 '''Perform the actual verification of a file in the store.
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
138 '''
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
139 raise NotImplementedError('abstract method')
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
140
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
141 import localstore, wirestore
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
142
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
143 _storeprovider = {
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
144 'file': [localstore.localstore],
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
145 'http': [wirestore.wirestore],
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
146 'https': [wirestore.wirestore],
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
147 'ssh': [wirestore.wirestore],
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
148 }
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
149
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
150 _scheme_re = re.compile(r'^([a-zA-Z0-9+-.]+)://')
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
151
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
152 # During clone this function is passed the src's ui object
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
153 # but it needs the dest's ui object so it can read out of
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
154 # the config file. Use repo.ui instead.
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
155 def _openstore(repo, remote=None, put=False):
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
156 ui = repo.ui
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
157
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
158 if not remote:
15943
f9efb325ea32 largefiles: fix caching largefiles from an aliased repo (issue3212)
Na'Tosha Bard <natosha@unity3d.com>
parents: 15319
diff changeset
159 lfpullsource = getattr(repo, 'lfpullsource', None)
f9efb325ea32 largefiles: fix caching largefiles from an aliased repo (issue3212)
Na'Tosha Bard <natosha@unity3d.com>
parents: 15319
diff changeset
160 if lfpullsource:
f9efb325ea32 largefiles: fix caching largefiles from an aliased repo (issue3212)
Na'Tosha Bard <natosha@unity3d.com>
parents: 15319
diff changeset
161 path = ui.expandpath(lfpullsource)
f9efb325ea32 largefiles: fix caching largefiles from an aliased repo (issue3212)
Na'Tosha Bard <natosha@unity3d.com>
parents: 15319
diff changeset
162 else:
f9efb325ea32 largefiles: fix caching largefiles from an aliased repo (issue3212)
Na'Tosha Bard <natosha@unity3d.com>
parents: 15319
diff changeset
163 path = ui.expandpath('default-push', 'default')
15252
6e809bb4f969 largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents: 15169
diff changeset
164
6e809bb4f969 largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents: 15169
diff changeset
165 # ui.expandpath() leaves 'default-push' and 'default' alone if
6e809bb4f969 largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents: 15169
diff changeset
166 # they cannot be expanded: fallback to the empty string,
6e809bb4f969 largefiles: improve comments, internal docstrings
Greg Ward <greg@gerg.ca>
parents: 15169
diff changeset
167 # meaning the current directory.
15168
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
168 if path == 'default-push' or path == 'default':
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
169 path = ''
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
170 remote = repo
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
171 else:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
172 remote = hg.peer(repo, {}, path)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
173
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
174 # The path could be a scheme so use Mercurial's normal functionality
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
175 # to resolve the scheme to a repository and use its path
15169
aa262fff87ac largefile: fix up hasattr usage
Matt Mackall <mpm@selenic.com>
parents: 15168
diff changeset
176 path = util.safehasattr(remote, 'url') and remote.url() or remote.path
15168
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
177
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
178 match = _scheme_re.match(path)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
179 if not match: # regular filesystem path
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
180 scheme = 'file'
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
181 else:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
182 scheme = match.group(1)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
183
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
184 try:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
185 storeproviders = _storeprovider[scheme]
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
186 except KeyError:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
187 raise util.Abort(_('unsupported URL scheme %r') % scheme)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
188
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
189 for class_obj in storeproviders:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
190 try:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
191 return class_obj(ui, repo, remote)
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
192 except lfutil.storeprotonotcapable:
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
193 pass
cfccd3bee7b3 hgext: add largefiles extension
various
parents:
diff changeset
194
15302
225d30bacabd largefiles: string formatting typo in basestore._openstore where comma is used instead of modulo
Hao Lian <hao@fogcreek.com>
parents: 15255
diff changeset
195 raise util.Abort(_('%s does not appear to be a largefile store') % path)