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 bzrlib.errors import BzrError
 
 
23
from bzrlib.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))
 
 
62
def kind_marker(kind):
 
 
65
    elif kind == 'directory':
 
 
67
    elif kind == 'symlink':
 
 
70
        raise BzrError('invalid file kind %r' % kind)
 
 
75
    """True if f is an accessible directory."""
 
 
77
        return S_ISDIR(os.lstat(f)[ST_MODE])
 
 
84
    """True if f is a regular file."""
 
 
86
        return S_ISREG(os.lstat(f)[ST_MODE])
 
 
91
def is_inside(dir, fname):
 
 
92
    """True if fname is inside dir.
 
 
94
    return os.path.commonprefix([dir, fname]) == dir
 
 
97
def is_inside_any(dir_list, fname):
 
 
98
    """True if fname is inside any of given dirs."""
 
 
99
    # quick scan for perfect match
 
 
100
    if fname in dir_list:
 
 
103
    for dirname in dir_list:
 
 
104
        if is_inside(dirname, fname):
 
 
110
def pumpfile(fromfile, tofile):
 
 
111
    """Copy contents of one file to another."""
 
 
112
    tofile.write(fromfile.read())
 
 
116
    """Return a new UUID"""
 
 
118
        return file('/proc/sys/kernel/random/uuid').readline().rstrip('\n')
 
 
120
        return chomp(os.popen('uuidgen').readline())
 
 
125
    if hasattr(f, 'tell'):
 
 
145
def fingerprint_file(f):
 
 
151
    return {'size': size,
 
 
152
            'sha1': s.hexdigest()}
 
 
156
    """Return per-user configuration directory.
 
 
158
    By default this is ~/.bzr.conf/
 
 
160
    TODO: Global option --config-dir to override this.
 
 
162
    return os.path.expanduser("~/.bzr.conf")
 
 
166
    """Calculate automatic user identification.
 
 
168
    Returns (realname, email).
 
 
170
    Only used when none is set in the environment or the id file.
 
 
172
    This previously used the FQDN as the default domain, but that can
 
 
173
    be very slow on machines where DNS is broken.  So now we simply
 
 
178
    # XXX: Any good way to get real user name on win32?
 
 
183
        w = pwd.getpwuid(uid)
 
 
184
        gecos = w.pw_gecos.decode(bzrlib.user_encoding)
 
 
185
        username = w.pw_name.decode(bzrlib.user_encoding)
 
 
186
        comma = gecos.find(',')
 
 
190
            realname = gecos[:comma]
 
 
196
        realname = username = getpass.getuser().decode(bzrlib.user_encoding)
 
 
198
    return realname, (username + '@' + socket.gethostname())
 
 
202
    """Return the full user id from a file or environment variable.
 
 
204
    TODO: Allow taking this from a file in the branch directory too
 
 
205
    for per-branch ids."""
 
 
206
    v = os.environ.get('BZREMAIL')
 
 
208
        return v.decode(bzrlib.user_encoding)
 
 
211
        return (open(os.path.join(config_dir(), "email"))
 
 
213
                .decode(bzrlib.user_encoding)
 
 
216
        if e.errno != errno.ENOENT:
 
 
219
    v = os.environ.get('EMAIL')
 
 
221
        return v.decode(bzrlib.user_encoding)
 
 
227
    """Return email-style username.
 
 
229
    Something similar to 'Martin Pool <mbp@sourcefrog.net>'
 
 
231
    TODO: Check it's reasonably well-formed.
 
 
237
    name, email = _auto_user_id()
 
 
239
        return '%s <%s>' % (name, email)
 
 
244
_EMAIL_RE = re.compile(r'[\w+.-]+@[\w+.-]+')
 
 
246
    """Return just the email component of a username."""
 
 
249
        m = _EMAIL_RE.search(e)
 
 
251
            raise BzrError("%r doesn't seem to contain a reasonable email address" % e)
 
 
254
    return _auto_user_id()[1]
 
 
258
def compare_files(a, b):
 
 
259
    """Returns true if equal in contents"""
 
 
271
def local_time_offset(t=None):
 
 
272
    """Return offset of local zone from GMT, either at present or at time t."""
 
 
273
    # python2.3 localtime() can't take None
 
 
277
    if time.localtime(t).tm_isdst and time.daylight:
 
 
280
        return -time.timezone
 
 
283
def format_date(t, offset=0, timezone='original'):
 
 
284
    ## TODO: Perhaps a global option to use either universal or local time?
 
 
285
    ## Or perhaps just let people set $TZ?
 
 
286
    assert isinstance(t, float)
 
 
288
    if timezone == 'utc':
 
 
291
    elif timezone == 'original':
 
 
294
        tt = time.gmtime(t + offset)
 
 
295
    elif timezone == 'local':
 
 
