/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
1 by mbp at sourcefrog
import from baz patch-364
1
# Bazaar-NG -- distributed version control
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
2
#
1 by mbp at sourcefrog
import from baz patch-364
3
# Copyright (C) 2005 by Canonical Ltd
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
4
#
1 by mbp at sourcefrog
import from baz patch-364
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.
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
9
#
1 by mbp at sourcefrog
import from baz patch-364
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.
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
14
#
1 by mbp at sourcefrog
import from baz patch-364
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
18
1185.1.46 by Robert Collins
Aarons branch --basis patch
19
from shutil import copyfile
1185.3.28 by John Arbash Meinel
Adding knowledge about fifo/block/etc, they will be unknown/ignored.
20
from stat import (S_ISREG, S_ISDIR, S_ISLNK, ST_MODE, ST_SIZE,
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
21
                  S_ISCHR, S_ISBLK, S_ISFIFO, S_ISSOCK)
1390 by Robert Collins
pair programming worx... merge integration and weave
22
from cStringIO import StringIO
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
23
import errno
24
import os
25
import re
1236 by Martin Pool
- fix up imports
26
import sha
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
27
import sys
28
import time
29
import types
1 by mbp at sourcefrog
import from baz patch-364
30
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
31
import bzrlib
1442.1.2 by Robert Collins
create a config module - there is enough config logic to make this worthwhile, and start testing config processing.
32
from bzrlib.config import config_dir, _get_user_id
694 by Martin Pool
- weed out all remaining calls to bailout() and remove the function
33
from bzrlib.errors import BzrError
34
from bzrlib.trace import mutter
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
35
1 by mbp at sourcefrog
import from baz patch-364
36
37
def make_readonly(filename):
38
    """Make a filename read-only."""
39
    mod = os.stat(filename).st_mode
40
    mod = mod & 0777555
41
    os.chmod(filename, mod)
42
43
44
def make_writable(filename):
45
    mod = os.stat(filename).st_mode
46
    mod = mod | 0200
47
    os.chmod(filename, mod)
48
49
1077 by Martin Pool
- avoid compiling REs at module load time
50
_QUOTE_RE = None
969 by Martin Pool
- Add less-sucky is_within_any
51
52
1 by mbp at sourcefrog
import from baz patch-364
53
def quotefn(f):
779 by Martin Pool
- better quotefn for windows: use doublequotes for strings with
54
    """Return a quoted filename filename
55
56
    This previously used backslash quoting, but that works poorly on
57
    Windows."""
58
    # TODO: I'm not really sure this is the best format either.x
1077 by Martin Pool
- avoid compiling REs at module load time
59
    global _QUOTE_RE
60
    if _QUOTE_RE == None:
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
61
        _QUOTE_RE = re.compile(r'([^a-zA-Z0-9.,:/\\_~-])')
1077 by Martin Pool
- avoid compiling REs at module load time
62
        
779 by Martin Pool
- better quotefn for windows: use doublequotes for strings with
63
    if _QUOTE_RE.search(f):
64
        return '"' + f + '"'
65
    else:
66
        return f
1 by mbp at sourcefrog
import from baz patch-364
67
68
69
def file_kind(f):
70
    mode = os.lstat(f)[ST_MODE]
71
    if S_ISREG(mode):
72
        return 'file'
73
    elif S_ISDIR(mode):
74
        return 'directory'
20 by mbp at sourcefrog
don't abort on trees that happen to contain symlinks
75
    elif S_ISLNK(mode):
76
        return 'symlink'
1185.3.28 by John Arbash Meinel
Adding knowledge about fifo/block/etc, they will be unknown/ignored.
77
    elif S_ISCHR(mode):
78
        return 'chardev'
79
    elif S_ISBLK(mode):
80
        return 'block'
81
    elif S_ISFIFO(mode):
82
        return 'fifo'
83
    elif S_ISSOCK(mode):
84
        return 'socket'
1 by mbp at sourcefrog
import from baz patch-364
85
    else:
1185.3.28 by John Arbash Meinel
Adding knowledge about fifo/block/etc, they will be unknown/ignored.
86
        return 'unknown'
488 by Martin Pool
- new helper function kind_marker()
87
88
89
def kind_marker(kind):
90
    if kind == 'file':
91
        return ''
92
    elif kind == 'directory':
93
        return '/'
94
    elif kind == 'symlink':
95
        return '@'
96
    else:
97
        raise BzrError('invalid file kind %r' % kind)
1 by mbp at sourcefrog
import from baz patch-364
98
1092.2.6 by Robert Collins
symlink support updated to work
99
def lexists(f):
100
    try:
101
        if hasattr(os, 'lstat'):
102
            os.lstat(f)
103
        else:
104
            os.stat(f)
105
        return True
106
    except OSError,e:
107
        if e.errno == errno.ENOENT:
108
            return False;
109
        else:
110
            raise BzrError("lstat/stat of (%r): %r" % (f, e))
1 by mbp at sourcefrog
import from baz patch-364
111
1092.2.6 by Robert Collins
symlink support updated to work
112
def normalizepath(f):
113
    if hasattr(os.path, 'realpath'):
114
        F = os.path.realpath
115
    else:
116
        F = os.path.abspath
117
    [p,e] = os.path.split(f)
118
    if e == "" or e == "." or e == "..":
119
        return F(f)
120
    else:
121
        return os.path.join(F(p), e)
122
    
1 by mbp at sourcefrog
import from baz patch-364
123
779 by Martin Pool
- better quotefn for windows: use doublequotes for strings with
124
def backup_file(fn):
125
    """Copy a file to a backup.
126
127
    Backups are named in GNU-style, with a ~ suffix.
128
129
    If the file is already a backup, it's not copied.
130
    """
131
    if fn[-1] == '~':
132
        return
133
    bfn = fn + '~'
134
135
    inf = file(fn, 'rb')
136
    try:
137
        content = inf.read()
138
    finally:
139
        inf.close()
140
    
141
    outf = file(bfn, 'wb')
142
    try:
143
        outf.write(content)
144
    finally:
145
        outf.close()
146
1185.1.40 by Robert Collins
Merge what applied of Alexander Belchenko's win32 patch.
147
if os.name == 'nt':
148
    import shutil
149
    rename = shutil.move
150
else:
151
    rename = os.rename
779 by Martin Pool
- better quotefn for windows: use doublequotes for strings with
152
153
1 by mbp at sourcefrog
import from baz patch-364
154
def isdir(f):
155
    """True if f is an accessible directory."""
156
    try:
157
        return S_ISDIR(os.lstat(f)[ST_MODE])
158
    except OSError:
159
        return False
160
161
162
def isfile(f):
163
    """True if f is a regular file."""
164
    try:
165
        return S_ISREG(os.lstat(f)[ST_MODE])
166
    except OSError:
167
        return False
168
1092.2.6 by Robert Collins
symlink support updated to work
169
def islink(f):
170
    """True if f is a symlink."""
171
    try:
172
        return S_ISLNK(os.lstat(f)[ST_MODE])
173
    except OSError:
174
        return False
1 by mbp at sourcefrog
import from baz patch-364
175
485 by Martin Pool
- move commit code into its own module
176
def is_inside(dir, fname):
177
    """True if fname is inside dir.
969 by Martin Pool
- Add less-sucky is_within_any
178
    
179
    The parameters should typically be passed to os.path.normpath first, so
180
    that . and .. and repeated slashes are eliminated, and the separators
181
    are canonical for the platform.
182
    
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
183
    The empty string as a dir name is taken as top-of-tree and matches 
184
    everything.
185
    
1185.1.40 by Robert Collins
Merge what applied of Alexander Belchenko's win32 patch.
186
    >>> is_inside('src', os.path.join('src', 'foo.c'))
969 by Martin Pool
- Add less-sucky is_within_any
187
    True
188
    >>> is_inside('src', 'srccontrol')
189
    False
1185.1.40 by Robert Collins
Merge what applied of Alexander Belchenko's win32 patch.
190
    >>> is_inside('src', os.path.join('src', 'a', 'a', 'a', 'foo.c'))
969 by Martin Pool
- Add less-sucky is_within_any
191
    True
192
    >>> is_inside('foo.c', 'foo.c')
193
    True
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
194
    >>> is_inside('foo.c', '')
195
    False
196
    >>> is_inside('', 'foo.c')
197
    True
485 by Martin Pool
- move commit code into its own module
198
    """
