##// END OF EJS Templates
util & scmutil: adapt read/write helpers as request by mpm
Dan Villiom Podlaski Christiansen -
r14167:0e475380 default
parent child Browse files
Show More
@@ -1,458 +1,465 b''
1 # scmutil.py - Mercurial core utility functions
1 # scmutil.py - Mercurial core utility functions
2 #
2 #
3 # Copyright Matt Mackall <mpm@selenic.com>
3 # Copyright Matt Mackall <mpm@selenic.com>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2 or any later version.
6 # GNU General Public License version 2 or any later version.
7
7
8 from i18n import _
8 from i18n import _
9 import util, error, osutil
9 import util, error, osutil
10 import os, errno, stat, sys
10 import os, errno, stat, sys
11
11
12 def checkfilename(f):
12 def checkfilename(f):
13 '''Check that the filename f is an acceptable filename for a tracked file'''
13 '''Check that the filename f is an acceptable filename for a tracked file'''
14 if '\r' in f or '\n' in f:
14 if '\r' in f or '\n' in f:
15 raise util.Abort(_("'\\n' and '\\r' disallowed in filenames: %r") % f)
15 raise util.Abort(_("'\\n' and '\\r' disallowed in filenames: %r") % f)
16
16
17 def checkportable(ui, f):
17 def checkportable(ui, f):
18 '''Check if filename f is portable and warn or abort depending on config'''
18 '''Check if filename f is portable and warn or abort depending on config'''
19 checkfilename(f)
19 checkfilename(f)
20 abort, warn = checkportabilityalert(ui)
20 abort, warn = checkportabilityalert(ui)
21 if abort or warn:
21 if abort or warn:
22 msg = util.checkwinfilename(f)
22 msg = util.checkwinfilename(f)
23 if msg:
23 if msg:
24 msg = "%s: %r" % (msg, f)
24 msg = "%s: %r" % (msg, f)
25 if abort:
25 if abort:
26 raise util.Abort(msg)
26 raise util.Abort(msg)
27 ui.warn(_("warning: %s\n") % msg)
27 ui.warn(_("warning: %s\n") % msg)
28
28
29 def checkportabilityalert(ui):
29 def checkportabilityalert(ui):
30 '''check if the user's config requests nothing, a warning, or abort for
30 '''check if the user's config requests nothing, a warning, or abort for
31 non-portable filenames'''
31 non-portable filenames'''
32 val = ui.config('ui', 'portablefilenames', 'warn')
32 val = ui.config('ui', 'portablefilenames', 'warn')
33 lval = val.lower()
33 lval = val.lower()
34 bval = util.parsebool(val)
34 bval = util.parsebool(val)
35 abort = os.name == 'nt' or lval == 'abort'
35 abort = os.name == 'nt' or lval == 'abort'
36 warn = bval or lval == 'warn'
36 warn = bval or lval == 'warn'
37 if bval is None and not (warn or abort or lval == 'ignore'):
37 if bval is None and not (warn or abort or lval == 'ignore'):
38 raise error.ConfigError(
38 raise error.ConfigError(
39 _("ui.portablefilenames value is invalid ('%s')") % val)
39 _("ui.portablefilenames value is invalid ('%s')") % val)
40 return abort, warn
40 return abort, warn
41
41
42 class casecollisionauditor(object):
42 class casecollisionauditor(object):
43 def __init__(self, ui, abort, existingiter):
43 def __init__(self, ui, abort, existingiter):
44 self._ui = ui
44 self._ui = ui
45 self._abort = abort
45 self._abort = abort
46 self._map = {}
46 self._map = {}
47 for f in existingiter:
47 for f in existingiter:
48 self._map[f.lower()] = f
48 self._map[f.lower()] = f
49
49
50 def __call__(self, f):
50 def __call__(self, f):
51 fl = f.lower()
51 fl = f.lower()
52 map = self._map
52 map = self._map
53 if fl in map and map[fl] != f:
53 if fl in map and map[fl] != f:
54 msg = _('possible case-folding collision for %s') % f
54 msg = _('possible case-folding collision for %s') % f
55 if self._abort:
55 if self._abort:
56 raise util.Abort(msg)
56 raise util.Abort(msg)
57 self._ui.warn(_("warning: %s\n") % msg)
57 self._ui.warn(_("warning: %s\n") % msg)
58 map[fl] = f
58 map[fl] = f
59
59
60 class path_auditor(object):
60 class path_auditor(object):
61 '''ensure that a filesystem path contains no banned components.
61 '''ensure that a filesystem path contains no banned components.
62 the following properties of a path are checked:
62 the following properties of a path are checked:
63
63
64 - ends with a directory separator
64 - ends with a directory separator
65 - under top-level .hg
65 - under top-level .hg
66 - starts at the root of a windows drive
66 - starts at the root of a windows drive
67 - contains ".."
67 - contains ".."
68 - traverses a symlink (e.g. a/symlink_here/b)
68 - traverses a symlink (e.g. a/symlink_here/b)
69 - inside a nested repository (a callback can be used to approve
69 - inside a nested repository (a callback can be used to approve
70 some nested repositories, e.g., subrepositories)
70 some nested repositories, e.g., subrepositories)
71 '''
71 '''
72
72
73 def __init__(self, root, callback=None):
73 def __init__(self, root, callback=None):
74 self.audited = set()
74 self.audited = set()
75 self.auditeddir = set()
75 self.auditeddir = set()
76 self.root = root
76 self.root = root
77 self.callback = callback
77 self.callback = callback
78
78
79 def __call__(self, path):
79 def __call__(self, path):
80 '''Check the relative path.
80 '''Check the relative path.
81 path may contain a pattern (e.g. foodir/**.txt)'''
81 path may contain a pattern (e.g. foodir/**.txt)'''
82
82
83 if path in self.audited:
83 if path in self.audited:
84 return
84 return
85 # AIX ignores "/" at end of path, others raise EISDIR.
85 # AIX ignores "/" at end of path, others raise EISDIR.
86 if util.endswithsep(path):
86 if util.endswithsep(path):
87 raise util.Abort(_("path ends in directory separator: %s") % path)
87 raise util.Abort(_("path ends in directory separator: %s") % path)
88 normpath = os.path.normcase(path)
88 normpath = os.path.normcase(path)
89 parts = util.splitpath(normpath)
89 parts = util.splitpath(normpath)
90 if (os.path.splitdrive(path)[0]
90 if (os.path.splitdrive(path)[0]
91 or parts[0].lower() in ('.hg', '.hg.', '')
91 or parts[0].lower() in ('.hg', '.hg.', '')
92 or os.pardir in parts):
92 or os.pardir in parts):
93 raise util.Abort(_("path contains illegal component: %s") % path)
93 raise util.Abort(_("path contains illegal component: %s") % path)
94 if '.hg' in path.lower():
94 if '.hg' in path.lower():
95 lparts = [p.lower() for p in parts]
95 lparts = [p.lower() for p in parts]
96 for p in '.hg', '.hg.':
96 for p in '.hg', '.hg.':
97 if p in lparts[1:]:
97 if p in lparts[1:]:
98 pos = lparts.index(p)
98 pos = lparts.index(p)
99 base = os.path.join(*parts[:pos])
99 base = os.path.join(*parts[:pos])
100 raise util.Abort(_('path %r is inside nested repo %r')
100 raise util.Abort(_('path %r is inside nested repo %r')
101 % (path, base))
101 % (path, base))
102
102
103 parts.pop()
103 parts.pop()
104 prefixes = []
104 prefixes = []
105 while parts:
105 while parts:
106 prefix = os.sep.join(parts)
106 prefix = os.sep.join(parts)
107 if prefix in self.auditeddir:
107 if prefix in self.auditeddir:
108 break
108 break
109 curpath = os.path.join(self.root, prefix)
109 curpath = os.path.join(self.root, prefix)
110 try:
110 try:
111 st = os.lstat(curpath)
111 st = os.lstat(curpath)
112 except OSError, err:
112 except OSError, err:
113 # EINVAL can be raised as invalid path syntax under win32.
113 # EINVAL can be raised as invalid path syntax under win32.
114 # They must be ignored for patterns can be checked too.
114 # They must be ignored for patterns can be checked too.
115 if err.errno not in (errno.ENOENT, errno.ENOTDIR, errno.EINVAL):
115 if err.errno not in (errno.ENOENT, errno.ENOTDIR, errno.EINVAL):
116 raise
116 raise
117 else:
117 else:
118 if stat.S_ISLNK(st.st_mode):
118 if stat.S_ISLNK(st.st_mode):
119 raise util.Abort(
119 raise util.Abort(
120 _('path %r traverses symbolic link %r')
120 _('path %r traverses symbolic link %r')
121 % (path, prefix))
121 % (path, prefix))
122 elif (stat.S_ISDIR(st.st_mode) and
122 elif (stat.S_ISDIR(st.st_mode) and
123 os.path.isdir(os.path.join(curpath, '.hg'))):
123 os.path.isdir(os.path.join(curpath, '.hg'))):
124 if not self.callback or not self.callback(curpath):
124 if not self.callback or not self.callback(curpath):
125 raise util.Abort(_('path %r is inside nested repo %r') %
125 raise util.Abort(_('path %r is inside nested repo %r') %
126 (path, prefix))
126 (path, prefix))
127 prefixes.append(prefix)
127 prefixes.append(prefix)
128 parts.pop()
128 parts.pop()
129
129
130 self.audited.add(path)
130 self.audited.add(path)
131 # only add prefixes to the cache after checking everything: we don't
131 # only add prefixes to the cache after checking everything: we don't
132 # want to add "foo/bar/baz" before checking if there's a "foo/.hg"
132 # want to add "foo/bar/baz" before checking if there's a "foo/.hg"
133 self.auditeddir.update(prefixes)
133 self.auditeddir.update(prefixes)
134
134
135 class abstractopener(object):
135 class abstractopener(object):
136 """Abstract base class; cannot be instantiated"""
136 """Abstract base class; cannot be instantiated"""
137
137
138 def __init__(self, *args, **kwargs):
138 def __init__(self, *args, **kwargs):
139 '''Prevent instantiation; don't call this from subclasses.'''
139 '''Prevent instantiation; don't call this from subclasses.'''
140 raise NotImplementedError('attempted instantiating ' + str(type(self)))
140 raise NotImplementedError('attempted instantiating ' + str(type(self)))
141
141
142 def read(self, *args, **kwargs):
142 def read(self, path):
143 fp = self(*args, **kwargs)
143 fp = self(path, 'rb')
144 try:
144 try:
145 return fp.read()
145 return fp.read()
146 finally:
146 finally:
147 fp.close()
147 fp.close()
148
148
149 def write(self, data, *args, **kwargs):
149 def write(self, path, data):
150 fp = self(*args, **kwargs)
150 fp = self(path, 'wb')
151 try:
152 return fp.write(data)
153 finally:
154 fp.close()
155
156 def append(self, path, data):
157 fp = self(path, 'ab')
151 try:
158 try:
152 return fp.write(data)
159 return fp.write(data)
153 finally:
160 finally:
154 fp.close()
161 fp.close()
155
162
156 class opener(abstractopener):
163 class opener(abstractopener):
157 '''Open files relative to a base directory
164 '''Open files relative to a base directory
158
165
159 This class is used to hide the details of COW semantics and
166 This class is used to hide the details of COW semantics and
160 remote file access from higher level code.
167 remote file access from higher level code.
161 '''
168 '''
162 def __init__(self, base, audit=True):
169 def __init__(self, base, audit=True):
163 self.base = base
170 self.base = base
164 if audit:
171 if audit:
165 self.auditor = path_auditor(base)
172 self.auditor = path_auditor(base)
166 else:
173 else:
167 self.auditor = util.always
174 self.auditor = util.always
168 self.createmode = None
175 self.createmode = None
169 self._trustnlink = None
176 self._trustnlink = None
170
177
171 @util.propertycache
178 @util.propertycache
172 def _can_symlink(self):
179 def _can_symlink(self):
173 return util.checklink(self.base)
180 return util.checklink(self.base)
174
181
175 def _fixfilemode(self, name):
182 def _fixfilemode(self, name):
176 if self.createmode is None:
183 if self.createmode is None:
177 return
184 return
178 os.chmod(name, self.createmode & 0666)
185 os.chmod(name, self.createmode & 0666)
179
186
180 def __call__(self, path, mode="r", text=False, atomictemp=False):
187 def __call__(self, path, mode="r", text=False, atomictemp=False):
181 r = util.checkosfilename(path)
188 r = util.checkosfilename(path)
182 if r:
189 if r:
183 raise util.Abort("%s: %r" % (r, path))
190 raise util.Abort("%s: %r" % (r, path))
184 self.auditor(path)
191 self.auditor(path)
185 f = os.path.join(self.base, path)
192 f = os.path.join(self.base, path)
186
193
187 if not text and "b" not in mode:
194 if not text and "b" not in mode:
188 mode += "b" # for that other OS
195 mode += "b" # for that other OS
189
196
190 nlink = -1
197 nlink = -1
191 dirname, basename = os.path.split(f)
198 dirname, basename = os.path.split(f)
192 # If basename is empty, then the path is malformed because it points
199 # If basename is empty, then the path is malformed because it points
193 # to a directory. Let the posixfile() call below raise IOError.
200 # to a directory. Let the posixfile() call below raise IOError.
194 if basename and mode not in ('r', 'rb'):
201 if basename and mode not in ('r', 'rb'):
195 if atomictemp:
202 if atomictemp:
196 if not os.path.isdir(dirname):
203 if not os.path.isdir(dirname):
197 util.makedirs(dirname, self.createmode)
204 util.makedirs(dirname, self.createmode)
198 return util.atomictempfile(f, mode, self.createmode)
205 return util.atomictempfile(f, mode, self.createmode)
199 try:
206 try:
200 if 'w' in mode:
207 if 'w' in mode:
201 util.unlink(f)
208 util.unlink(f)
202 nlink = 0
209 nlink = 0
203 else:
210 else:
204 # nlinks() may behave differently for files on Windows
211 # nlinks() may behave differently for files on Windows
205 # shares if the file is open.
212 # shares if the file is open.
206 fd = util.posixfile(f)
213 fd = util.posixfile(f)
207 nlink = util.nlinks(f)
214 nlink = util.nlinks(f)
208 if nlink < 1:
215 if nlink < 1:
209 nlink = 2 # force mktempcopy (issue1922)
216 nlink = 2 # force mktempcopy (issue1922)
210 fd.close()
217 fd.close()
211 except (OSError, IOError), e:
218 except (OSError, IOError), e:
212 if e.errno != errno.ENOENT:
219 if e.errno != errno.ENOENT:
213 raise
220 raise
214 nlink = 0
221 nlink = 0
215 if not os.path.isdir(dirname):
222 if not os.path.isdir(dirname):
216 util.makedirs(dirname, self.createmode)
223 util.makedirs(dirname, self.createmode)
217 if nlink > 0:
224 if nlink > 0:
218 if self._trustnlink is None:
225 if self._trustnlink is None:
219 self._trustnlink = nlink > 1 or util.checknlink(f)
226 self._trustnlink = nlink > 1 or util.checknlink(f)
220 if nlink > 1 or not self._trustnlink:
227 if nlink > 1 or not self._trustnlink:
221 util.rename(util.mktempcopy(f), f)
228 util.rename(util.mktempcopy(f), f)
222 fp = util.posixfile(f, mode)
229 fp = util.posixfile(f, mode)
223 if nlink == 0:
230 if nlink == 0:
224 self._fixfilemode(f)
231 self._fixfilemode(f)
225 return fp
232 return fp
226
233
227 def symlink(self, src, dst):
234 def symlink(self, src, dst):
228 self.auditor(dst)
235 self.auditor(dst)
229 linkname = os.path.join(self.base, dst)
236 linkname = os.path.join(self.base, dst)
230 try:
237 try:
231 os.unlink(linkname)
238 os.unlink(linkname)
232 except OSError:
239 except OSError:
233 pass
240 pass
234
241
235 dirname = os.path.dirname(linkname)
242 dirname = os.path.dirname(linkname)
236 if not os.path.exists(dirname):
243 if not os.path.exists(dirname):
237 util.makedirs(dirname, self.createmode)
244 util.makedirs(dirname, self.createmode)
238
245
239 if self._can_symlink:
246 if self._can_symlink:
240 try:
247 try:
241 os.symlink(src, linkname)
248 os.symlink(src, linkname)
242 except OSError, err:
249 except OSError, err:
243 raise OSError(err.errno, _('could not symlink to %r: %s') %
250 raise OSError(err.errno, _('could not symlink to %r: %s') %
244 (src, err.strerror), linkname)
251 (src, err.strerror), linkname)
245 else:
252 else:
246 f = self(dst, "w")
253 f = self(dst, "w")
247 f.write(src)
254 f.write(src)
248 f.close()
255 f.close()
249 self._fixfilemode(dst)
256 self._fixfilemode(dst)
250
257
251 class filteropener(abstractopener):
258 class filteropener(abstractopener):
252 '''Wrapper opener for filtering filenames with a function.'''
259 '''Wrapper opener for filtering filenames with a function.'''
253
260
254 def __init__(self, opener, filter):
261 def __init__(self, opener, filter):
255 self._filter = filter
262 self._filter = filter
256 self._orig = opener
263 self._orig = opener
257
264
258 def __call__(self, path, *args, **kwargs):
265 def __call__(self, path, *args, **kwargs):
259 return self._orig(self._filter(path), *args, **kwargs)
266 return self._orig(self._filter(path), *args, **kwargs)
260
267
261 def canonpath(root, cwd, myname, auditor=None):
268 def canonpath(root, cwd, myname, auditor=None):
262 '''return the canonical path of myname, given cwd and root'''
269 '''return the canonical path of myname, given cwd and root'''
263 if util.endswithsep(root):
270 if util.endswithsep(root):
264 rootsep = root
271 rootsep = root
265 else:
272 else:
266 rootsep = root + os.sep
273 rootsep = root + os.sep
267 name = myname
274 name = myname
268 if not os.path.isabs(name):
275 if not os.path.isabs(name):
269 name = os.path.join(root, cwd, name)
276 name = os.path.join(root, cwd, name)
270 name = os.path.normpath(name)
277 name = os.path.normpath(name)
271 if auditor is None:
278 if auditor is None:
272 auditor = path_auditor(root)
279 auditor = path_auditor(root)
273 if name != rootsep and name.startswith(rootsep):
280 if name != rootsep and name.startswith(rootsep):
274 name = name[len(rootsep):]
281 name = name[len(rootsep):]
275 auditor(name)
282 auditor(name)
276 return util.pconvert(name)
283 return util.pconvert(name)
277 elif name == root:
284 elif name == root:
278 return ''
285 return ''
279 else:
286 else:
280 # Determine whether `name' is in the hierarchy at or beneath `root',
287 # Determine whether `name' is in the hierarchy at or beneath `root',
281 # by iterating name=dirname(name) until that causes no change (can't
288 # by iterating name=dirname(name) until that causes no change (can't
282 # check name == '/', because that doesn't work on windows). For each
289 # check name == '/', because that doesn't work on windows). For each
283 # `name', compare dev/inode numbers. If they match, the list `rel'
290 # `name', compare dev/inode numbers. If they match, the list `rel'
284 # holds the reversed list of components making up the relative file
291 # holds the reversed list of components making up the relative file
285 # name we want.
292 # name we want.
286 root_st = os.stat(root)
293 root_st = os.stat(root)
287 rel = []
294 rel = []
288 while True:
295 while True:
289 try:
296 try:
290 name_st = os.stat(name)
297 name_st = os.stat(name)
291 except OSError:
298 except OSError:
292 break
299 break
293 if util.samestat(name_st, root_st):
300 if util.samestat(name_st, root_st):
294 if not rel:
301 if not rel:
295 # name was actually the same as root (maybe a symlink)
302 # name was actually the same as root (maybe a symlink)
296 return ''
303 return ''
297 rel.reverse()
304 rel.reverse()
298 name = os.path.join(*rel)
305 name = os.path.join(*rel)
299 auditor(name)
306 auditor(name)
300 return util.pconvert(name)
307 return util.pconvert(name)
301 dirname, basename = os.path.split(name)
308 dirname, basename = os.path.split(name)
302 rel.append(basename)
309 rel.append(basename)
303 if dirname == name:
310 if dirname == name:
304 break
311 break
305 name = dirname
312 name = dirname
306
313
307 raise util.Abort('%s not under root' % myname)
314 raise util.Abort('%s not under root' % myname)
308
315
309 def walkrepos(path, followsym=False, seen_dirs=None, recurse=False):
316 def walkrepos(path, followsym=False, seen_dirs=None, recurse=False):
310 '''yield every hg repository under path, recursively.'''
317 '''yield every hg repository under path, recursively.'''
311 def errhandler(err):
318 def errhandler(err):
312 if err.filename == path:
319 if err.filename == path:
313 raise err
320 raise err
314 if followsym and hasattr(os.path, 'samestat'):
321 if followsym and hasattr(os.path, 'samestat'):
315 def _add_dir_if_not_there(dirlst, dirname):
322 def _add_dir_if_not_there(dirlst, dirname):
316 match = False
323 match = False
317 samestat = os.path.samestat
324 samestat = os.path.samestat
318 dirstat = os.stat(dirname)
325 dirstat = os.stat(dirname)
319 for lstdirstat in dirlst:
326 for lstdirstat in dirlst:
320 if samestat(dirstat, lstdirstat):
327 if samestat(dirstat, lstdirstat):
321 match = True
328 match = True
322 break
329 break
323 if not match:
330 if not match:
324 dirlst.append(dirstat)
331 dirlst.append(dirstat)
325 return not match
332 return not match
326 else:
333 else:
327 followsym = False
334 followsym = False
328
335
329 if (seen_dirs is None) and followsym:
336 if (seen_dirs is None) and followsym:
330 seen_dirs = []
337 seen_dirs = []
331 _add_dir_if_not_there(seen_dirs, path)
338 _add_dir_if_not_there(seen_dirs, path)
332 for root, dirs, files in os.walk(path, topdown=True, onerror=errhandler):
339 for root, dirs, files in os.walk(path, topdown=True, onerror=errhandler):
333 dirs.sort()
340 dirs.sort()
334 if '.hg' in dirs:
341 if '.hg' in dirs:
335 yield root # found a repository
342 yield root # found a repository
336 qroot = os.path.join(root, '.hg', 'patches')
343 qroot = os.path.join(root, '.hg', 'patches')
337 if os.path.isdir(os.path.join(qroot, '.hg')):
344 if os.path.isdir(os.path.join(qroot, '.hg')):
338 yield qroot # we have a patch queue repo here
345 yield qroot # we have a patch queue repo here
339 if recurse:
346 if recurse:
340 # avoid recursing inside the .hg directory
347 # avoid recursing inside the .hg directory
341 dirs.remove('.hg')
348 dirs.remove('.hg')
342 else:
349 else:
343 dirs[:] = [] # don't descend further
350 dirs[:] = [] # don't descend further
344 elif followsym:
351 elif followsym:
345 newdirs = []
352 newdirs = []
346 for d in dirs:
353 for d in dirs:
347 fname = os.path.join(root, d)
354 fname = os.path.join(root, d)
348 if _add_dir_if_not_there(seen_dirs, fname):
355 if _add_dir_if_not_there(seen_dirs, fname):
349 if os.path.islink(fname):
356 if os.path.islink(fname):
350 for hgname in walkrepos(fname, True, seen_dirs):
357 for hgname in walkrepos(fname, True, seen_dirs):
351 yield hgname
358 yield hgname
352 else:
359 else:
353 newdirs.append(d)
360 newdirs.append(d)
354 dirs[:] = newdirs
361 dirs[:] = newdirs
355
362
356 def os_rcpath():
363 def os_rcpath():
357 '''return default os-specific hgrc search path'''
364 '''return default os-specific hgrc search path'''
358 path = system_rcpath()
365 path = system_rcpath()
359 path.extend(user_rcpath())
366 path.extend(user_rcpath())
360 path = [os.path.normpath(f) for f in path]
367 path = [os.path.normpath(f) for f in path]
361 return path
368 return path
362
369
363 _rcpath = None
370 _rcpath = None
364
371
365 def rcpath():
372 def rcpath():
366 '''return hgrc search path. if env var HGRCPATH is set, use it.
373 '''return hgrc search path. if env var HGRCPATH is set, use it.
367 for each item in path, if directory, use files ending in .rc,
374 for each item in path, if directory, use files ending in .rc,
368 else use item.
375 else use item.
369 make HGRCPATH empty to only look in .hg/hgrc of current repo.
376 make HGRCPATH empty to only look in .hg/hgrc of current repo.
370 if no HGRCPATH, use default os-specific path.'''
377 if no HGRCPATH, use default os-specific path.'''
371 global _rcpath
378 global _rcpath
372 if _rcpath is None:
379 if _rcpath is None:
373 if 'HGRCPATH' in os.environ:
380 if 'HGRCPATH' in os.environ:
374 _rcpath = []
381 _rcpath = []
375 for p in os.environ['HGRCPATH'].split(os.pathsep):
382 for p in os.environ['HGRCPATH'].split(os.pathsep):
376 if not p:
383 if not p:
377 continue
384 continue
378 p = util.expandpath(p)
385 p = util.expandpath(p)
379 if os.path.isdir(p):
386 if os.path.isdir(p):
380 for f, kind in osutil.listdir(p):
387 for f, kind in osutil.listdir(p):
381 if f.endswith('.rc'):
388 if f.endswith('.rc'):
382 _rcpath.append(os.path.join(p, f))
389 _rcpath.append(os.path.join(p, f))
383 else:
390 else:
384 _rcpath.append(p)
391 _rcpath.append(p)
385 else:
392 else:
386 _rcpath = os_rcpath()
393 _rcpath = os_rcpath()
387 return _rcpath
394 return _rcpath
388
395
389 if os.name != 'nt':
396 if os.name != 'nt':
390
397
391 def rcfiles(path):
398 def rcfiles(path):
392 rcs = [os.path.join(path, 'hgrc')]
399 rcs = [os.path.join(path, 'hgrc')]
393 rcdir = os.path.join(path, 'hgrc.d')
400 rcdir = os.path.join(path, 'hgrc.d')
394 try:
401 try:
395 rcs.extend([os.path.join(rcdir, f)
402 rcs.extend([os.path.join(rcdir, f)
396 for f, kind in osutil.listdir(rcdir)
403 for f, kind in osutil.listdir(rcdir)
397 if f.endswith(".rc")])
404 if f.endswith(".rc")])
398 except OSError:
405 except OSError:
399 pass
406 pass
400 return rcs
407 return rcs
401
408
402 def system_rcpath():
409 def system_rcpath():
403 path = []
410 path = []
404 # old mod_python does not set sys.argv
411 # old mod_python does not set sys.argv
405 if len(getattr(sys, 'argv', [])) > 0:
412 if len(getattr(sys, 'argv', [])) > 0:
406 path.extend(rcfiles(os.path.dirname(sys.argv[0]) +
413 path.extend(rcfiles(os.path.dirname(sys.argv[0]) +
407 '/../etc/mercurial'))
414 '/../etc/mercurial'))
408 path.extend(rcfiles('/etc/mercurial'))
415 path.extend(rcfiles('/etc/mercurial'))
409 return path
416 return path
410
417
411 def user_rcpath():
418 def user_rcpath():
412 return [os.path.expanduser('~/.hgrc')]
419 return [os.path.expanduser('~/.hgrc')]
413
420
414 else:
421 else:
415
422
416 _HKEY_LOCAL_MACHINE = 0x80000002L
423 _HKEY_LOCAL_MACHINE = 0x80000002L
417
424
418 def system_rcpath():
425 def system_rcpath():
419 '''return default os-specific hgrc search path'''
426 '''return default os-specific hgrc search path'''
420 rcpath = []
427 rcpath = []
421 filename = util.executable_path()
428 filename = util.executable_path()
422 # Use mercurial.ini found in directory with hg.exe
429 # Use mercurial.ini found in directory with hg.exe
423 progrc = os.path.join(os.path.dirname(filename), 'mercurial.ini')
430 progrc = os.path.join(os.path.dirname(filename), 'mercurial.ini')
424 if os.path.isfile(progrc):
431 if os.path.isfile(progrc):
425 rcpath.append(progrc)
432 rcpath.append(progrc)
426 return rcpath
433 return rcpath
427 # Use hgrc.d found in directory with hg.exe
434 # Use hgrc.d found in directory with hg.exe
428 progrcd = os.path.join(os.path.dirname(filename), 'hgrc.d')
435 progrcd = os.path.join(os.path.dirname(filename), 'hgrc.d')
429 if os.path.isdir(progrcd):
436 if os.path.isdir(progrcd):
430 for f, kind in osutil.listdir(progrcd):
437 for f, kind in osutil.listdir(progrcd):
431 if f.endswith('.rc'):
438 if f.endswith('.rc'):
432 rcpath.append(os.path.join(progrcd, f))
439 rcpath.append(os.path.join(progrcd, f))
433 return rcpath
440 return rcpath
434 # else look for a system rcpath in the registry
441 # else look for a system rcpath in the registry
435 value = util.lookup_reg('SOFTWARE\\Mercurial', None,
442 value = util.lookup_reg('SOFTWARE\\Mercurial', None,
436 _HKEY_LOCAL_MACHINE)
443 _HKEY_LOCAL_MACHINE)
437 if not isinstance(value, str) or not value:
444 if not isinstance(value, str) or not value:
438 return rcpath
445 return rcpath
439 value = value.replace('/', os.sep)
446 value = value.replace('/', os.sep)
440 for p in value.split(os.pathsep):
447 for p in value.split(os.pathsep):
441 if p.lower().endswith('mercurial.ini'):
448 if p.lower().endswith('mercurial.ini'):
442 rcpath.append(p)
449 rcpath.append(p)
443 elif os.path.isdir(p):
450 elif os.path.isdir(p):
444 for f, kind in osutil.listdir(p):
451 for f, kind in osutil.listdir(p):
445 if f.endswith('.rc'):
452 if f.endswith('.rc'):
446 rcpath.append(os.path.join(p, f))
453 rcpath.append(os.path.join(p, f))
447 return rcpath
454 return rcpath
448
455
449 def user_rcpath():
456 def user_rcpath():
450 '''return os-specific hgrc search path to the user dir'''
457 '''return os-specific hgrc search path to the user dir'''
451 home = os.path.expanduser('~')
458 home = os.path.expanduser('~')
452 path = [os.path.join(home, 'mercurial.ini'),
459 path = [os.path.join(home, 'mercurial.ini'),
453 os.path.join(home, '.hgrc')]
460 os.path.join(home, '.hgrc')]
454 userprofile = os.environ.get('USERPROFILE')
461 userprofile = os.environ.get('USERPROFILE')
455 if userprofile:
462 if userprofile:
456 path.append(os.path.join(userprofile, 'mercurial.ini'))
463 path.append(os.path.join(userprofile, 'mercurial.ini'))
457 path.append(os.path.join(userprofile, '.hgrc'))
464 path.append(os.path.join(userprofile, '.hgrc'))
458 return path
465 return path
@@ -1,1583 +1,1590 b''
1 # util.py - Mercurial utility functions and platform specfic implementations
1 # util.py - Mercurial utility functions and platform specfic implementations
2 #
2 #
3 # Copyright 2005 K. Thananchayan <thananck@yahoo.com>
3 # Copyright 2005 K. Thananchayan <thananck@yahoo.com>
4 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
4 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
5 # Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com>
5 # Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com>
6 #
6 #
7 # This software may be used and distributed according to the terms of the
7 # This software may be used and distributed according to the terms of the
8 # GNU General Public License version 2 or any later version.
8 # GNU General Public License version 2 or any later version.
9
9
10 """Mercurial utility functions and platform specfic implementations.
10 """Mercurial utility functions and platform specfic implementations.
11
11
12 This contains helper routines that are independent of the SCM core and
12 This contains helper routines that are independent of the SCM core and
13 hide platform-specific details from the core.
13 hide platform-specific details from the core.
14 """
14 """
15
15
16 from i18n import _
16 from i18n import _
17 import error, osutil, encoding
17 import error, osutil, encoding
18 import errno, re, shutil, sys, tempfile, traceback
18 import errno, re, shutil, sys, tempfile, traceback
19 import os, time, calendar, textwrap, unicodedata, signal
19 import os, time, calendar, textwrap, unicodedata, signal
20 import imp, socket, urllib
20 import imp, socket, urllib
21
21
22 # Python compatibility
22 # Python compatibility
23
23
24 def sha1(s):
24 def sha1(s):
25 return _fastsha1(s)
25 return _fastsha1(s)
26
26
27 def _fastsha1(s):
27 def _fastsha1(s):
28 # This function will import sha1 from hashlib or sha (whichever is
28 # This function will import sha1 from hashlib or sha (whichever is
29 # available) and overwrite itself with it on the first call.
29 # available) and overwrite itself with it on the first call.
30 # Subsequent calls will go directly to the imported function.
30 # Subsequent calls will go directly to the imported function.
31 if sys.version_info >= (2, 5):
31 if sys.version_info >= (2, 5):
32 from hashlib import sha1 as _sha1
32 from hashlib import sha1 as _sha1
33 else:
33 else:
34 from sha import sha as _sha1
34 from sha import sha as _sha1
35 global _fastsha1, sha1
35 global _fastsha1, sha1
36 _fastsha1 = sha1 = _sha1
36 _fastsha1 = sha1 = _sha1
37 return _sha1(s)
37 return _sha1(s)
38
38
39 import __builtin__
39 import __builtin__
40
40
41 if sys.version_info[0] < 3:
41 if sys.version_info[0] < 3:
42 def fakebuffer(sliceable, offset=0):
42 def fakebuffer(sliceable, offset=0):
43 return sliceable[offset:]
43 return sliceable[offset:]
44 else:
44 else:
45 def fakebuffer(sliceable, offset=0):
45 def fakebuffer(sliceable, offset=0):
46 return memoryview(sliceable)[offset:]
46 return memoryview(sliceable)[offset:]
47 try:
47 try:
48 buffer
48 buffer
49 except NameError:
49 except NameError:
50 __builtin__.buffer = fakebuffer
50 __builtin__.buffer = fakebuffer
51
51
52 import subprocess
52 import subprocess
53 closefds = os.name == 'posix'
53 closefds = os.name == 'posix'
54
54
55 def popen2(cmd, env=None, newlines=False):
55 def popen2(cmd, env=None, newlines=False):
56 # Setting bufsize to -1 lets the system decide the buffer size.
56 # Setting bufsize to -1 lets the system decide the buffer size.
57 # The default for bufsize is 0, meaning unbuffered. This leads to
57 # The default for bufsize is 0, meaning unbuffered. This leads to
58 # poor performance on Mac OS X: http://bugs.python.org/issue4194
58 # poor performance on Mac OS X: http://bugs.python.org/issue4194
59 p = subprocess.Popen(cmd, shell=True, bufsize=-1,
59 p = subprocess.Popen(cmd, shell=True, bufsize=-1,
60 close_fds=closefds,
60 close_fds=closefds,
61 stdin=subprocess.PIPE, stdout=subprocess.PIPE,
61 stdin=subprocess.PIPE, stdout=subprocess.PIPE,
62 universal_newlines=newlines,
62 universal_newlines=newlines,
63 env=env)
63 env=env)
64 return p.stdin, p.stdout
64 return p.stdin, p.stdout
65
65
66 def popen3(cmd, env=None, newlines=False):
66 def popen3(cmd, env=None, newlines=False):
67 p = subprocess.Popen(cmd, shell=True, bufsize=-1,
67 p = subprocess.Popen(cmd, shell=True, bufsize=-1,
68 close_fds=closefds,
68 close_fds=closefds,
69 stdin=subprocess.PIPE, stdout=subprocess.PIPE,
69 stdin=subprocess.PIPE, stdout=subprocess.PIPE,
70 stderr=subprocess.PIPE,
70 stderr=subprocess.PIPE,
71 universal_newlines=newlines,
71 universal_newlines=newlines,
72 env=env)
72 env=env)
73 return p.stdin, p.stdout, p.stderr
73 return p.stdin, p.stdout, p.stderr
74
74
75 def version():
75 def version():
76 """Return version information if available."""
76 """Return version information if available."""
77 try:
77 try:
78 import __version__
78 import __version__
79 return __version__.version
79 return __version__.version
80 except ImportError:
80 except ImportError:
81 return 'unknown'
81 return 'unknown'
82
82
83 # used by parsedate
83 # used by parsedate
84 defaultdateformats = (
84 defaultdateformats = (
85 '%Y-%m-%d %H:%M:%S',
85 '%Y-%m-%d %H:%M:%S',
86 '%Y-%m-%d %I:%M:%S%p',
86 '%Y-%m-%d %I:%M:%S%p',
87 '%Y-%m-%d %H:%M',
87 '%Y-%m-%d %H:%M',
88 '%Y-%m-%d %I:%M%p',
88 '%Y-%m-%d %I:%M%p',
89 '%Y-%m-%d',
89 '%Y-%m-%d',
90 '%m-%d',
90 '%m-%d',
91 '%m/%d',
91 '%m/%d',
92 '%m/%d/%y',
92 '%m/%d/%y',
93 '%m/%d/%Y',
93 '%m/%d/%Y',
94 '%a %b %d %H:%M:%S %Y',
94 '%a %b %d %H:%M:%S %Y',
95 '%a %b %d %I:%M:%S%p %Y',
95 '%a %b %d %I:%M:%S%p %Y',
96 '%a, %d %b %Y %H:%M:%S', # GNU coreutils "/bin/date --rfc-2822"
96 '%a, %d %b %Y %H:%M:%S', # GNU coreutils "/bin/date --rfc-2822"
97 '%b %d %H:%M:%S %Y',
97 '%b %d %H:%M:%S %Y',
98 '%b %d %I:%M:%S%p %Y',
98 '%b %d %I:%M:%S%p %Y',
99 '%b %d %H:%M:%S',
99 '%b %d %H:%M:%S',
100 '%b %d %I:%M:%S%p',
100 '%b %d %I:%M:%S%p',
101 '%b %d %H:%M',
101 '%b %d %H:%M',
102 '%b %d %I:%M%p',
102 '%b %d %I:%M%p',
103 '%b %d %Y',
103 '%b %d %Y',
104 '%b %d',
104 '%b %d',
105 '%H:%M:%S',
105 '%H:%M:%S',
106 '%I:%M:%S%p',
106 '%I:%M:%S%p',
107 '%H:%M',
107 '%H:%M',
108 '%I:%M%p',
108 '%I:%M%p',
109 )
109 )
110
110
111 extendeddateformats = defaultdateformats + (
111 extendeddateformats = defaultdateformats + (
112 "%Y",
112 "%Y",
113 "%Y-%m",
113 "%Y-%m",
114 "%b",
114 "%b",
115 "%b %Y",
115 "%b %Y",
116 )
116 )
117
117
118 def cachefunc(func):
118 def cachefunc(func):
119 '''cache the result of function calls'''
119 '''cache the result of function calls'''
120 # XXX doesn't handle keywords args
120 # XXX doesn't handle keywords args
121 cache = {}
121 cache = {}
122 if func.func_code.co_argcount == 1:
122 if func.func_code.co_argcount == 1:
123 # we gain a small amount of time because
123 # we gain a small amount of time because
124 # we don't need to pack/unpack the list
124 # we don't need to pack/unpack the list
125 def f(arg):
125 def f(arg):
126 if arg not in cache:
126 if arg not in cache:
127 cache[arg] = func(arg)
127 cache[arg] = func(arg)
128 return cache[arg]
128 return cache[arg]
129 else:
129 else:
130 def f(*args):
130 def f(*args):
131 if args not in cache:
131 if args not in cache:
132 cache[args] = func(*args)
132 cache[args] = func(*args)
133 return cache[args]
133 return cache[args]
134
134
135 return f
135 return f
136
136
137 def lrucachefunc(func):
137 def lrucachefunc(func):
138 '''cache most recent results of function calls'''
138 '''cache most recent results of function calls'''
139 cache = {}
139 cache = {}
140 order = []
140 order = []
141 if func.func_code.co_argcount == 1:
141 if func.func_code.co_argcount == 1:
142 def f(arg):
142 def f(arg):
143 if arg not in cache:
143 if arg not in cache:
144 if len(cache) > 20:
144 if len(cache) > 20:
145 del cache[order.pop(0)]
145 del cache[order.pop(0)]
146 cache[arg] = func(arg)
146 cache[arg] = func(arg)
147 else:
147 else:
148 order.remove(arg)
148 order.remove(arg)
149 order.append(arg)
149 order.append(arg)
150 return cache[arg]
150 return cache[arg]
151 else:
151 else:
152 def f(*args):
152 def f(*args):
153 if args not in cache:
153 if args not in cache:
154 if len(cache) > 20:
154 if len(cache) > 20:
155 del cache[order.pop(0)]
155 del cache[order.pop(0)]
156 cache[args] = func(*args)
156 cache[args] = func(*args)
157 else:
157 else:
158 order.remove(args)
158 order.remove(args)
159 order.append(args)
159 order.append(args)
160 return cache[args]
160 return cache[args]
161
161
162 return f
162 return f
163
163
164 class propertycache(object):
164 class propertycache(object):
165 def __init__(self, func):
165 def __init__(self, func):
166 self.func = func
166 self.func = func
167 self.name = func.__name__
167 self.name = func.__name__
168 def __get__(self, obj, type=None):
168 def __get__(self, obj, type=None):
169 result = self.func(obj)
169 result = self.func(obj)
170 setattr(obj, self.name, result)
170 setattr(obj, self.name, result)
171 return result
171 return result
172
172
173 def pipefilter(s, cmd):
173 def pipefilter(s, cmd):
174 '''filter string S through command CMD, returning its output'''
174 '''filter string S through command CMD, returning its output'''
175 p = subprocess.Popen(cmd, shell=True, close_fds=closefds,
175 p = subprocess.Popen(cmd, shell=True, close_fds=closefds,
176 stdin=subprocess.PIPE, stdout=subprocess.PIPE)
176 stdin=subprocess.PIPE, stdout=subprocess.PIPE)
177 pout, perr = p.communicate(s)
177 pout, perr = p.communicate(s)
178 return pout
178 return pout
179
179
180 def tempfilter(s, cmd):
180 def tempfilter(s, cmd):
181 '''filter string S through a pair of temporary files with CMD.
181 '''filter string S through a pair of temporary files with CMD.
182 CMD is used as a template to create the real command to be run,
182 CMD is used as a template to create the real command to be run,
183 with the strings INFILE and OUTFILE replaced by the real names of
183 with the strings INFILE and OUTFILE replaced by the real names of
184 the temporary files generated.'''
184 the temporary files generated.'''
185 inname, outname = None, None
185 inname, outname = None, None
186 try:
186 try:
187 infd, inname = tempfile.mkstemp(prefix='hg-filter-in-')
187 infd, inname = tempfile.mkstemp(prefix='hg-filter-in-')
188 fp = os.fdopen(infd, 'wb')
188 fp = os.fdopen(infd, 'wb')
189 fp.write(s)
189 fp.write(s)
190 fp.close()
190 fp.close()
191 outfd, outname = tempfile.mkstemp(prefix='hg-filter-out-')
191 outfd, outname = tempfile.mkstemp(prefix='hg-filter-out-')
192 os.close(outfd)
192 os.close(outfd)
193 cmd = cmd.replace('INFILE', inname)
193 cmd = cmd.replace('INFILE', inname)
194 cmd = cmd.replace('OUTFILE', outname)
194 cmd = cmd.replace('OUTFILE', outname)
195 code = os.system(cmd)
195 code = os.system(cmd)
196 if sys.platform == 'OpenVMS' and code & 1:
196 if sys.platform == 'OpenVMS' and code & 1:
197 code = 0
197 code = 0
198 if code:
198 if code:
199 raise Abort(_("command '%s' failed: %s") %
199 raise Abort(_("command '%s' failed: %s") %
200 (cmd, explain_exit(code)))
200 (cmd, explain_exit(code)))
201 fp = open(outname, 'rb')
201 fp = open(outname, 'rb')
202 r = fp.read()
202 r = fp.read()
203 fp.close()
203 fp.close()
204 return r
204 return r
205 finally:
205 finally:
206 try:
206 try:
207 if inname:
207 if inname:
208 os.unlink(inname)
208 os.unlink(inname)
209 except OSError:
209 except OSError:
210 pass
210 pass
211 try:
211 try:
212 if outname:
212 if outname:
213 os.unlink(outname)
213 os.unlink(outname)
214 except OSError:
214 except OSError:
215 pass
215 pass
216
216
217 filtertable = {
217 filtertable = {
218 'tempfile:': tempfilter,
218 'tempfile:': tempfilter,
219 'pipe:': pipefilter,
219 'pipe:': pipefilter,
220 }
220 }
221
221
222 def filter(s, cmd):
222 def filter(s, cmd):
223 "filter a string through a command that transforms its input to its output"
223 "filter a string through a command that transforms its input to its output"
224 for name, fn in filtertable.iteritems():
224 for name, fn in filtertable.iteritems():
225 if cmd.startswith(name):
225 if cmd.startswith(name):
226 return fn(s, cmd[len(name):].lstrip())
226 return fn(s, cmd[len(name):].lstrip())
227 return pipefilter(s, cmd)
227 return pipefilter(s, cmd)
228
228
229 def binary(s):
229 def binary(s):
230 """return true if a string is binary data"""
230 """return true if a string is binary data"""
231 return bool(s and '\0' in s)
231 return bool(s and '\0' in s)
232
232
233 def increasingchunks(source, min=1024, max=65536):
233 def increasingchunks(source, min=1024, max=65536):
234 '''return no less than min bytes per chunk while data remains,
234 '''return no less than min bytes per chunk while data remains,
235 doubling min after each chunk until it reaches max'''
235 doubling min after each chunk until it reaches max'''
236 def log2(x):
236 def log2(x):
237 if not x:
237 if not x:
238 return 0
238 return 0
239 i = 0
239 i = 0
240 while x:
240 while x:
241 x >>= 1
241 x >>= 1
242 i += 1
242 i += 1
243 return i - 1
243 return i - 1
244
244
245 buf = []
245 buf = []
246 blen = 0
246 blen = 0
247 for chunk in source:
247 for chunk in source:
248 buf.append(chunk)
248 buf.append(chunk)
249 blen += len(chunk)
249 blen += len(chunk)
250 if blen >= min:
250 if blen >= min:
251 if min < max:
251 if min < max:
252 min = min << 1
252 min = min << 1
253 nmin = 1 << log2(blen)
253 nmin = 1 << log2(blen)
254 if nmin > min:
254 if nmin > min:
255 min = nmin
255 min = nmin
256 if min > max:
256 if min > max:
257 min = max
257 min = max
258 yield ''.join(buf)
258 yield ''.join(buf)
259 blen = 0
259 blen = 0
260 buf = []
260 buf = []
261 if buf:
261 if buf:
262 yield ''.join(buf)
262 yield ''.join(buf)
263
263
264 Abort = error.Abort
264 Abort = error.Abort
265
265
266 def always(fn):
266 def always(fn):
267 return True
267 return True
268
268
269 def never(fn):
269 def never(fn):
270 return False
270 return False
271
271
272 def pathto(root, n1, n2):
272 def pathto(root, n1, n2):
273 '''return the relative path from one place to another.
273 '''return the relative path from one place to another.
274 root should use os.sep to separate directories
274 root should use os.sep to separate directories
275 n1 should use os.sep to separate directories
275 n1 should use os.sep to separate directories
276 n2 should use "/" to separate directories
276 n2 should use "/" to separate directories
277 returns an os.sep-separated path.
277 returns an os.sep-separated path.
278
278
279 If n1 is a relative path, it's assumed it's
279 If n1 is a relative path, it's assumed it's
280 relative to root.
280 relative to root.
281 n2 should always be relative to root.
281 n2 should always be relative to root.
282 '''
282 '''
283 if not n1:
283 if not n1:
284 return localpath(n2)
284 return localpath(n2)
285 if os.path.isabs(n1):
285 if os.path.isabs(n1):
286 if os.path.splitdrive(root)[0] != os.path.splitdrive(n1)[0]:
286 if os.path.splitdrive(root)[0] != os.path.splitdrive(n1)[0]:
287 return os.path.join(root, localpath(n2))
287 return os.path.join(root, localpath(n2))
288 n2 = '/'.join((pconvert(root), n2))
288 n2 = '/'.join((pconvert(root), n2))
289 a, b = splitpath(n1), n2.split('/')
289 a, b = splitpath(n1), n2.split('/')
290 a.reverse()
290 a.reverse()
291 b.reverse()
291 b.reverse()
292 while a and b and a[-1] == b[-1]:
292 while a and b and a[-1] == b[-1]:
293 a.pop()
293 a.pop()
294 b.pop()
294 b.pop()
295 b.reverse()
295 b.reverse()
296 return os.sep.join((['..'] * len(a)) + b) or '.'
296 return os.sep.join((['..'] * len(a)) + b) or '.'
297
297
298 _hgexecutable = None
298 _hgexecutable = None
299
299
300 def main_is_frozen():
300 def main_is_frozen():
301 """return True if we are a frozen executable.
301 """return True if we are a frozen executable.
302
302
303 The code supports py2exe (most common, Windows only) and tools/freeze
303 The code supports py2exe (most common, Windows only) and tools/freeze
304 (portable, not much used).
304 (portable, not much used).
305 """
305 """
306 return (hasattr(sys, "frozen") or # new py2exe
306 return (hasattr(sys, "frozen") or # new py2exe
307 hasattr(sys, "importers") or # old py2exe
307 hasattr(sys, "importers") or # old py2exe
308 imp.is_frozen("__main__")) # tools/freeze
308 imp.is_frozen("__main__")) # tools/freeze
309
309
310 def hgexecutable():
310 def hgexecutable():
311 """return location of the 'hg' executable.
311 """return location of the 'hg' executable.
312
312
313 Defaults to $HG or 'hg' in the search path.
313 Defaults to $HG or 'hg' in the search path.
314 """
314 """
315 if _hgexecutable is None:
315 if _hgexecutable is None:
316 hg = os.environ.get('HG')
316 hg = os.environ.get('HG')
317 if hg:
317 if hg:
318 set_hgexecutable(hg)
318 set_hgexecutable(hg)
319 elif main_is_frozen():
319 elif main_is_frozen():
320 set_hgexecutable(sys.executable)
320 set_hgexecutable(sys.executable)
321 else:
321 else:
322 exe = find_exe('hg') or os.path.basename(sys.argv[0])
322 exe = find_exe('hg') or os.path.basename(sys.argv[0])
323 set_hgexecutable(exe)
323 set_hgexecutable(exe)
324 return _hgexecutable
324 return _hgexecutable
325
325
326 def set_hgexecutable(path):
326 def set_hgexecutable(path):
327 """set location of the 'hg' executable"""
327 """set location of the 'hg' executable"""
328 global _hgexecutable
328 global _hgexecutable
329 _hgexecutable = path
329 _hgexecutable = path
330
330
331 def system(cmd, environ={}, cwd=None, onerr=None, errprefix=None, out=None):
331 def system(cmd, environ={}, cwd=None, onerr=None, errprefix=None, out=None):
332 '''enhanced shell command execution.
332 '''enhanced shell command execution.
333 run with environment maybe modified, maybe in different dir.
333 run with environment maybe modified, maybe in different dir.
334
334
335 if command fails and onerr is None, return status. if ui object,
335 if command fails and onerr is None, return status. if ui object,
336 print error message and return status, else raise onerr object as
336 print error message and return status, else raise onerr object as
337 exception.
337 exception.
338
338
339 if out is specified, it is assumed to be a file-like object that has a
339 if out is specified, it is assumed to be a file-like object that has a
340 write() method. stdout and stderr will be redirected to out.'''
340 write() method. stdout and stderr will be redirected to out.'''
341 try:
341 try:
342 sys.stdout.flush()
342 sys.stdout.flush()
343 except Exception:
343 except Exception:
344 pass
344 pass
345 def py2shell(val):
345 def py2shell(val):
346 'convert python object into string that is useful to shell'
346 'convert python object into string that is useful to shell'
347 if val is None or val is False:
347 if val is None or val is False:
348 return '0'
348 return '0'
349 if val is True:
349 if val is True:
350 return '1'
350 return '1'
351 return str(val)
351 return str(val)
352 origcmd = cmd
352 origcmd = cmd
353 cmd = quotecommand(cmd)
353 cmd = quotecommand(cmd)
354 env = dict(os.environ)
354 env = dict(os.environ)
355 env.update((k, py2shell(v)) for k, v in environ.iteritems())
355 env.update((k, py2shell(v)) for k, v in environ.iteritems())
356 env['HG'] = hgexecutable()
356 env['HG'] = hgexecutable()
357 if out is None:
357 if out is None:
358 rc = subprocess.call(cmd, shell=True, close_fds=closefds,
358 rc = subprocess.call(cmd, shell=True, close_fds=closefds,
359 env=env, cwd=cwd)
359 env=env, cwd=cwd)
360 else:
360 else:
361 proc = subprocess.Popen(cmd, shell=True, close_fds=closefds,
361 proc = subprocess.Popen(cmd, shell=True, close_fds=closefds,
362 env=env, cwd=cwd, stdout=subprocess.PIPE,
362 env=env, cwd=cwd, stdout=subprocess.PIPE,
363 stderr=subprocess.STDOUT)
363 stderr=subprocess.STDOUT)
364 for line in proc.stdout:
364 for line in proc.stdout:
365 out.write(line)
365 out.write(line)
366 proc.wait()
366 proc.wait()
367 rc = proc.returncode
367 rc = proc.returncode
368 if sys.platform == 'OpenVMS' and rc & 1:
368 if sys.platform == 'OpenVMS' and rc & 1:
369 rc = 0
369 rc = 0
370 if rc and onerr:
370 if rc and onerr:
371 errmsg = '%s %s' % (os.path.basename(origcmd.split(None, 1)[0]),
371 errmsg = '%s %s' % (os.path.basename(origcmd.split(None, 1)[0]),
372 explain_exit(rc)[0])
372 explain_exit(rc)[0])
373 if errprefix:
373 if errprefix:
374 errmsg = '%s: %s' % (errprefix, errmsg)
374 errmsg = '%s: %s' % (errprefix, errmsg)
375 try:
375 try:
376 onerr.warn(errmsg + '\n')
376 onerr.warn(errmsg + '\n')
377 except AttributeError:
377 except AttributeError:
378 raise onerr(errmsg)
378 raise onerr(errmsg)
379 return rc
379 return rc
380
380
381 def checksignature(func):
381 def checksignature(func):
382 '''wrap a function with code to check for calling errors'''
382 '''wrap a function with code to check for calling errors'''
383 def check(*args, **kwargs):
383 def check(*args, **kwargs):
384 try:
384 try:
385 return func(*args, **kwargs)
385 return func(*args, **kwargs)
386 except TypeError:
386 except TypeError:
387 if len(traceback.extract_tb(sys.exc_info()[2])) == 1:
387 if len(traceback.extract_tb(sys.exc_info()[2])) == 1:
388 raise error.SignatureError
388 raise error.SignatureError
389 raise
389 raise
390
390
391 return check
391 return check
392
392
393 def makedir(path, notindexed):
393 def makedir(path, notindexed):
394 os.mkdir(path)
394 os.mkdir(path)
395
395
396 def unlinkpath(f):
396 def unlinkpath(f):
397 """unlink and remove the directory if it is empty"""
397 """unlink and remove the directory if it is empty"""
398 os.unlink(f)
398 os.unlink(f)
399 # try removing directories that might now be empty
399 # try removing directories that might now be empty
400 try:
400 try:
401 os.removedirs(os.path.dirname(f))
401 os.removedirs(os.path.dirname(f))
402 except OSError:
402 except OSError:
403 pass
403 pass
404
404
405 def copyfile(src, dest):
405 def copyfile(src, dest):
406 "copy a file, preserving mode and atime/mtime"
406 "copy a file, preserving mode and atime/mtime"
407 if os.path.islink(src):
407 if os.path.islink(src):
408 try:
408 try:
409 os.unlink(dest)
409 os.unlink(dest)
410 except OSError:
410 except OSError:
411 pass
411 pass
412 os.symlink(os.readlink(src), dest)
412 os.symlink(os.readlink(src), dest)
413 else:
413 else:
414 try:
414 try:
415 shutil.copyfile(src, dest)
415 shutil.copyfile(src, dest)
416 shutil.copymode(src, dest)
416 shutil.copymode(src, dest)
417 except shutil.Error, inst:
417 except shutil.Error, inst:
418 raise Abort(str(inst))
418 raise Abort(str(inst))
419
419
420 def copyfiles(src, dst, hardlink=None):
420 def copyfiles(src, dst, hardlink=None):
421 """Copy a directory tree using hardlinks if possible"""
421 """Copy a directory tree using hardlinks if possible"""
422
422
423 if hardlink is None:
423 if hardlink is None:
424 hardlink = (os.stat(src).st_dev ==
424 hardlink = (os.stat(src).st_dev ==
425 os.stat(os.path.dirname(dst)).st_dev)
425 os.stat(os.path.dirname(dst)).st_dev)
426
426
427 num = 0
427 num = 0
428 if os.path.isdir(src):
428 if os.path.isdir(src):
429 os.mkdir(dst)
429 os.mkdir(dst)
430 for name, kind in osutil.listdir(src):
430 for name, kind in osutil.listdir(src):
431 srcname = os.path.join(src, name)
431 srcname = os.path.join(src, name)
432 dstname = os.path.join(dst, name)
432 dstname = os.path.join(dst, name)
433 hardlink, n = copyfiles(srcname, dstname, hardlink)
433 hardlink, n = copyfiles(srcname, dstname, hardlink)
434 num += n
434 num += n
435 else:
435 else:
436 if hardlink:
436 if hardlink:
437 try:
437 try:
438 os_link(src, dst)
438 os_link(src, dst)
439 except (IOError, OSError):
439 except (IOError, OSError):
440 hardlink = False
440 hardlink = False
441 shutil.copy(src, dst)
441 shutil.copy(src, dst)
442 else:
442 else:
443 shutil.copy(src, dst)
443 shutil.copy(src, dst)
444 num += 1
444 num += 1
445
445
446 return hardlink, num
446 return hardlink, num
447
447
448 _windows_reserved_filenames = '''con prn aux nul
448 _windows_reserved_filenames = '''con prn aux nul
449 com1 com2 com3 com4 com5 com6 com7 com8 com9
449 com1 com2 com3 com4 com5 com6 com7 com8 com9
450 lpt1 lpt2 lpt3 lpt4 lpt5 lpt6 lpt7 lpt8 lpt9'''.split()
450 lpt1 lpt2 lpt3 lpt4 lpt5 lpt6 lpt7 lpt8 lpt9'''.split()
451 _windows_reserved_chars = ':*?"<>|'
451 _windows_reserved_chars = ':*?"<>|'
452 def checkwinfilename(path):
452 def checkwinfilename(path):
453 '''Check that the base-relative path is a valid filename on Windows.
453 '''Check that the base-relative path is a valid filename on Windows.
454 Returns None if the path is ok, or a UI string describing the problem.
454 Returns None if the path is ok, or a UI string describing the problem.
455
455
456 >>> checkwinfilename("just/a/normal/path")
456 >>> checkwinfilename("just/a/normal/path")
457 >>> checkwinfilename("foo/bar/con.xml")
457 >>> checkwinfilename("foo/bar/con.xml")
458 "filename contains 'con', which is reserved on Windows"
458 "filename contains 'con', which is reserved on Windows"
459 >>> checkwinfilename("foo/con.xml/bar")
459 >>> checkwinfilename("foo/con.xml/bar")
460 "filename contains 'con', which is reserved on Windows"
460 "filename contains 'con', which is reserved on Windows"
461 >>> checkwinfilename("foo/bar/xml.con")
461 >>> checkwinfilename("foo/bar/xml.con")
462 >>> checkwinfilename("foo/bar/AUX/bla.txt")
462 >>> checkwinfilename("foo/bar/AUX/bla.txt")
463 "filename contains 'AUX', which is reserved on Windows"
463 "filename contains 'AUX', which is reserved on Windows"
464 >>> checkwinfilename("foo/bar/bla:.txt")
464 >>> checkwinfilename("foo/bar/bla:.txt")
465 "filename contains ':', which is reserved on Windows"
465 "filename contains ':', which is reserved on Windows"
466 >>> checkwinfilename("foo/bar/b\07la.txt")
466 >>> checkwinfilename("foo/bar/b\07la.txt")
467 "filename contains '\\\\x07', which is invalid on Windows"
467 "filename contains '\\\\x07', which is invalid on Windows"
468 >>> checkwinfilename("foo/bar/bla ")
468 >>> checkwinfilename("foo/bar/bla ")
469 "filename ends with ' ', which is not allowed on Windows"
469 "filename ends with ' ', which is not allowed on Windows"
470 '''
470 '''
471 for n in path.replace('\\', '/').split('/'):
471 for n in path.replace('\\', '/').split('/'):
472 if not n:
472 if not n:
473 continue
473 continue
474 for c in n:
474 for c in n:
475 if c in _windows_reserved_chars:
475 if c in _windows_reserved_chars:
476 return _("filename contains '%s', which is reserved "
476 return _("filename contains '%s', which is reserved "
477 "on Windows") % c
477 "on Windows") % c
478 if ord(c) <= 31:
478 if ord(c) <= 31:
479 return _("filename contains %r, which is invalid "
479 return _("filename contains %r, which is invalid "
480 "on Windows") % c
480 "on Windows") % c
481 base = n.split('.')[0]
481 base = n.split('.')[0]
482 if base and base.lower() in _windows_reserved_filenames:
482 if base and base.lower() in _windows_reserved_filenames:
483 return _("filename contains '%s', which is reserved "
483 return _("filename contains '%s', which is reserved "
484 "on Windows") % base
484 "on Windows") % base
485 t = n[-1]
485 t = n[-1]
486 if t in '. ':
486 if t in '. ':
487 return _("filename ends with '%s', which is not allowed "
487 return _("filename ends with '%s', which is not allowed "
488 "on Windows") % t
488 "on Windows") % t
489
489
490 def lookup_reg(key, name=None, scope=None):
490 def lookup_reg(key, name=None, scope=None):
491 return None
491 return None
492
492
493 def hidewindow():
493 def hidewindow():
494 """Hide current shell window.
494 """Hide current shell window.
495
495
496 Used to hide the window opened when starting asynchronous
496 Used to hide the window opened when starting asynchronous
497 child process under Windows, unneeded on other systems.
497 child process under Windows, unneeded on other systems.
498 """
498 """
499 pass
499 pass
500
500
501 if os.name == 'nt':
501 if os.name == 'nt':
502 checkosfilename = checkwinfilename
502 checkosfilename = checkwinfilename
503 from windows import *
503 from windows import *
504 else:
504 else:
505 from posix import *
505 from posix import *
506
506
507 def makelock(info, pathname):
507 def makelock(info, pathname):
508 try:
508 try:
509 return os.symlink(info, pathname)
509 return os.symlink(info, pathname)
510 except OSError, why:
510 except OSError, why:
511 if why.errno == errno.EEXIST:
511 if why.errno == errno.EEXIST:
512 raise
512 raise
513 except AttributeError: # no symlink in os
513 except AttributeError: # no symlink in os
514 pass
514 pass
515
515
516 ld = os.open(pathname, os.O_CREAT | os.O_WRONLY | os.O_EXCL)
516 ld = os.open(pathname, os.O_CREAT | os.O_WRONLY | os.O_EXCL)
517 os.write(ld, info)
517 os.write(ld, info)
518 os.close(ld)
518 os.close(ld)
519
519
520 def readlock(pathname):
520 def readlock(pathname):
521 try:
521 try:
522 return os.readlink(pathname)
522 return os.readlink(pathname)
523 except OSError, why:
523 except OSError, why:
524 if why.errno not in (errno.EINVAL, errno.ENOSYS):
524 if why.errno not in (errno.EINVAL, errno.ENOSYS):
525 raise
525 raise
526 except AttributeError: # no symlink in os
526 except AttributeError: # no symlink in os
527 pass
527 pass
528 fp = posixfile(pathname)
528 fp = posixfile(pathname)
529 r = fp.read()
529 r = fp.read()
530 fp.close()
530 fp.close()
531 return r
531 return r
532
532
533 def fstat(fp):
533 def fstat(fp):
534 '''stat file object that may not have fileno method.'''
534 '''stat file object that may not have fileno method.'''
535 try:
535 try:
536 return os.fstat(fp.fileno())
536 return os.fstat(fp.fileno())
537 except AttributeError:
537 except AttributeError:
538 return os.stat(fp.name)
538 return os.stat(fp.name)
539
539
540 # File system features
540 # File system features
541
541
542 def checkcase(path):
542 def checkcase(path):
543 """
543 """
544 Check whether the given path is on a case-sensitive filesystem
544 Check whether the given path is on a case-sensitive filesystem
545
545
546 Requires a path (like /foo/.hg) ending with a foldable final
546 Requires a path (like /foo/.hg) ending with a foldable final
547 directory component.
547 directory component.
548 """
548 """
549 s1 = os.stat(path)
549 s1 = os.stat(path)
550 d, b = os.path.split(path)
550 d, b = os.path.split(path)
551 p2 = os.path.join(d, b.upper())
551 p2 = os.path.join(d, b.upper())
552 if path == p2:
552 if path == p2:
553 p2 = os.path.join(d, b.lower())
553 p2 = os.path.join(d, b.lower())
554 try:
554 try:
555 s2 = os.stat(p2)
555 s2 = os.stat(p2)
556 if s2 == s1:
556 if s2 == s1:
557 return False
557 return False
558 return True
558 return True
559 except OSError:
559 except OSError:
560 return True
560 return True
561
561
562 _fspathcache = {}
562 _fspathcache = {}
563 def fspath(name, root):
563 def fspath(name, root):
564 '''Get name in the case stored in the filesystem
564 '''Get name in the case stored in the filesystem
565
565
566 The name is either relative to root, or it is an absolute path starting
566 The name is either relative to root, or it is an absolute path starting
567 with root. Note that this function is unnecessary, and should not be
567 with root. Note that this function is unnecessary, and should not be
568 called, for case-sensitive filesystems (simply because it's expensive).
568 called, for case-sensitive filesystems (simply because it's expensive).
569 '''
569 '''
570 # If name is absolute, make it relative
570 # If name is absolute, make it relative
571 if name.lower().startswith(root.lower()):
571 if name.lower().startswith(root.lower()):
572 l = len(root)
572 l = len(root)
573 if name[l] == os.sep or name[l] == os.altsep:
573 if name[l] == os.sep or name[l] == os.altsep:
574 l = l + 1
574 l = l + 1
575 name = name[l:]
575 name = name[l:]
576
576
577 if not os.path.lexists(os.path.join(root, name)):
577 if not os.path.lexists(os.path.join(root, name)):
578 return None
578 return None
579
579
580 seps = os.sep
580 seps = os.sep
581 if os.altsep:
581 if os.altsep:
582 seps = seps + os.altsep
582 seps = seps + os.altsep
583 # Protect backslashes. This gets silly very quickly.
583 # Protect backslashes. This gets silly very quickly.
584 seps.replace('\\','\\\\')
584 seps.replace('\\','\\\\')
585 pattern = re.compile(r'([^%s]+)|([%s]+)' % (seps, seps))
585 pattern = re.compile(r'([^%s]+)|([%s]+)' % (seps, seps))
586 dir = os.path.normcase(os.path.normpath(root))
586 dir = os.path.normcase(os.path.normpath(root))
587 result = []
587 result = []
588 for part, sep in pattern.findall(name):
588 for part, sep in pattern.findall(name):
589 if sep:
589 if sep:
590 result.append(sep)
590 result.append(sep)
591 continue
591 continue
592
592
593 if dir not in _fspathcache:
593 if dir not in _fspathcache:
594 _fspathcache[dir] = os.listdir(dir)
594 _fspathcache[dir] = os.listdir(dir)
595 contents = _fspathcache[dir]
595 contents = _fspathcache[dir]
596
596
597 lpart = part.lower()
597 lpart = part.lower()
598 lenp = len(part)
598 lenp = len(part)
599 for n in contents:
599 for n in contents:
600 if lenp == len(n) and n.lower() == lpart:
600 if lenp == len(n) and n.lower() == lpart:
601 result.append(n)
601 result.append(n)
602 break
602 break
603 else:
603 else:
604 # Cannot happen, as the file exists!
604 # Cannot happen, as the file exists!
605 result.append(part)
605 result.append(part)
606 dir = os.path.join(dir, lpart)
606 dir = os.path.join(dir, lpart)
607
607
608 return ''.join(result)
608 return ''.join(result)
609
609
610 def checknlink(testfile):
610 def checknlink(testfile):
611 '''check whether hardlink count reporting works properly'''
611 '''check whether hardlink count reporting works properly'''
612
612
613 # testfile may be open, so we need a separate file for checking to
613 # testfile may be open, so we need a separate file for checking to
614 # work around issue2543 (or testfile may get lost on Samba shares)
614 # work around issue2543 (or testfile may get lost on Samba shares)
615 f1 = testfile + ".hgtmp1"
615 f1 = testfile + ".hgtmp1"
616 if os.path.lexists(f1):
616 if os.path.lexists(f1):
617 return False
617 return False
618 try:
618 try:
619 posixfile(f1, 'w').close()
619 posixfile(f1, 'w').close()
620 except IOError:
620 except IOError:
621 return False
621 return False
622
622
623 f2 = testfile + ".hgtmp2"
623 f2 = testfile + ".hgtmp2"
624 fd = None
624 fd = None
625 try:
625 try:
626 try:
626 try:
627 os_link(f1, f2)
627 os_link(f1, f2)
628 except OSError:
628 except OSError:
629 return False
629 return False
630
630
631 # nlinks() may behave differently for files on Windows shares if
631 # nlinks() may behave differently for files on Windows shares if
632 # the file is open.
632 # the file is open.
633 fd = posixfile(f2)
633 fd = posixfile(f2)
634 return nlinks(f2) > 1
634 return nlinks(f2) > 1
635 finally:
635 finally:
636 if fd is not None:
636 if fd is not None:
637 fd.close()
637 fd.close()
638 for f in (f1, f2):
638 for f in (f1, f2):
639 try:
639 try:
640 os.unlink(f)
640 os.unlink(f)
641 except OSError:
641 except OSError:
642 pass
642 pass
643
643
644 return False
644 return False
645
645
646 def endswithsep(path):
646 def endswithsep(path):
647 '''Check path ends with os.sep or os.altsep.'''
647 '''Check path ends with os.sep or os.altsep.'''
648 return path.endswith(os.sep) or os.altsep and path.endswith(os.altsep)
648 return path.endswith(os.sep) or os.altsep and path.endswith(os.altsep)
649
649
650 def splitpath(path):
650 def splitpath(path):
651 '''Split path by os.sep.
651 '''Split path by os.sep.
652 Note that this function does not use os.altsep because this is
652 Note that this function does not use os.altsep because this is
653 an alternative of simple "xxx.split(os.sep)".
653 an alternative of simple "xxx.split(os.sep)".
654 It is recommended to use os.path.normpath() before using this
654 It is recommended to use os.path.normpath() before using this
655 function if need.'''
655 function if need.'''
656 return path.split(os.sep)
656 return path.split(os.sep)
657
657
658 def gui():
658 def gui():
659 '''Are we running in a GUI?'''
659 '''Are we running in a GUI?'''
660 if sys.platform == 'darwin':
660 if sys.platform == 'darwin':
661 if 'SSH_CONNECTION' in os.environ:
661 if 'SSH_CONNECTION' in os.environ:
662 # handle SSH access to a box where the user is logged in
662 # handle SSH access to a box where the user is logged in
663 return False
663 return False
664 elif getattr(osutil, 'isgui', None):
664 elif getattr(osutil, 'isgui', None):
665 # check if a CoreGraphics session is available
665 # check if a CoreGraphics session is available
666 return osutil.isgui()
666 return osutil.isgui()
667 else:
667 else:
668 # pure build; use a safe default
668 # pure build; use a safe default
669 return True
669 return True
670 else:
670 else:
671 return os.name == "nt" or os.environ.get("DISPLAY")
671 return os.name == "nt" or os.environ.get("DISPLAY")
672
672
673 def mktempcopy(name, emptyok=False, createmode=None):
673 def mktempcopy(name, emptyok=False, createmode=None):
674 """Create a temporary file with the same contents from name
674 """Create a temporary file with the same contents from name
675
675
676 The permission bits are copied from the original file.
676 The permission bits are copied from the original file.
677
677
678 If the temporary file is going to be truncated immediately, you
678 If the temporary file is going to be truncated immediately, you
679 can use emptyok=True as an optimization.
679 can use emptyok=True as an optimization.
680
680
681 Returns the name of the temporary file.
681 Returns the name of the temporary file.
682 """
682 """
683 d, fn = os.path.split(name)
683 d, fn = os.path.split(name)
684 fd, temp = tempfile.mkstemp(prefix='.%s-' % fn, dir=d)
684 fd, temp = tempfile.mkstemp(prefix='.%s-' % fn, dir=d)
685 os.close(fd)
685 os.close(fd)
686 # Temporary files are created with mode 0600, which is usually not
686 # Temporary files are created with mode 0600, which is usually not
687 # what we want. If the original file already exists, just copy
687 # what we want. If the original file already exists, just copy
688 # its mode. Otherwise, manually obey umask.
688 # its mode. Otherwise, manually obey umask.
689 try:
689 try:
690 st_mode = os.lstat(name).st_mode & 0777
690 st_mode = os.lstat(name).st_mode & 0777
691 except OSError, inst:
691 except OSError, inst:
692 if inst.errno != errno.ENOENT:
692 if inst.errno != errno.ENOENT:
693 raise
693 raise
694 st_mode = createmode
694 st_mode = createmode
695 if st_mode is None:
695 if st_mode is None:
696 st_mode = ~umask
696 st_mode = ~umask
697 st_mode &= 0666
697 st_mode &= 0666
698 os.chmod(temp, st_mode)
698 os.chmod(temp, st_mode)
699 if emptyok:
699 if emptyok:
700 return temp
700 return temp
701 try:
701 try:
702 try:
702 try:
703 ifp = posixfile(name, "rb")
703 ifp = posixfile(name, "rb")
704 except IOError, inst:
704 except IOError, inst:
705 if inst.errno == errno.ENOENT:
705 if inst.errno == errno.ENOENT:
706 return temp
706 return temp
707 if not getattr(inst, 'filename', None):
707 if not getattr(inst, 'filename', None):
708 inst.filename = name
708 inst.filename = name
709 raise
709 raise
710 ofp = posixfile(temp, "wb")
710 ofp = posixfile(temp, "wb")
711 for chunk in filechunkiter(ifp):
711 for chunk in filechunkiter(ifp):
712 ofp.write(chunk)
712 ofp.write(chunk)
713 ifp.close()
713 ifp.close()
714 ofp.close()
714 ofp.close()
715 except:
715 except:
716 try: os.unlink(temp)
716 try: os.unlink(temp)
717 except: pass
717 except: pass
718 raise
718 raise
719 return temp
719 return temp
720
720
721 class atomictempfile(object):
721 class atomictempfile(object):
722 '''writeable file object that atomically updates a file
722 '''writeable file object that atomically updates a file
723
723
724 All writes will go to a temporary copy of the original file. Call
724 All writes will go to a temporary copy of the original file. Call
725 rename() when you are done writing, and atomictempfile will rename
725 rename() when you are done writing, and atomictempfile will rename
726 the temporary copy to the original name, making the changes visible.
726 the temporary copy to the original name, making the changes visible.
727
727
728 Unlike other file-like objects, close() discards your writes by
728 Unlike other file-like objects, close() discards your writes by
729 simply deleting the temporary file.
729 simply deleting the temporary file.
730 '''
730 '''
731 def __init__(self, name, mode='w+b', createmode=None):
731 def __init__(self, name, mode='w+b', createmode=None):
732 self.__name = name # permanent name
732 self.__name = name # permanent name
733 self._tempname = mktempcopy(name, emptyok=('w' in mode),
733 self._tempname = mktempcopy(name, emptyok=('w' in mode),
734 createmode=createmode)
734 createmode=createmode)
735 self._fp = posixfile(self._tempname, mode)
735 self._fp = posixfile(self._tempname, mode)
736
736
737 # delegated methods
737 # delegated methods
738 self.write = self._fp.write
738 self.write = self._fp.write
739 self.fileno = self._fp.fileno
739 self.fileno = self._fp.fileno
740
740
741 def rename(self):
741 def rename(self):
742 if not self._fp.closed:
742 if not self._fp.closed:
743 self._fp.close()
743 self._fp.close()
744 rename(self._tempname, localpath(self.__name))
744 rename(self._tempname, localpath(self.__name))
745
745
746 def close(self):
746 def close(self):
747 if not self._fp.closed:
747 if not self._fp.closed:
748 try:
748 try:
749 os.unlink(self._tempname)
749 os.unlink(self._tempname)
750 except OSError:
750 except OSError:
751 pass
751 pass
752 self._fp.close()
752 self._fp.close()
753
753
754 def __del__(self):
754 def __del__(self):
755 if hasattr(self, '_fp'): # constructor actually did something
755 if hasattr(self, '_fp'): # constructor actually did something
756 self.close()
756 self.close()
757
757
758 def makedirs(name, mode=None):
758 def makedirs(name, mode=None):
759 """recursive directory creation with parent mode inheritance"""
759 """recursive directory creation with parent mode inheritance"""
760 parent = os.path.abspath(os.path.dirname(name))
760 parent = os.path.abspath(os.path.dirname(name))
761 try:
761 try:
762 os.mkdir(name)
762 os.mkdir(name)
763 if mode is not None:
763 if mode is not None:
764 os.chmod(name, mode)
764 os.chmod(name, mode)
765 return
765 return
766 except OSError, err:
766 except OSError, err:
767 if err.errno == errno.EEXIST:
767 if err.errno == errno.EEXIST:
768 return
768 return
769 if not name or parent == name or err.errno != errno.ENOENT:
769 if not name or parent == name or err.errno != errno.ENOENT:
770 raise
770 raise
771 makedirs(parent, mode)
771 makedirs(parent, mode)
772 makedirs(name, mode)
772 makedirs(name, mode)
773
773
774 def readfile(path):
774 def readfile(path):
775 fp = open(path)
775 fp = open(path)
776 try:
776 try:
777 return fp.read()
777 return fp.read()
778 finally:
778 finally:
779 fp.close()
779 fp.close()
780
780
781 def writefile(path, mode, text):
781 def writefile(path, text):
782 fp = open(path, mode)
782 fp = open(path, 'wb')
783 try:
784 fp.write(text)
785 finally:
786 fp.close()
787
788 def appendfile(path, text):
789 fp = open(path, 'ab')
783 try:
790 try:
784 fp.write(text)
791 fp.write(text)
785 finally:
792 finally:
786 fp.close()
793 fp.close()
787
794
788 class chunkbuffer(object):
795 class chunkbuffer(object):
789 """Allow arbitrary sized chunks of data to be efficiently read from an
796 """Allow arbitrary sized chunks of data to be efficiently read from an
790 iterator over chunks of arbitrary size."""
797 iterator over chunks of arbitrary size."""
791
798
792 def __init__(self, in_iter):
799 def __init__(self, in_iter):
793 """in_iter is the iterator that's iterating over the input chunks.
800 """in_iter is the iterator that's iterating over the input chunks.
794 targetsize is how big a buffer to try to maintain."""
801 targetsize is how big a buffer to try to maintain."""
795 def splitbig(chunks):
802 def splitbig(chunks):
796 for chunk in chunks:
803 for chunk in chunks:
797 if len(chunk) > 2**20:
804 if len(chunk) > 2**20:
798 pos = 0
805 pos = 0
799 while pos < len(chunk):
806 while pos < len(chunk):
800 end = pos + 2 ** 18
807 end = pos + 2 ** 18
801 yield chunk[pos:end]
808 yield chunk[pos:end]
802 pos = end
809 pos = end
803 else:
810 else:
804 yield chunk
811 yield chunk
805 self.iter = splitbig(in_iter)
812 self.iter = splitbig(in_iter)
806 self._queue = []
813 self._queue = []
807
814
808 def read(self, l):
815 def read(self, l):
809 """Read L bytes of data from the iterator of chunks of data.
816 """Read L bytes of data from the iterator of chunks of data.
810 Returns less than L bytes if the iterator runs dry."""
817 Returns less than L bytes if the iterator runs dry."""
811 left = l
818 left = l
812 buf = ''
819 buf = ''
813 queue = self._queue
820 queue = self._queue
814 while left > 0:
821 while left > 0:
815 # refill the queue
822 # refill the queue
816 if not queue:
823 if not queue:
817 target = 2**18
824 target = 2**18
818 for chunk in self.iter:
825 for chunk in self.iter:
819 queue.append(chunk)
826 queue.append(chunk)
820 target -= len(chunk)
827 target -= len(chunk)
821 if target <= 0:
828 if target <= 0:
822 break
829 break
823 if not queue:
830 if not queue:
824 break
831 break
825
832
826 chunk = queue.pop(0)
833 chunk = queue.pop(0)
827 left -= len(chunk)
834 left -= len(chunk)
828 if left < 0:
835 if left < 0:
829 queue.insert(0, chunk[left:])
836 queue.insert(0, chunk[left:])
830 buf += chunk[:left]
837 buf += chunk[:left]
831 else:
838 else:
832 buf += chunk
839 buf += chunk
833
840
834 return buf
841 return buf
835
842
836 def filechunkiter(f, size=65536, limit=None):
843 def filechunkiter(f, size=65536, limit=None):
837 """Create a generator that produces the data in the file size
844 """Create a generator that produces the data in the file size
838 (default 65536) bytes at a time, up to optional limit (default is
845 (default 65536) bytes at a time, up to optional limit (default is
839 to read all data). Chunks may be less than size bytes if the
846 to read all data). Chunks may be less than size bytes if the
840 chunk is the last chunk in the file, or the file is a socket or
847 chunk is the last chunk in the file, or the file is a socket or
841 some other type of file that sometimes reads less data than is
848 some other type of file that sometimes reads less data than is
842 requested."""
849 requested."""
843 assert size >= 0
850 assert size >= 0
844 assert limit is None or limit >= 0
851 assert limit is None or limit >= 0
845 while True:
852 while True:
846 if limit is None:
853 if limit is None:
847 nbytes = size
854 nbytes = size
848 else:
855 else:
849 nbytes = min(limit, size)
856 nbytes = min(limit, size)
850 s = nbytes and f.read(nbytes)
857 s = nbytes and f.read(nbytes)
851 if not s:
858 if not s:
852 break
859 break
853 if limit:
860 if limit:
854 limit -= len(s)
861 limit -= len(s)
855 yield s
862 yield s
856
863
857 def makedate():
864 def makedate():
858 lt = time.localtime()
865 lt = time.localtime()
859 if lt[8] == 1 and time.daylight:
866 if lt[8] == 1 and time.daylight:
860 tz = time.altzone
867 tz = time.altzone
861 else:
868 else:
862 tz = time.timezone
869 tz = time.timezone
863 t = time.mktime(lt)
870 t = time.mktime(lt)
864 if t < 0:
871 if t < 0:
865 hint = _("check your clock")
872 hint = _("check your clock")
866 raise Abort(_("negative timestamp: %d") % t, hint=hint)
873 raise Abort(_("negative timestamp: %d") % t, hint=hint)
867 return t, tz
874 return t, tz
868
875
869 def datestr(date=None, format='%a %b %d %H:%M:%S %Y %1%2'):
876 def datestr(date=None, format='%a %b %d %H:%M:%S %Y %1%2'):
870 """represent a (unixtime, offset) tuple as a localized time.
877 """represent a (unixtime, offset) tuple as a localized time.
871 unixtime is seconds since the epoch, and offset is the time zone's
878 unixtime is seconds since the epoch, and offset is the time zone's
872 number of seconds away from UTC. if timezone is false, do not
879 number of seconds away from UTC. if timezone is false, do not
873 append time zone to string."""
880 append time zone to string."""
874 t, tz = date or makedate()
881 t, tz = date or makedate()
875 if t < 0:
882 if t < 0:
876 t = 0 # time.gmtime(lt) fails on Windows for lt < -43200
883 t = 0 # time.gmtime(lt) fails on Windows for lt < -43200
877 tz = 0
884 tz = 0
878 if "%1" in format or "%2" in format:
885 if "%1" in format or "%2" in format:
879 sign = (tz > 0) and "-" or "+"
886 sign = (tz > 0) and "-" or "+"
880 minutes = abs(tz) // 60
887 minutes = abs(tz) // 60
881 format = format.replace("%1", "%c%02d" % (sign, minutes // 60))
888 format = format.replace("%1", "%c%02d" % (sign, minutes // 60))
882 format = format.replace("%2", "%02d" % (minutes % 60))
889 format = format.replace("%2", "%02d" % (minutes % 60))
883 s = time.strftime(format, time.gmtime(float(t) - tz))
890 s = time.strftime(format, time.gmtime(float(t) - tz))
884 return s
891 return s
885
892
886 def shortdate(date=None):
893 def shortdate(date=None):
887 """turn (timestamp, tzoff) tuple into iso 8631 date."""
894 """turn (timestamp, tzoff) tuple into iso 8631 date."""
888 return datestr(date, format='%Y-%m-%d')
895 return datestr(date, format='%Y-%m-%d')
889
896
890 def strdate(string, format, defaults=[]):
897 def strdate(string, format, defaults=[]):
891 """parse a localized time string and return a (unixtime, offset) tuple.
898 """parse a localized time string and return a (unixtime, offset) tuple.
892 if the string cannot be parsed, ValueError is raised."""
899 if the string cannot be parsed, ValueError is raised."""
893 def timezone(string):
900 def timezone(string):
894 tz = string.split()[-1]
901 tz = string.split()[-1]
895 if tz[0] in "+-" and len(tz) == 5 and tz[1:].isdigit():
902 if tz[0] in "+-" and len(tz) == 5 and tz[1:].isdigit():
896 sign = (tz[0] == "+") and 1 or -1
903 sign = (tz[0] == "+") and 1 or -1
897 hours = int(tz[1:3])
904 hours = int(tz[1:3])
898 minutes = int(tz[3:5])
905 minutes = int(tz[3:5])
899 return -sign * (hours * 60 + minutes) * 60
906 return -sign * (hours * 60 + minutes) * 60
900 if tz == "GMT" or tz == "UTC":
907 if tz == "GMT" or tz == "UTC":
901 return 0
908 return 0
902 return None
909 return None
903
910
904 # NOTE: unixtime = localunixtime + offset
911 # NOTE: unixtime = localunixtime + offset
905 offset, date = timezone(string), string
912 offset, date = timezone(string), string
906 if offset is not None:
913 if offset is not None:
907 date = " ".join(string.split()[:-1])
914 date = " ".join(string.split()[:-1])
908
915
909 # add missing elements from defaults
916 # add missing elements from defaults
910 usenow = False # default to using biased defaults
917 usenow = False # default to using biased defaults
911 for part in ("S", "M", "HI", "d", "mb", "yY"): # decreasing specificity
918 for part in ("S", "M", "HI", "d", "mb", "yY"): # decreasing specificity
912 found = [True for p in part if ("%"+p) in format]
919 found = [True for p in part if ("%"+p) in format]
913 if not found:
920 if not found:
914 date += "@" + defaults[part][usenow]
921 date += "@" + defaults[part][usenow]
915 format += "@%" + part[0]
922 format += "@%" + part[0]
916 else:
923 else:
917 # We've found a specific time element, less specific time
924 # We've found a specific time element, less specific time
918 # elements are relative to today
925 # elements are relative to today
919 usenow = True
926 usenow = True
920
927
921 timetuple = time.strptime(date, format)
928 timetuple = time.strptime(date, format)
922 localunixtime = int(calendar.timegm(timetuple))
929 localunixtime = int(calendar.timegm(timetuple))
923 if offset is None:
930 if offset is None:
924 # local timezone
931 # local timezone
925 unixtime = int(time.mktime(timetuple))
932 unixtime = int(time.mktime(timetuple))
926 offset = unixtime - localunixtime
933 offset = unixtime - localunixtime
927 else:
934 else:
928 unixtime = localunixtime + offset
935 unixtime = localunixtime + offset
929 return unixtime, offset
936 return unixtime, offset
930
937
931 def parsedate(date, formats=None, bias={}):
938 def parsedate(date, formats=None, bias={}):
932 """parse a localized date/time and return a (unixtime, offset) tuple.
939 """parse a localized date/time and return a (unixtime, offset) tuple.
933
940
934 The date may be a "unixtime offset" string or in one of the specified
941 The date may be a "unixtime offset" string or in one of the specified
935 formats. If the date already is a (unixtime, offset) tuple, it is returned.
942 formats. If the date already is a (unixtime, offset) tuple, it is returned.
936 """
943 """
937 if not date:
944 if not date:
938 return 0, 0
945 return 0, 0
939 if isinstance(date, tuple) and len(date) == 2:
946 if isinstance(date, tuple) and len(date) == 2:
940 return date
947 return date
941 if not formats:
948 if not formats:
942 formats = defaultdateformats
949 formats = defaultdateformats
943 date = date.strip()
950 date = date.strip()
944 try:
951 try:
945 when, offset = map(int, date.split(' '))
952 when, offset = map(int, date.split(' '))
946 except ValueError:
953 except ValueError:
947 # fill out defaults
954 # fill out defaults
948 now = makedate()
955 now = makedate()
949 defaults = {}
956 defaults = {}
950 for part in ("d", "mb", "yY", "HI", "M", "S"):
957 for part in ("d", "mb", "yY", "HI", "M", "S"):
951 # this piece is for rounding the specific end of unknowns
958 # this piece is for rounding the specific end of unknowns
952 b = bias.get(part)
959 b = bias.get(part)
953 if b is None:
960 if b is None:
954 if part[0] in "HMS":
961 if part[0] in "HMS":
955 b = "00"
962 b = "00"
956 else:
963 else:
957 b = "0"
964 b = "0"
958
965
959 # this piece is for matching the generic end to today's date
966 # this piece is for matching the generic end to today's date
960 n = datestr(now, "%" + part[0])
967 n = datestr(now, "%" + part[0])
961
968
962 defaults[part] = (b, n)
969 defaults[part] = (b, n)
963
970
964 for format in formats:
971 for format in formats:
965 try:
972 try:
966 when, offset = strdate(date, format, defaults)
973 when, offset = strdate(date, format, defaults)
967 except (ValueError, OverflowError):
974 except (ValueError, OverflowError):
968 pass
975 pass
969 else:
976 else:
970 break
977 break
971 else:
978 else:
972 raise Abort(_('invalid date: %r') % date)
979 raise Abort(_('invalid date: %r') % date)
973 # validate explicit (probably user-specified) date and
980 # validate explicit (probably user-specified) date and
974 # time zone offset. values must fit in signed 32 bits for
981 # time zone offset. values must fit in signed 32 bits for
975 # current 32-bit linux runtimes. timezones go from UTC-12
982 # current 32-bit linux runtimes. timezones go from UTC-12
976 # to UTC+14
983 # to UTC+14
977 if abs(when) > 0x7fffffff:
984 if abs(when) > 0x7fffffff:
978 raise Abort(_('date exceeds 32 bits: %d') % when)
985 raise Abort(_('date exceeds 32 bits: %d') % when)
979 if when < 0:
986 if when < 0:
980 raise Abort(_('negative date value: %d') % when)
987 raise Abort(_('negative date value: %d') % when)
981 if offset < -50400 or offset > 43200:
988 if offset < -50400 or offset > 43200:
982 raise Abort(_('impossible time zone offset: %d') % offset)
989 raise Abort(_('impossible time zone offset: %d') % offset)
983 return when, offset
990 return when, offset
984
991
985 def matchdate(date):
992 def matchdate(date):
986 """Return a function that matches a given date match specifier
993 """Return a function that matches a given date match specifier
987
994
988 Formats include:
995 Formats include:
989
996
990 '{date}' match a given date to the accuracy provided
997 '{date}' match a given date to the accuracy provided
991
998
992 '<{date}' on or before a given date
999 '<{date}' on or before a given date
993
1000
994 '>{date}' on or after a given date
1001 '>{date}' on or after a given date
995
1002
996 >>> p1 = parsedate("10:29:59")
1003 >>> p1 = parsedate("10:29:59")
997 >>> p2 = parsedate("10:30:00")
1004 >>> p2 = parsedate("10:30:00")
998 >>> p3 = parsedate("10:30:59")
1005 >>> p3 = parsedate("10:30:59")
999 >>> p4 = parsedate("10:31:00")
1006 >>> p4 = parsedate("10:31:00")
1000 >>> p5 = parsedate("Sep 15 10:30:00 1999")
1007 >>> p5 = parsedate("Sep 15 10:30:00 1999")
1001 >>> f = matchdate("10:30")
1008 >>> f = matchdate("10:30")
1002 >>> f(p1[0])
1009 >>> f(p1[0])
1003 False
1010 False
1004 >>> f(p2[0])
1011 >>> f(p2[0])
1005 True
1012 True
1006 >>> f(p3[0])
1013 >>> f(p3[0])
1007 True
1014 True
1008 >>> f(p4[0])
1015 >>> f(p4[0])
1009 False
1016 False
1010 >>> f(p5[0])
1017 >>> f(p5[0])
1011 False
1018 False
1012 """
1019 """
1013
1020
1014 def lower(date):
1021 def lower(date):
1015 d = dict(mb="1", d="1")
1022 d = dict(mb="1", d="1")
1016 return parsedate(date, extendeddateformats, d)[0]
1023 return parsedate(date, extendeddateformats, d)[0]
1017
1024
1018 def upper(date):
1025 def upper(date):
1019 d = dict(mb="12", HI="23", M="59", S="59")
1026 d = dict(mb="12", HI="23", M="59", S="59")
1020 for days in ("31", "30", "29"):
1027 for days in ("31", "30", "29"):
1021 try:
1028 try:
1022 d["d"] = days
1029 d["d"] = days
1023 return parsedate(date, extendeddateformats, d)[0]
1030 return parsedate(date, extendeddateformats, d)[0]
1024 except:
1031 except:
1025 pass
1032 pass
1026 d["d"] = "28"
1033 d["d"] = "28"
1027 return parsedate(date, extendeddateformats, d)[0]
1034 return parsedate(date, extendeddateformats, d)[0]
1028
1035
1029 date = date.strip()
1036 date = date.strip()
1030
1037
1031 if not date:
1038 if not date:
1032 raise Abort(_("dates cannot consist entirely of whitespace"))
1039 raise Abort(_("dates cannot consist entirely of whitespace"))
1033 elif date[0] == "<":
1040 elif date[0] == "<":
1034 if not date[1:]:
1041 if not date[1:]:
1035 raise Abort(_("invalid day spec, use '<DATE'"))
1042 raise Abort(_("invalid day spec, use '<DATE'"))
1036 when = upper(date[1:])
1043 when = upper(date[1:])
1037 return lambda x: x <= when
1044 return lambda x: x <= when
1038 elif date[0] == ">":
1045 elif date[0] == ">":
1039 if not date[1:]:
1046 if not date[1:]:
1040 raise Abort(_("invalid day spec, use '>DATE'"))
1047 raise Abort(_("invalid day spec, use '>DATE'"))
1041 when = lower(date[1:])
1048 when = lower(date[1:])
1042 return lambda x: x >= when
1049 return lambda x: x >= when
1043 elif date[0] == "-":
1050 elif date[0] == "-":
1044 try:
1051 try:
1045 days = int(date[1:])
1052 days = int(date[1:])
1046 except ValueError:
1053 except ValueError:
1047 raise Abort(_("invalid day spec: %s") % date[1:])
1054 raise Abort(_("invalid day spec: %s") % date[1:])
1048 if days < 0:
1055 if days < 0:
1049 raise Abort(_("%s must be nonnegative (see 'hg help dates')")
1056 raise Abort(_("%s must be nonnegative (see 'hg help dates')")
1050 % date[1:])
1057 % date[1:])
1051 when = makedate()[0] - days * 3600 * 24
1058 when = makedate()[0] - days * 3600 * 24
1052 return lambda x: x >= when
1059 return lambda x: x >= when
1053 elif " to " in date:
1060 elif " to " in date:
1054 a, b = date.split(" to ")
1061 a, b = date.split(" to ")
1055 start, stop = lower(a), upper(b)
1062 start, stop = lower(a), upper(b)
1056 return lambda x: x >= start and x <= stop
1063 return lambda x: x >= start and x <= stop
1057 else:
1064 else:
1058 start, stop = lower(date), upper(date)
1065 start, stop = lower(date), upper(date)
1059 return lambda x: x >= start and x <= stop
1066 return lambda x: x >= start and x <= stop
1060
1067
1061 def shortuser(user):
1068 def shortuser(user):
1062 """Return a short representation of a user name or email address."""
1069 """Return a short representation of a user name or email address."""
1063 f = user.find('@')
1070 f = user.find('@')
1064 if f >= 0:
1071 if f >= 0:
1065 user = user[:f]
1072 user = user[:f]
1066 f = user.find('<')
1073 f = user.find('<')
1067 if f >= 0:
1074 if f >= 0:
1068 user = user[f + 1:]
1075 user = user[f + 1:]
1069 f = user.find(' ')
1076 f = user.find(' ')
1070 if f >= 0:
1077 if f >= 0:
1071 user = user[:f]
1078 user = user[:f]
1072 f = user.find('.')
1079 f = user.find('.')
1073 if f >= 0:
1080 if f >= 0:
1074 user = user[:f]
1081 user = user[:f]
1075 return user
1082 return user
1076
1083
1077 def email(author):
1084 def email(author):
1078 '''get email of author.'''
1085 '''get email of author.'''
1079 r = author.find('>')
1086 r = author.find('>')
1080 if r == -1:
1087 if r == -1:
1081 r = None
1088 r = None
1082 return author[author.find('<') + 1:r]
1089 return author[author.find('<') + 1:r]
1083
1090
1084 def _ellipsis(text, maxlength):
1091 def _ellipsis(text, maxlength):
1085 if len(text) <= maxlength:
1092 if len(text) <= maxlength:
1086 return text, False
1093 return text, False
1087 else:
1094 else:
1088 return "%s..." % (text[:maxlength - 3]), True
1095 return "%s..." % (text[:maxlength - 3]), True
1089
1096
1090 def ellipsis(text, maxlength=400):
1097 def ellipsis(text, maxlength=400):
1091 """Trim string to at most maxlength (default: 400) characters."""
1098 """Trim string to at most maxlength (default: 400) characters."""
1092 try:
1099 try:
1093 # use unicode not to split at intermediate multi-byte sequence
1100 # use unicode not to split at intermediate multi-byte sequence
1094 utext, truncated = _ellipsis(text.decode(encoding.encoding),
1101 utext, truncated = _ellipsis(text.decode(encoding.encoding),
1095 maxlength)
1102 maxlength)
1096 if not truncated:
1103 if not truncated:
1097 return text
1104 return text
1098 return utext.encode(encoding.encoding)
1105 return utext.encode(encoding.encoding)
1099 except (UnicodeDecodeError, UnicodeEncodeError):
1106 except (UnicodeDecodeError, UnicodeEncodeError):
1100 return _ellipsis(text, maxlength)[0]
1107 return _ellipsis(text, maxlength)[0]
1101
1108
1102 def bytecount(nbytes):
1109 def bytecount(nbytes):
1103 '''return byte count formatted as readable string, with units'''
1110 '''return byte count formatted as readable string, with units'''
1104
1111
1105 units = (
1112 units = (
1106 (100, 1 << 30, _('%.0f GB')),
1113 (100, 1 << 30, _('%.0f GB')),
1107 (10, 1 << 30, _('%.1f GB')),
1114 (10, 1 << 30, _('%.1f GB')),
1108 (1, 1 << 30, _('%.2f GB')),
1115 (1, 1 << 30, _('%.2f GB')),
1109 (100, 1 << 20, _('%.0f MB')),
1116 (100, 1 << 20, _('%.0f MB')),
1110 (10, 1 << 20, _('%.1f MB')),
1117 (10, 1 << 20, _('%.1f MB')),
1111 (1, 1 << 20, _('%.2f MB')),
1118 (1, 1 << 20, _('%.2f MB')),
1112 (100, 1 << 10, _('%.0f KB')),
1119 (100, 1 << 10, _('%.0f KB')),
1113 (10, 1 << 10, _('%.1f KB')),
1120 (10, 1 << 10, _('%.1f KB')),
1114 (1, 1 << 10, _('%.2f KB')),
1121 (1, 1 << 10, _('%.2f KB')),
1115 (1, 1, _('%.0f bytes')),
1122 (1, 1, _('%.0f bytes')),
1116 )
1123 )
1117
1124
1118 for multiplier, divisor, format in units:
1125 for multiplier, divisor, format in units:
1119 if nbytes >= divisor * multiplier:
1126 if nbytes >= divisor * multiplier:
1120 return format % (nbytes / float(divisor))
1127 return format % (nbytes / float(divisor))
1121 return units[-1][2] % nbytes
1128 return units[-1][2] % nbytes
1122
1129
1123 def uirepr(s):
1130 def uirepr(s):
1124 # Avoid double backslash in Windows path repr()
1131 # Avoid double backslash in Windows path repr()
1125 return repr(s).replace('\\\\', '\\')
1132 return repr(s).replace('\\\\', '\\')
1126
1133
1127 # delay import of textwrap
1134 # delay import of textwrap
1128 def MBTextWrapper(**kwargs):
1135 def MBTextWrapper(**kwargs):
1129 class tw(textwrap.TextWrapper):
1136 class tw(textwrap.TextWrapper):
1130 """
1137 """
1131 Extend TextWrapper for double-width characters.
1138 Extend TextWrapper for double-width characters.
1132
1139
1133 Some Asian characters use two terminal columns instead of one.
1140 Some Asian characters use two terminal columns instead of one.
1134 A good example of this behavior can be seen with u'\u65e5\u672c',
1141 A good example of this behavior can be seen with u'\u65e5\u672c',
1135 the two Japanese characters for "Japan":
1142 the two Japanese characters for "Japan":
1136 len() returns 2, but when printed to a terminal, they eat 4 columns.
1143 len() returns 2, but when printed to a terminal, they eat 4 columns.
1137
1144
1138 (Note that this has nothing to do whatsoever with unicode
1145 (Note that this has nothing to do whatsoever with unicode
1139 representation, or encoding of the underlying string)
1146 representation, or encoding of the underlying string)
1140 """
1147 """
1141 def __init__(self, **kwargs):
1148 def __init__(self, **kwargs):
1142 textwrap.TextWrapper.__init__(self, **kwargs)
1149 textwrap.TextWrapper.__init__(self, **kwargs)
1143
1150
1144 def _cutdown(self, str, space_left):
1151 def _cutdown(self, str, space_left):
1145 l = 0
1152 l = 0
1146 ucstr = unicode(str, encoding.encoding)
1153 ucstr = unicode(str, encoding.encoding)
1147 colwidth = unicodedata.east_asian_width
1154 colwidth = unicodedata.east_asian_width
1148 for i in xrange(len(ucstr)):
1155 for i in xrange(len(ucstr)):
1149 l += colwidth(ucstr[i]) in 'WFA' and 2 or 1
1156 l += colwidth(ucstr[i]) in 'WFA' and 2 or 1
1150 if space_left < l:
1157 if space_left < l:
1151 return (ucstr[:i].encode(encoding.encoding),
1158 return (ucstr[:i].encode(encoding.encoding),
1152 ucstr[i:].encode(encoding.encoding))
1159 ucstr[i:].encode(encoding.encoding))
1153 return str, ''
1160 return str, ''
1154
1161
1155 # overriding of base class
1162 # overriding of base class
1156 def _handle_long_word(self, reversed_chunks, cur_line, cur_len, width):
1163 def _handle_long_word(self, reversed_chunks, cur_line, cur_len, width):
1157 space_left = max(width - cur_len, 1)
1164 space_left = max(width - cur_len, 1)
1158
1165
1159 if self.break_long_words:
1166 if self.break_long_words:
1160 cut, res = self._cutdown(reversed_chunks[-1], space_left)
1167 cut, res = self._cutdown(reversed_chunks[-1], space_left)
1161 cur_line.append(cut)
1168 cur_line.append(cut)
1162 reversed_chunks[-1] = res
1169 reversed_chunks[-1] = res
1163 elif not cur_line:
1170 elif not cur_line:
1164 cur_line.append(reversed_chunks.pop())
1171 cur_line.append(reversed_chunks.pop())
1165
1172
1166 global MBTextWrapper
1173 global MBTextWrapper
1167 MBTextWrapper = tw
1174 MBTextWrapper = tw
1168 return tw(**kwargs)
1175 return tw(**kwargs)
1169
1176
1170 def wrap(line, width, initindent='', hangindent=''):
1177 def wrap(line, width, initindent='', hangindent=''):
1171 maxindent = max(len(hangindent), len(initindent))
1178 maxindent = max(len(hangindent), len(initindent))
1172 if width <= maxindent:
1179 if width <= maxindent:
1173 # adjust for weird terminal size
1180 # adjust for weird terminal size
1174 width = max(78, maxindent + 1)
1181 width = max(78, maxindent + 1)
1175 wrapper = MBTextWrapper(width=width,
1182 wrapper = MBTextWrapper(width=width,
1176 initial_indent=initindent,
1183 initial_indent=initindent,
1177 subsequent_indent=hangindent)
1184 subsequent_indent=hangindent)
1178 return wrapper.fill(line)
1185 return wrapper.fill(line)
1179
1186
1180 def iterlines(iterator):
1187 def iterlines(iterator):
1181 for chunk in iterator:
1188 for chunk in iterator:
1182 for line in chunk.splitlines():
1189 for line in chunk.splitlines():
1183 yield line
1190 yield line
1184
1191
1185 def expandpath(path):
1192 def expandpath(path):
1186 return os.path.expanduser(os.path.expandvars(path))
1193 return os.path.expanduser(os.path.expandvars(path))
1187
1194
1188 def hgcmd():
1195 def hgcmd():
1189 """Return the command used to execute current hg
1196 """Return the command used to execute current hg
1190
1197
1191 This is different from hgexecutable() because on Windows we want
1198 This is different from hgexecutable() because on Windows we want
1192 to avoid things opening new shell windows like batch files, so we
1199 to avoid things opening new shell windows like batch files, so we
1193 get either the python call or current executable.
1200 get either the python call or current executable.
1194 """
1201 """
1195 if main_is_frozen():
1202 if main_is_frozen():
1196 return [sys.executable]
1203 return [sys.executable]
1197 return gethgcmd()
1204 return gethgcmd()
1198
1205
1199 def rundetached(args, condfn):
1206 def rundetached(args, condfn):
1200 """Execute the argument list in a detached process.
1207 """Execute the argument list in a detached process.
1201
1208
1202 condfn is a callable which is called repeatedly and should return
1209 condfn is a callable which is called repeatedly and should return
1203 True once the child process is known to have started successfully.
1210 True once the child process is known to have started successfully.
1204 At this point, the child process PID is returned. If the child
1211 At this point, the child process PID is returned. If the child
1205 process fails to start or finishes before condfn() evaluates to
1212 process fails to start or finishes before condfn() evaluates to
1206 True, return -1.
1213 True, return -1.
1207 """
1214 """
1208 # Windows case is easier because the child process is either
1215 # Windows case is easier because the child process is either
1209 # successfully starting and validating the condition or exiting
1216 # successfully starting and validating the condition or exiting
1210 # on failure. We just poll on its PID. On Unix, if the child
1217 # on failure. We just poll on its PID. On Unix, if the child
1211 # process fails to start, it will be left in a zombie state until
1218 # process fails to start, it will be left in a zombie state until
1212 # the parent wait on it, which we cannot do since we expect a long
1219 # the parent wait on it, which we cannot do since we expect a long
1213 # running process on success. Instead we listen for SIGCHLD telling
1220 # running process on success. Instead we listen for SIGCHLD telling
1214 # us our child process terminated.
1221 # us our child process terminated.
1215 terminated = set()
1222 terminated = set()
1216 def handler(signum, frame):
1223 def handler(signum, frame):
1217 terminated.add(os.wait())
1224 terminated.add(os.wait())
1218 prevhandler = None
1225 prevhandler = None
1219 if hasattr(signal, 'SIGCHLD'):
1226 if hasattr(signal, 'SIGCHLD'):
1220 prevhandler = signal.signal(signal.SIGCHLD, handler)
1227 prevhandler = signal.signal(signal.SIGCHLD, handler)
1221 try:
1228 try:
1222 pid = spawndetached(args)
1229 pid = spawndetached(args)
1223 while not condfn():
1230 while not condfn():
1224 if ((pid in terminated or not testpid(pid))
1231 if ((pid in terminated or not testpid(pid))
1225 and not condfn()):
1232 and not condfn()):
1226 return -1
1233 return -1
1227 time.sleep(0.1)
1234 time.sleep(0.1)
1228 return pid
1235 return pid
1229 finally:
1236 finally:
1230 if prevhandler is not None:
1237 if prevhandler is not None:
1231 signal.signal(signal.SIGCHLD, prevhandler)
1238 signal.signal(signal.SIGCHLD, prevhandler)
1232
1239
1233 try:
1240 try:
1234 any, all = any, all
1241 any, all = any, all
1235 except NameError:
1242 except NameError:
1236 def any(iterable):
1243 def any(iterable):
1237 for i in iterable:
1244 for i in iterable:
1238 if i:
1245 if i:
1239 return True
1246 return True
1240 return False
1247 return False
1241
1248
1242 def all(iterable):
1249 def all(iterable):
1243 for i in iterable:
1250 for i in iterable:
1244 if not i:
1251 if not i:
1245 return False
1252 return False
1246 return True
1253 return True
1247
1254
1248 def interpolate(prefix, mapping, s, fn=None, escape_prefix=False):
1255 def interpolate(prefix, mapping, s, fn=None, escape_prefix=False):
1249 """Return the result of interpolating items in the mapping into string s.
1256 """Return the result of interpolating items in the mapping into string s.
1250
1257
1251 prefix is a single character string, or a two character string with
1258 prefix is a single character string, or a two character string with
1252 a backslash as the first character if the prefix needs to be escaped in
1259 a backslash as the first character if the prefix needs to be escaped in
1253 a regular expression.
1260 a regular expression.
1254
1261
1255 fn is an optional function that will be applied to the replacement text
1262 fn is an optional function that will be applied to the replacement text
1256 just before replacement.
1263 just before replacement.
1257
1264
1258 escape_prefix is an optional flag that allows using doubled prefix for
1265 escape_prefix is an optional flag that allows using doubled prefix for
1259 its escaping.
1266 its escaping.
1260 """
1267 """
1261 fn = fn or (lambda s: s)
1268 fn = fn or (lambda s: s)
1262 patterns = '|'.join(mapping.keys())
1269 patterns = '|'.join(mapping.keys())
1263 if escape_prefix:
1270 if escape_prefix:
1264 patterns += '|' + prefix
1271 patterns += '|' + prefix
1265 if len(prefix) > 1:
1272 if len(prefix) > 1:
1266 prefix_char = prefix[1:]
1273 prefix_char = prefix[1:]
1267 else:
1274 else:
1268 prefix_char = prefix
1275 prefix_char = prefix
1269 mapping[prefix_char] = prefix_char
1276 mapping[prefix_char] = prefix_char
1270 r = re.compile(r'%s(%s)' % (prefix, patterns))
1277 r = re.compile(r'%s(%s)' % (prefix, patterns))
1271 return r.sub(lambda x: fn(mapping[x.group()[1:]]), s)
1278 return r.sub(lambda x: fn(mapping[x.group()[1:]]), s)
1272
1279
1273 def getport(port):
1280 def getport(port):
1274 """Return the port for a given network service.
1281 """Return the port for a given network service.
1275
1282
1276 If port is an integer, it's returned as is. If it's a string, it's
1283 If port is an integer, it's returned as is. If it's a string, it's
1277 looked up using socket.getservbyname(). If there's no matching
1284 looked up using socket.getservbyname(). If there's no matching
1278 service, util.Abort is raised.
1285 service, util.Abort is raised.
1279 """
1286 """
1280 try:
1287 try:
1281 return int(port)
1288 return int(port)
1282 except ValueError:
1289 except ValueError:
1283 pass
1290 pass
1284
1291
1285 try:
1292 try:
1286 return socket.getservbyname(port)
1293 return socket.getservbyname(port)
1287 except socket.error:
1294 except socket.error:
1288 raise Abort(_("no port number associated with service '%s'") % port)
1295 raise Abort(_("no port number associated with service '%s'") % port)
1289
1296
1290 _booleans = {'1': True, 'yes': True, 'true': True, 'on': True, 'always': True,
1297 _booleans = {'1': True, 'yes': True, 'true': True, 'on': True, 'always': True,
1291 '0': False, 'no': False, 'false': False, 'off': False,
1298 '0': False, 'no': False, 'false': False, 'off': False,
1292 'never': False}
1299 'never': False}
1293
1300
1294 def parsebool(s):
1301 def parsebool(s):
1295 """Parse s into a boolean.
1302 """Parse s into a boolean.
1296
1303
1297 If s is not a valid boolean, returns None.
1304 If s is not a valid boolean, returns None.
1298 """
1305 """
1299 return _booleans.get(s.lower(), None)
1306 return _booleans.get(s.lower(), None)
1300
1307
1301 _hexdig = '0123456789ABCDEFabcdef'
1308 _hexdig = '0123456789ABCDEFabcdef'
1302 _hextochr = dict((a + b, chr(int(a + b, 16)))
1309 _hextochr = dict((a + b, chr(int(a + b, 16)))
1303 for a in _hexdig for b in _hexdig)
1310 for a in _hexdig for b in _hexdig)
1304
1311
1305 def _urlunquote(s):
1312 def _urlunquote(s):
1306 """unquote('abc%20def') -> 'abc def'."""
1313 """unquote('abc%20def') -> 'abc def'."""
1307 res = s.split('%')
1314 res = s.split('%')
1308 # fastpath
1315 # fastpath
1309 if len(res) == 1:
1316 if len(res) == 1:
1310 return s
1317 return s
1311 s = res[0]
1318 s = res[0]
1312 for item in res[1:]:
1319 for item in res[1:]:
1313 try:
1320 try:
1314 s += _hextochr[item[:2]] + item[2:]
1321 s += _hextochr[item[:2]] + item[2:]
1315 except KeyError:
1322 except KeyError:
1316 s += '%' + item
1323 s += '%' + item
1317 except UnicodeDecodeError:
1324 except UnicodeDecodeError:
1318 s += unichr(int(item[:2], 16)) + item[2:]
1325 s += unichr(int(item[:2], 16)) + item[2:]
1319 return s
1326 return s
1320
1327
1321 class url(object):
1328 class url(object):
1322 r"""Reliable URL parser.
1329 r"""Reliable URL parser.
1323
1330
1324 This parses URLs and provides attributes for the following
1331 This parses URLs and provides attributes for the following
1325 components:
1332 components:
1326
1333
1327 <scheme>://<user>:<passwd>@<host>:<port>/<path>?<query>#<fragment>
1334 <scheme>://<user>:<passwd>@<host>:<port>/<path>?<query>#<fragment>
1328
1335
1329 Missing components are set to None. The only exception is
1336 Missing components are set to None. The only exception is
1330 fragment, which is set to '' if present but empty.
1337 fragment, which is set to '' if present but empty.
1331
1338
1332 If parsefragment is False, fragment is included in query. If
1339 If parsefragment is False, fragment is included in query. If
1333 parsequery is False, query is included in path. If both are
1340 parsequery is False, query is included in path. If both are
1334 False, both fragment and query are included in path.
1341 False, both fragment and query are included in path.
1335
1342
1336 See http://www.ietf.org/rfc/rfc2396.txt for more information.
1343 See http://www.ietf.org/rfc/rfc2396.txt for more information.
1337
1344
1338 Note that for backward compatibility reasons, bundle URLs do not
1345 Note that for backward compatibility reasons, bundle URLs do not
1339 take host names. That means 'bundle://../' has a path of '../'.
1346 take host names. That means 'bundle://../' has a path of '../'.
1340
1347
1341 Examples:
1348 Examples:
1342
1349
1343 >>> url('http://www.ietf.org/rfc/rfc2396.txt')
1350 >>> url('http://www.ietf.org/rfc/rfc2396.txt')
1344 <url scheme: 'http', host: 'www.ietf.org', path: 'rfc/rfc2396.txt'>
1351 <url scheme: 'http', host: 'www.ietf.org', path: 'rfc/rfc2396.txt'>
1345 >>> url('ssh://[::1]:2200//home/joe/repo')
1352 >>> url('ssh://[::1]:2200//home/joe/repo')
1346 <url scheme: 'ssh', host: '[::1]', port: '2200', path: '/home/joe/repo'>
1353 <url scheme: 'ssh', host: '[::1]', port: '2200', path: '/home/joe/repo'>
1347 >>> url('file:///home/joe/repo')
1354 >>> url('file:///home/joe/repo')
1348 <url scheme: 'file', path: '/home/joe/repo'>
1355 <url scheme: 'file', path: '/home/joe/repo'>
1349 >>> url('bundle:foo')
1356 >>> url('bundle:foo')
1350 <url scheme: 'bundle', path: 'foo'>
1357 <url scheme: 'bundle', path: 'foo'>
1351 >>> url('bundle://../foo')
1358 >>> url('bundle://../foo')
1352 <url scheme: 'bundle', path: '../foo'>
1359 <url scheme: 'bundle', path: '../foo'>
1353 >>> url(r'c:\foo\bar')
1360 >>> url(r'c:\foo\bar')
1354 <url path: 'c:\\foo\\bar'>
1361 <url path: 'c:\\foo\\bar'>
1355
1362
1356 Authentication credentials:
1363 Authentication credentials:
1357
1364
1358 >>> url('ssh://joe:xyz@x/repo')
1365 >>> url('ssh://joe:xyz@x/repo')
1359 <url scheme: 'ssh', user: 'joe', passwd: 'xyz', host: 'x', path: 'repo'>
1366 <url scheme: 'ssh', user: 'joe', passwd: 'xyz', host: 'x', path: 'repo'>
1360 >>> url('ssh://joe@x/repo')
1367 >>> url('ssh://joe@x/repo')
1361 <url scheme: 'ssh', user: 'joe', host: 'x', path: 'repo'>
1368 <url scheme: 'ssh', user: 'joe', host: 'x', path: 'repo'>
1362
1369
1363 Query strings and fragments:
1370 Query strings and fragments:
1364
1371
1365 >>> url('http://host/a?b#c')
1372 >>> url('http://host/a?b#c')
1366 <url scheme: 'http', host: 'host', path: 'a', query: 'b', fragment: 'c'>
1373 <url scheme: 'http', host: 'host', path: 'a', query: 'b', fragment: 'c'>
1367 >>> url('http://host/a?b#c', parsequery=False, parsefragment=False)
1374 >>> url('http://host/a?b#c', parsequery=False, parsefragment=False)
1368 <url scheme: 'http', host: 'host', path: 'a?b#c'>
1375 <url scheme: 'http', host: 'host', path: 'a?b#c'>
1369 """
1376 """
1370
1377
1371 _safechars = "!~*'()+"
1378 _safechars = "!~*'()+"
1372 _safepchars = "/!~*'()+"
1379 _safepchars = "/!~*'()+"
1373 _matchscheme = re.compile(r'^[a-zA-Z0-9+.\-]+:').match
1380 _matchscheme = re.compile(r'^[a-zA-Z0-9+.\-]+:').match
1374
1381
1375 def __init__(self, path, parsequery=True, parsefragment=True):
1382 def __init__(self, path, parsequery=True, parsefragment=True):
1376 # We slowly chomp away at path until we have only the path left
1383 # We slowly chomp away at path until we have only the path left
1377 self.scheme = self.user = self.passwd = self.host = None
1384 self.scheme = self.user = self.passwd = self.host = None
1378 self.port = self.path = self.query = self.fragment = None
1385 self.port = self.path = self.query = self.fragment = None
1379 self._localpath = True
1386 self._localpath = True
1380 self._hostport = ''
1387 self._hostport = ''
1381 self._origpath = path
1388 self._origpath = path
1382
1389
1383 # special case for Windows drive letters
1390 # special case for Windows drive letters
1384 if hasdriveletter(path):
1391 if hasdriveletter(path):
1385 self.path = path
1392 self.path = path
1386 return
1393 return
1387
1394
1388 # For compatibility reasons, we can't handle bundle paths as
1395 # For compatibility reasons, we can't handle bundle paths as
1389 # normal URLS
1396 # normal URLS
1390 if path.startswith('bundle:'):
1397 if path.startswith('bundle:'):
1391 self.scheme = 'bundle'
1398 self.scheme = 'bundle'
1392 path = path[7:]
1399 path = path[7:]
1393 if path.startswith('//'):
1400 if path.startswith('//'):
1394 path = path[2:]
1401 path = path[2:]
1395 self.path = path
1402 self.path = path
1396 return
1403 return
1397
1404
1398 if self._matchscheme(path):
1405 if self._matchscheme(path):
1399 parts = path.split(':', 1)
1406 parts = path.split(':', 1)
1400 if parts[0]:
1407 if parts[0]:
1401 self.scheme, path = parts
1408 self.scheme, path = parts
1402 self._localpath = False
1409 self._localpath = False
1403
1410
1404 if not path:
1411 if not path:
1405 path = None
1412 path = None
1406 if self._localpath:
1413 if self._localpath:
1407 self.path = ''
1414 self.path = ''
1408 return
1415 return
1409 else:
1416 else:
1410 if parsefragment and '#' in path:
1417 if parsefragment and '#' in path:
1411 path, self.fragment = path.split('#', 1)
1418 path, self.fragment = path.split('#', 1)
1412 if not path:
1419 if not path:
1413 path = None
1420 path = None
1414 if self._localpath:
1421 if self._localpath:
1415 self.path = path
1422 self.path = path
1416 return
1423 return
1417
1424
1418 if parsequery and '?' in path:
1425 if parsequery and '?' in path:
1419 path, self.query = path.split('?', 1)
1426 path, self.query = path.split('?', 1)
1420 if not path:
1427 if not path:
1421 path = None
1428 path = None
1422 if not self.query:
1429 if not self.query:
1423 self.query = None
1430 self.query = None
1424
1431
1425 # // is required to specify a host/authority
1432 # // is required to specify a host/authority
1426 if path and path.startswith('//'):
1433 if path and path.startswith('//'):
1427 parts = path[2:].split('/', 1)
1434 parts = path[2:].split('/', 1)
1428 if len(parts) > 1:
1435 if len(parts) > 1:
1429 self.host, path = parts
1436 self.host, path = parts
1430 path = path
1437 path = path
1431 else:
1438 else:
1432 self.host = parts[0]
1439 self.host = parts[0]
1433 path = None
1440 path = None
1434 if not self.host:
1441 if not self.host:
1435 self.host = None
1442 self.host = None
1436 if path:
1443 if path:
1437 path = '/' + path
1444 path = '/' + path
1438
1445
1439 if self.host and '@' in self.host:
1446 if self.host and '@' in self.host:
1440 self.user, self.host = self.host.rsplit('@', 1)
1447 self.user, self.host = self.host.rsplit('@', 1)
1441 if ':' in self.user:
1448 if ':' in self.user:
1442 self.user, self.passwd = self.user.split(':', 1)
1449 self.user, self.passwd = self.user.split(':', 1)
1443 if not self.host:
1450 if not self.host:
1444 self.host = None
1451 self.host = None
1445
1452
1446 # Don't split on colons in IPv6 addresses without ports
1453 # Don't split on colons in IPv6 addresses without ports
1447 if (self.host and ':' in self.host and
1454 if (self.host and ':' in self.host and
1448 not (self.host.startswith('[') and self.host.endswith(']'))):
1455 not (self.host.startswith('[') and self.host.endswith(']'))):
1449 self._hostport = self.host
1456 self._hostport = self.host
1450 self.host, self.port = self.host.rsplit(':', 1)
1457 self.host, self.port = self.host.rsplit(':', 1)
1451 if not self.host:
1458 if not self.host:
1452 self.host = None
1459 self.host = None
1453
1460
1454 if (self.host and self.scheme == 'file' and
1461 if (self.host and self.scheme == 'file' and
1455 self.host not in ('localhost', '127.0.0.1', '[::1]')):
1462 self.host not in ('localhost', '127.0.0.1', '[::1]')):
1456 raise Abort(_('file:// URLs can only refer to localhost'))
1463 raise Abort(_('file:// URLs can only refer to localhost'))
1457
1464
1458 self.path = path
1465 self.path = path
1459
1466
1460 for a in ('user', 'passwd', 'host', 'port',
1467 for a in ('user', 'passwd', 'host', 'port',
1461 'path', 'query', 'fragment'):
1468 'path', 'query', 'fragment'):
1462 v = getattr(self, a)
1469 v = getattr(self, a)
1463 if v is not None:
1470 if v is not None:
1464 setattr(self, a, _urlunquote(v))
1471 setattr(self, a, _urlunquote(v))
1465
1472
1466 def __repr__(self):
1473 def __repr__(self):
1467 attrs = []
1474 attrs = []
1468 for a in ('scheme', 'user', 'passwd', 'host', 'port', 'path',
1475 for a in ('scheme', 'user', 'passwd', 'host', 'port', 'path',
1469 'query', 'fragment'):
1476 'query', 'fragment'):
1470 v = getattr(self, a)
1477 v = getattr(self, a)
1471 if v is not None:
1478 if v is not None:
1472 attrs.append('%s: %r' % (a, v))
1479 attrs.append('%s: %r' % (a, v))
1473 return '<url %s>' % ', '.join(attrs)
1480 return '<url %s>' % ', '.join(attrs)
1474
1481
1475 def __str__(self):
1482 def __str__(self):
1476 r"""Join the URL's components back into a URL string.
1483 r"""Join the URL's components back into a URL string.
1477
1484
1478 Examples:
1485 Examples:
1479
1486
1480 >>> str(url('http://user:pw@host:80/?foo#bar'))
1487 >>> str(url('http://user:pw@host:80/?foo#bar'))
1481 'http://user:pw@host:80/?foo#bar'
1488 'http://user:pw@host:80/?foo#bar'
1482 >>> str(url('ssh://user:pw@[::1]:2200//home/joe#'))
1489 >>> str(url('ssh://user:pw@[::1]:2200//home/joe#'))
1483 'ssh://user:pw@[::1]:2200//home/joe#'
1490 'ssh://user:pw@[::1]:2200//home/joe#'
1484 >>> str(url('http://localhost:80//'))
1491 >>> str(url('http://localhost:80//'))
1485 'http://localhost:80//'
1492 'http://localhost:80//'
1486 >>> str(url('http://localhost:80/'))
1493 >>> str(url('http://localhost:80/'))
1487 'http://localhost:80/'
1494 'http://localhost:80/'
1488 >>> str(url('http://localhost:80'))
1495 >>> str(url('http://localhost:80'))
1489 'http://localhost:80/'
1496 'http://localhost:80/'
1490 >>> str(url('bundle:foo'))
1497 >>> str(url('bundle:foo'))
1491 'bundle:foo'
1498 'bundle:foo'
1492 >>> str(url('bundle://../foo'))
1499 >>> str(url('bundle://../foo'))
1493 'bundle:../foo'
1500 'bundle:../foo'
1494 >>> str(url('path'))
1501 >>> str(url('path'))
1495 'path'
1502 'path'
1496 >>> print url(r'bundle:foo\bar')
1503 >>> print url(r'bundle:foo\bar')
1497 bundle:foo\bar
1504 bundle:foo\bar
1498 """
1505 """
1499 if self._localpath:
1506 if self._localpath:
1500 s = self.path
1507 s = self.path
1501 if self.scheme == 'bundle':
1508 if self.scheme == 'bundle':
1502 s = 'bundle:' + s
1509 s = 'bundle:' + s
1503 if self.fragment:
1510 if self.fragment:
1504 s += '#' + self.fragment
1511 s += '#' + self.fragment
1505 return s
1512 return s
1506
1513
1507 s = self.scheme + ':'
1514 s = self.scheme + ':'
1508 if (self.user or self.passwd or self.host or
1515 if (self.user or self.passwd or self.host or
1509 self.scheme and not self.path):
1516 self.scheme and not self.path):
1510 s += '//'
1517 s += '//'
1511 if self.user:
1518 if self.user:
1512 s += urllib.quote(self.user, safe=self._safechars)
1519 s += urllib.quote(self.user, safe=self._safechars)
1513 if self.passwd:
1520 if self.passwd:
1514 s += ':' + urllib.quote(self.passwd, safe=self._safechars)
1521 s += ':' + urllib.quote(self.passwd, safe=self._safechars)
1515 if self.user or self.passwd:
1522 if self.user or self.passwd:
1516 s += '@'
1523 s += '@'
1517 if self.host:
1524 if self.host:
1518 if not (self.host.startswith('[') and self.host.endswith(']')):
1525 if not (self.host.startswith('[') and self.host.endswith(']')):
1519 s += urllib.quote(self.host)
1526 s += urllib.quote(self.host)
1520 else:
1527 else:
1521 s += self.host
1528 s += self.host
1522 if self.port:
1529 if self.port:
1523 s += ':' + urllib.quote(self.port)
1530 s += ':' + urllib.quote(self.port)
1524 if self.host:
1531 if self.host:
1525 s += '/'
1532 s += '/'
1526 if self.path:
1533 if self.path:
1527 s += urllib.quote(self.path, safe=self._safepchars)
1534 s += urllib.quote(self.path, safe=self._safepchars)
1528 if self.query:
1535 if self.query:
1529 s += '?' + urllib.quote(self.query, safe=self._safepchars)
1536 s += '?' + urllib.quote(self.query, safe=self._safepchars)
1530 if self.fragment is not None:
1537 if self.fragment is not None:
1531 s += '#' + urllib.quote(self.fragment, safe=self._safepchars)
1538 s += '#' + urllib.quote(self.fragment, safe=self._safepchars)
1532 return s
1539 return s
1533
1540
1534 def authinfo(self):
1541 def authinfo(self):
1535 user, passwd = self.user, self.passwd
1542 user, passwd = self.user, self.passwd
1536 try:
1543 try:
1537 self.user, self.passwd = None, None
1544 self.user, self.passwd = None, None
1538 s = str(self)
1545 s = str(self)
1539 finally:
1546 finally:
1540 self.user, self.passwd = user, passwd
1547 self.user, self.passwd = user, passwd
1541 if not self.user:
1548 if not self.user:
1542 return (s, None)
1549 return (s, None)
1543 return (s, (None, (str(self), self.host),
1550 return (s, (None, (str(self), self.host),
1544 self.user, self.passwd or ''))
1551 self.user, self.passwd or ''))
1545
1552
1546 def localpath(self):
1553 def localpath(self):
1547 if self.scheme == 'file' or self.scheme == 'bundle':
1554 if self.scheme == 'file' or self.scheme == 'bundle':
1548 path = self.path or '/'
1555 path = self.path or '/'
1549 # For Windows, we need to promote hosts containing drive
1556 # For Windows, we need to promote hosts containing drive
1550 # letters to paths with drive letters.
1557 # letters to paths with drive letters.
1551 if hasdriveletter(self._hostport):
1558 if hasdriveletter(self._hostport):
1552 path = self._hostport + '/' + self.path
1559 path = self._hostport + '/' + self.path
1553 elif self.host is not None and self.path:
1560 elif self.host is not None and self.path:
1554 path = '/' + path
1561 path = '/' + path
1555 # We also need to handle the case of file:///C:/, which
1562 # We also need to handle the case of file:///C:/, which
1556 # should return C:/, not /C:/.
1563 # should return C:/, not /C:/.
1557 elif hasdriveletter(path):
1564 elif hasdriveletter(path):
1558 # Strip leading slash from paths with drive names
1565 # Strip leading slash from paths with drive names
1559 return path[1:]
1566 return path[1:]
1560 return path
1567 return path
1561 return self._origpath
1568 return self._origpath
1562
1569
1563 def hasscheme(path):
1570 def hasscheme(path):
1564 return bool(url(path).scheme)
1571 return bool(url(path).scheme)
1565
1572
1566 def hasdriveletter(path):
1573 def hasdriveletter(path):
1567 return path[1:2] == ':' and path[0:1].isalpha()
1574 return path[1:2] == ':' and path[0:1].isalpha()
1568
1575
1569 def localpath(path):
1576 def localpath(path):
1570 return url(path, parsequery=False, parsefragment=False).localpath()
1577 return url(path, parsequery=False, parsefragment=False).localpath()
1571
1578
1572 def hidepassword(u):
1579 def hidepassword(u):
1573 '''hide user credential in a url string'''
1580 '''hide user credential in a url string'''
1574 u = url(u)
1581 u = url(u)
1575 if u.passwd:
1582 if u.passwd:
1576 u.passwd = '***'
1583 u.passwd = '***'
1577 return str(u)
1584 return str(u)
1578
1585
1579 def removeauth(u):
1586 def removeauth(u):
1580 '''remove all authentication information from a url string'''
1587 '''remove all authentication information from a url string'''
1581 u = url(u)
1588 u = url(u)
1582 u.user = u.passwd = None
1589 u.user = u.passwd = None
1583 return str(u)
1590 return str(u)
General Comments 0
You need to be logged in to leave comments. Login now