1
# Bazaar-NG -- distributed version control
3
# Copyright (C) 2005 by Canonical Ltd
5
# This program is free software; you can redistribute it and/or modify
6
# it under the terms of the GNU General Public License as published by
7
# the Free Software Foundation; either version 2 of the License, or
8
# (at your option) any later version.
10
# This program is distributed in the hope that it will be useful,
11
# but WITHOUT ANY WARRANTY; without even the implied warranty of
12
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
# GNU General Public License for more details.
15
# You should have received a copy of the GNU General Public License
16
# along with this program; if not, write to the Free Software
17
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19
import os, types, re, time, errno, sys
20
from stat import S_ISREG, S_ISDIR, S_ISLNK, ST_MODE, ST_SIZE
22
from errors import bailout, BzrError
23
from trace import mutter
26
def make_readonly(filename):
27
"""Make a filename read-only."""
28
# TODO: probably needs to be fixed for windows
29
mod = os.stat(filename).st_mode
31
os.chmod(filename, mod)
34
def make_writable(filename):
35
mod = os.stat(filename).st_mode
37
os.chmod(filename, mod)
40
_QUOTE_RE = re.compile(r'([^a-zA-Z0-9.,:/_~-])')
42
"""Return shell-quoted filename"""
43
## We could be a bit more terse by using double-quotes etc
44
f = _QUOTE_RE.sub(r'\\\1', f)
51
mode = os.lstat(f)[ST_MODE]
59
raise BzrError("can't handle file kind with mode %o of %r" % (mode, f))
64
"""True if f is an accessible directory."""
66
return S_ISDIR(os.lstat(f)[ST_MODE])
73
"""True if f is a regular file."""
75
return S_ISREG(os.lstat(f)[ST_MODE])
80
def is_inside(dir, fname):
81
"""True if fname is inside dir.
83
return os.path.commonprefix([dir, fname]) == dir
86
def is_inside_any(dir_list, fname):
87
"""True if fname is inside any of given dirs."""
88
# quick scan for perfect match
92
for dirname in dir_list:
93
if is_inside(dirname, fname):
99
def pumpfile(fromfile, tofile):
100
"""Copy contents of one file to another."""
101
tofile.write(fromfile.read())
105
"""Return a new UUID"""
107
return file('/proc/sys/kernel/random/uuid').readline().rstrip('\n')
109
return chomp(os.popen('uuidgen').readline())
114
if hasattr(f, 'tell'):
134
def fingerprint_file(f):
140
return {'size': size,
141
'sha1': s.hexdigest()}
145
"""Return per-user configuration directory.
147
By default this is ~/.bzr.conf/
149
TODO: Global option --config-dir to override this.
151
return os.path.expanduser("~/.bzr.conf")
155
"""Calculate automatic user identification.
157
Returns (realname, email).
159
Only used when none is set in the environment or the id file.
161
This previously used the FQDN as the default domain, but that can
162
be very slow on machines where DNS is broken. So now we simply
167
# XXX: Any good way to get real user name on win32?
172
w = pwd.getpwuid(uid)
173
gecos = w.pw_gecos.decode(bzrlib.user_encoding)
174
username = w.pw_name.decode(bzrlib.user_encoding)
175
comma = gecos.find(',')
179
realname = gecos[:comma]
185
realname = username = getpass.getuser().decode(bzrlib.user_encoding)
187
return realname, (username + '@' + socket.gethostname())
191
"""Return the full user id from a file or environment variable.
193
TODO: Allow taking this from a file in the branch directory too
194
for per-branch ids."""
195
v = os.environ.get('BZREMAIL')
197
return v.decode(bzrlib.user_encoding)
200
return (open(os.path.join(config_dir(), "email"))
202
.decode(bzrlib.user_encoding)
205
if e.errno != errno.ENOENT:
208
v = os.environ.get('EMAIL')
210
return v.decode(bzrlib.user_encoding)
216
"""Return email-style username.
218
Something similar to 'Martin Pool <mbp@sourcefrog.net>'
220
TODO: Check it's reasonably well-formed.
226
name, email = _auto_user_id()
228
return '%s <%s>' % (name, email)
233
_EMAIL_RE = re.compile(r'[\w+.-]+@[\w+.-]+')
235
"""Return just the email component of a username."""
238
m = _EMAIL_RE.search(e)
240
bailout("%r doesn't seem to contain a reasonable email address" % e)
243
return _auto_user_id()[1]
247
def compare_files(a, b):
248
"""Returns true if equal in contents"""
260
def local_time_offset(t=None):
261
"""Return offset of local zone from GMT, either at present or at time t."""
262
# python2.3 localtime() can't take None
266
if time.localtime(t).tm_isdst and time.daylight:
269
return -time.timezone
272
def format_date(t, offset=0, timezone='original'):
273
## TODO: Perhaps a global option to use either universal or local time?
274
## Or perhaps just let people set $TZ?
275
assert isinstance(t, float)
277
if timezone == 'utc':
280
elif timezone == 'original':
283
tt = time.gmtime(t + offset)
284
elif timezone == 'local':
285
tt = time.localtime(t)
286
offset = local_time_offset(t)
288
bailout("unsupported timezone format %r",
289
['options are "utc", "original", "local"'])
291
return (time.strftime("%a %Y-%m-%d %H:%M:%S", tt)
292
+ ' %+03d%02d' % (offset / 3600, (offset / 60) % 60))
295
def compact_date(when):
296
return time.strftime('%Y%m%d%H%M%S', time.gmtime(when))
301
"""Return size of given open file."""
302
return os.fstat(f.fileno())[ST_SIZE]
305
if hasattr(os, 'urandom'): # python 2.4 and later
306
rand_bytes = os.urandom
307
elif sys.platform == 'linux2':
308
rand_bytes = file('/dev/urandom', 'rb').read
310
# not well seeded, but better than nothing
315
s += chr(random.randint(0, 255))
320
## TODO: We could later have path objects that remember their list
321
## decomposition (might be too tricksy though.)
324
"""Turn string into list of parts.
330
>>> splitpath('a/./b')
332
>>> splitpath('a/.b')
334
>>> splitpath('a/../b')
335
Traceback (most recent call last):
337
BzrError: ("sorry, '..' not allowed in path", [])
339
assert isinstance(p, types.StringTypes)
341
# split on either delimiter because people might use either on
343
ps = re.split(r'[\\/]', p)
348
bailout("sorry, %r not allowed in path" % f)
349
elif (f == '.') or (f == ''):
356
assert isinstance(p, list)
358
if (f == '..') or (f == None) or (f == ''):
359
bailout("sorry, %r not allowed in path" % f)
360
return os.path.join(*p)
363
def appendpath(p1, p2):
367
return os.path.join(p1, p2)
370
def extern_command(cmd, ignore_errors = False):
371
mutter('external command: %s' % `cmd`)
373
if not ignore_errors:
374
bailout('command failed')