969 by Martin Pool
- Add less-sucky is_within_any
199
    # XXX: Most callers of this can actually do something smarter by 
200
    # looking at the inventory
972 by Martin Pool
- less dodgy is_inside function
201
    if dir == fname:
202
        return True
203
    
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
204
    if dir == '':
205
        return True
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
206
972 by Martin Pool
- less dodgy is_inside function
207
    if dir[-1] != os.sep:
208
        dir += os.sep
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
209
972 by Martin Pool
- less dodgy is_inside function
210
    return fname.startswith(dir)
211
485 by Martin Pool
- move commit code into its own module
212
213
def is_inside_any(dir_list, fname):
214
    """True if fname is inside any of given dirs."""
215
    for dirname in dir_list:
216
        if is_inside(dirname, fname):
217
            return True
218
    else:
219
        return False
220
221
1 by mbp at sourcefrog
import from baz patch-364
222
def pumpfile(fromfile, tofile):
223
    """Copy contents of one file to another."""
224
    tofile.write(fromfile.read())
225
226
227
def sha_file(f):
228
    if hasattr(f, 'tell'):
229
        assert f.tell() == 0
230
    s = sha.new()
320 by Martin Pool
- Compute SHA-1 of files in chunks
231
    BUFSIZE = 128<<10
232
    while True:
233
        b = f.read(BUFSIZE)
234
        if not b:
235
            break
236
        s.update(b)
1 by mbp at sourcefrog
import from baz patch-364
237
    return s.hexdigest()
238
239
1235 by Martin Pool
- split sha_strings into osutils
240
241
def sha_strings(strings):
242
    """Return the sha-1 of concatenation of strings"""
243
    s = sha.new()
244
    map(s.update, strings)
245
    return s.hexdigest()
246
247
1 by mbp at sourcefrog
import from baz patch-364
248
def sha_string(f):
249
    s = sha.new()
250
    s.update(f)
251
    return s.hexdigest()
252
253
124 by mbp at sourcefrog
- check file text for past revisions is correct
254
def fingerprint_file(f):
255
    s = sha.new()
126 by mbp at sourcefrog
Use just one big read to fingerprint files
256
    b = f.read()
257
    s.update(b)
258
    size = len(b)
124 by mbp at sourcefrog
- check file text for past revisions is correct
259
    return {'size': size,
260
            'sha1': s.hexdigest()}
261
262
1 by mbp at sourcefrog
import from baz patch-364
263
def compare_files(a, b):
264
    """Returns true if equal in contents"""
74 by mbp at sourcefrog
compare_files: read in one page at a time rather than
265
    BUFSIZE = 4096
266
    while True:
267
        ai = a.read(BUFSIZE)
268
        bi = b.read(BUFSIZE)
269
        if ai != bi:
270
            return False
271
        if ai == '':
272
            return True
1 by mbp at sourcefrog
import from baz patch-364
273
274
49 by mbp at sourcefrog
fix local-time-offset calculation
275
def local_time_offset(t=None):
276
    """Return offset of local zone from GMT, either at present or at time t."""
73 by mbp at sourcefrog
fix time.localtime call for python 2.3
277
    # python2.3 localtime() can't take None
183 by mbp at sourcefrog
pychecker fixups
278
    if t == None:
73 by mbp at sourcefrog
fix time.localtime call for python 2.3
279
        t = time.time()
280
        