296
        tt = time.localtime(t)
 
 
297
        offset = local_time_offset(t)
 
 
299
        raise BzrError("unsupported timezone format %r",
 
 
300
                ['options are "utc", "original", "local"'])
 
 
302
    return (time.strftime("%a %Y-%m-%d %H:%M:%S", tt)
 
 
303
            + ' %+03d%02d' % (offset / 3600, (offset / 60) % 60))
 
 
306
def compact_date(when):
 
 
307
    return time.strftime('%Y%m%d%H%M%S', time.gmtime(when))
 
 
312
    """Return size of given open file."""
 
 
313
    return os.fstat(f.fileno())[ST_SIZE]
 
 
316
if hasattr(os, 'urandom'): # python 2.4 and later
 
 
317
    rand_bytes = os.urandom
 
 
318
elif sys.platform == 'linux2':
 
 
319
    rand_bytes = file('/dev/urandom', 'rb').read
 
 
321
    # not well seeded, but better than nothing
 
 
326
            s += chr(random.randint(0, 255))
 
 
331
## TODO: We could later have path objects that remember their list
 
 
332
## decomposition (might be too tricksy though.)
 
 
335
    """Turn string into list of parts.
 
 
341
    >>> splitpath('a/./b')
 
 
343
    >>> splitpath('a/.b')
 
 
345
    >>> splitpath('a/../b')
 
 
346
    Traceback (most recent call last):
 
 
348
    BzrError: sorry, '..' not allowed in path
 
 
350
    assert isinstance(p, types.StringTypes)
 
 
352
    # split on either delimiter because people might use either on
 
 
354
    ps = re.split(r'[\\/]', p)
 
 
359
            raise BzrError("sorry, %r not allowed in path" % f)
 
 
360
        elif (f == '.') or (f == ''):
 
 
367
    assert isinstance(p, list)
 
 
369
        if (f == '..') or (f == None) or (f == ''):
 
 
370
            raise BzrError("sorry, %r not allowed in path" % f)
 
 
371
    return os.path.join(*p)
 
 
374
def appendpath(p1, p2):
 
 
378
        return os.path.join(p1, p2)
 
 
381
def extern_command(cmd, ignore_errors = False):
 
 
382
    mutter('external command: %s' % `cmd`)
 
 
384
        if not ignore_errors:
 
 
385
            raise BzrError('command failed')
 
 
388
def _read_config_value(name):
 
 
389
    """Read a config value from the file ~/.bzr.conf/<name>
 
 
390
    Return None if the file does not exist"""
 
 
392
        f = file(os.path.join(config_dir(), name), "r")
 
 
393
        return f.read().decode(bzrlib.user_encoding).rstrip("\r\n")
 
 
395
        if e.errno == errno.ENOENT:
 
 
401
    """Return a sequence of possible editor binaries for the current platform"""
 
 
402
    e = _read_config_value("editor")
 
 
406
    if os.name == "windows":
 
 
408
    elif os.name == "posix":
 
 
410
            yield os.environ["EDITOR"]
 
 
415
def _run_editor(filename):
 
 
416
    """Try to execute an editor to edit the commit message. Returns True on success,
 
 
418
    for e in _get_editor():
 
 
419
        x = os.spawnvp(os.P_WAIT, e, (e, filename))
 
 
426
    raise BzrError("Could not start any editor. Please specify $EDITOR or use ~/.bzr.conf/editor")
 
 
430
def get_text_message(infotext, ignoreline = "default"):
 
 
433
    if ignoreline == "default":
 
 
434
        ignoreline = "-- This line and the following will be ignored --"
 
 
437
        tmp_fileno, msgfilename = tempfile.mkstemp()
 
 
438
        msgfile = os.close(tmp_fileno)
 
 
439
        if infotext is not None and infotext != "":
 
 
441
            msgfile = file(msgfilename, "w")
 
 
442
            msgfile.write("\n\n%s\n\n%s" % (ignoreline, infotext))
 
 
447
        if not _run_editor(msgfilename):
 
 
452
        lastline, nlines = 0, 0
 
 
453
        for line in file(msgfilename, "r"):
 
 
454
            stripped_line = line.strip()
 
 
455
            # strip empty line before the log message starts
 
 
457
                if stripped_line != "":
 
 
461
            # check for the ignore line only if there
 
 
462
            # is additional information at the end
 
 
463
            if hasinfo and stripped_line == ignoreline:
 
 
466
            # keep track of the last line that had some content
 
 
467
            if stripped_line != "":
 
 
473
        # delete empty lines at the end
 
 
475
        # add a newline at the end, if needed
 
 
476
        if not msg[-1].endswith("\n"):
 
 
477
            return "%s%s" % ("".join(msg), "\n")
 
 
481
        # delete the msg file in any case
 
 
482
        try: os.unlink(msgfilename)