49 by mbp at sourcefrog
fix local-time-offset calculation
281
    if time.localtime(t).tm_isdst and time.daylight:
8 by mbp at sourcefrog
store committer's timezone in revision and show
282
        return -time.altzone
283
    else:
284
        return -time.timezone
285
286
    
287
def format_date(t, offset=0, timezone='original'):
1 by mbp at sourcefrog
import from baz patch-364
288
    ## TODO: Perhaps a global option to use either universal or local time?
289
    ## Or perhaps just let people set $TZ?
290
    assert isinstance(t, float)
291
    
8 by mbp at sourcefrog
store committer's timezone in revision and show
292
    if timezone == 'utc':
1 by mbp at sourcefrog
import from baz patch-364
293
        tt = time.gmtime(t)
294
        offset = 0
8 by mbp at sourcefrog
store committer's timezone in revision and show
295
    elif timezone == 'original':
23 by mbp at sourcefrog
format_date: handle revisions with no timezone offset
296
        if offset == None:
297
            offset = 0
16 by mbp at sourcefrog
fix inverted calculation for original timezone -> utc
298
        tt = time.gmtime(t + offset)
12 by mbp at sourcefrog
new --timezone option for bzr log
299
    elif timezone == 'local':
1 by mbp at sourcefrog
import from baz patch-364
300
        tt = time.localtime(t)
49 by mbp at sourcefrog
fix local-time-offset calculation
301
        offset = local_time_offset(t)
12 by mbp at sourcefrog
new --timezone option for bzr log
302
    else:
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
303
        raise BzrError("unsupported timezone format %r" % timezone,
304
                       ['options are "utc", "original", "local"'])
8 by mbp at sourcefrog
store committer's timezone in revision and show
305
1 by mbp at sourcefrog
import from baz patch-364
306
    return (time.strftime("%a %Y-%m-%d %H:%M:%S", tt)
8 by mbp at sourcefrog
store committer's timezone in revision and show
307
            + ' %+03d%02d' % (offset / 3600, (offset / 60) % 60))
1 by mbp at sourcefrog
import from baz patch-364
308
309
310
def compact_date(when):
311
    return time.strftime('%Y%m%d%H%M%S', time.gmtime(when))
312
    
313
314
315
def filesize(f):
316
    """Return size of given open file."""
317
    return os.fstat(f.fileno())[ST_SIZE]
318
1185.1.7 by Robert Collins
Nathaniel McCallums patch for urandom friendliness on aix.
319
# Define rand_bytes based on platform.
320
try:
321
    # Python 2.4 and later have os.urandom,
322
    # but it doesn't work on some arches
323
    os.urandom(1)
1 by mbp at sourcefrog
import from baz patch-364
324
    rand_bytes = os.urandom
1185.1.7 by Robert Collins
Nathaniel McCallums patch for urandom friendliness on aix.
325
except (NotImplementedError, AttributeError):
326
    # If python doesn't have os.urandom, or it doesn't work,
327
    # then try to first pull random data from /dev/urandom
328
    if os.path.exists("/dev/urandom"):
329
        rand_bytes = file('/dev/urandom', 'rb').read
330
    # Otherwise, use this hack as a last resort
331
    else:
332
        # not well seeded, but better than nothing
333
        def rand_bytes(n):
334
            import random
335
            s = ''
336
            while n:
337
                s += chr(random.randint(0, 255))
338
                n -= 1
339
            return s
1 by mbp at sourcefrog
import from baz patch-364
340
341
## TODO: We could later have path objects that remember their list
342
## decomposition (might be too tricksy though.)
343
344
def splitpath(p):
345
    """Turn string into list of parts.
346
347
    >>> splitpath('a')
348
    ['a']
349
    >>> splitpath('a/b')
350
    ['a', 'b']
351
    >>> splitpath('a/./b')
352
    ['a', 'b']
353
    >>> splitpath('a/.b')
354
    ['a', '.b']
355
    >>> splitpath('a/../b')
184 by mbp at sourcefrog
pychecker fixups
356
    Traceback (most recent call last):
1 by mbp at sourcefrog
import from baz patch-364
357
    ...
694 by Martin Pool
- weed out all remaining calls to bailout() and remove the function
358
    BzrError: sorry, '..' not allowed in path
1 by mbp at sourcefrog
import from baz patch-364
359
    """
360
    assert isinstance(p, types.StringTypes)
271 by Martin Pool
- Windows path fixes
361
362
    # split on either delimiter because people might use either on
363
    # Windows
364
    ps = re.split(r'[\\/]', p)
365
366
    rps = []
1 by mbp at sourcefrog
import from baz patch-364
367
    for f in ps:
368
        if f == '..':
694 by Martin Pool
- weed out all remaining calls to bailout() and remove the function
369
            raise BzrError("sorry, %r not allowed in path" % f)
271 by Martin Pool
- Windows path fixes
370
        elif (f == '.') or (f == ''):
371
            pass
372
        else:
373
            rps.append(f)
374
    return rps
1 by mbp at sourcefrog
import from baz patch-364
375
376
def joinpath(p):
377
    assert isinstance(p, list)
378
    for f in p:
183 by mbp at sourcefrog
pychecker fixups
379
        if (f == '..') or (f == None) or (f == ''):
694 by Martin Pool
- weed out all remaining calls to bailout() and remove the function
380
            raise BzrError("sorry, %r not allowed in path" % f)
271 by Martin Pool
- Windows path fixes
381
    return os.path.join(*p)
1 by mbp at sourcefrog
import from baz patch-364
382
383
384
def appendpath(p1, p2):
385
    if p1 == '':
386
        return p2
387
    else:
271 by Martin Pool
- Windows path fixes
388
        return os.path.join(p1, p2)
1 by mbp at sourcefrog
import from baz patch-364
389
    
390
763 by Martin Pool
- Patch from Torsten Marek to take commit messages through an
391
def _read_config_value(name):
392
    """Read a config value from the file ~/.bzr.conf/<name>
393
    Return None if the file does not exist"""
394
    try:
395
        f = file(os.path.join(config_dir(), name), "r")
396
        return f.read().decode(bzrlib.user_encoding).rstrip("\r\n")
397
    except IOError, e:
398
        if e.errno == errno.ENOENT:
399
            return None
400
        raise
401
1185.1.46 by Robert Collins
Aarons branch --basis patch
402
1231 by Martin Pool
- more progress on fetch on top of weaves
403
def split_lines(s):
404
    """Split s into lines, but without removing the newline characters."""
405
    return StringIO(s).readlines()
1391 by Robert Collins
merge from integration
406
407
1185.10.4 by Aaron Bentley
Disabled hardlinks on cygwin, mac OS
408
def hardlinks_good():
1185.10.5 by Aaron Bentley
Fixed hardlinks_good test
409
    return sys.platform not in ('win32', 'cygwin', 'darwin')
1185.10.4 by Aaron Bentley
Disabled hardlinks on cygwin, mac OS
410
1185.1.46 by Robert Collins
Aarons branch --basis patch
411
1185.10.3 by Aaron Bentley
Made copy_multi_immutable create hardlinks opportunistically
412
def link_or_copy(src, dest):
413
    """Hardlink a file, or copy it if it can't be hardlinked."""
1185.10.4 by Aaron Bentley
Disabled hardlinks on cygwin, mac OS
414
    if not hardlinks_good():
1185.10.3 by Aaron Bentley
Made copy_multi_immutable create hardlinks opportunistically
415
        copyfile(src, dest)
416
        return
417
    try:
418
        os.link(src, dest)
419
    except (OSError, IOError), e:
420
        if e.errno != errno.EXDEV:
421
            raise
422
        copyfile(src, dest)
1399.1.4 by Robert Collins
move diff and symlink conditionals into inventory.py from diff.py
423
424
425
def has_symlinks():
426
    if hasattr(os, 'symlink'):
427
        return True
428
    else:
429
        return False