/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.16.38 by Martin Pool
- move contains_whitespace and contains_linebreaks to osutils
27
import string
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
28
import sys
29
import time
30
import types
1 by mbp at sourcefrog
import from baz patch-364
31
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
32
import bzrlib
1457.1.2 by Robert Collins
move branch._relpath into osutils as relpath
33
from bzrlib.errors import BzrError, NotBranchError
694 by Martin Pool
- weed out all remaining calls to bailout() and remove the function
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)
1185.16.70 by Martin Pool
- improved handling of non-ascii branch names and test
122
123
if os.name == "posix":
124
    # In Python 2.4.2 and older, os.path.abspath and os.path.realpath
125
    # choke on a Unicode string containing a relative path if
126
    # os.getcwd() returns a non-sys.getdefaultencoding()-encoded
127
    # string.
128
    _fs_enc = sys.getfilesystemencoding()
129
    def abspath(path):
130
        return os.path.abspath(path.encode(_fs_enc)).decode(_fs_enc)
131
    def realpath(path):
132
        return os.path.realpath(path.encode(_fs_enc)).decode(_fs_enc)
133
else:
134
    # We need to use the Unicode-aware os.path.abspath and
135
    # os.path.realpath on Windows systems.
136
    abspath = os.path.abspath
137
    realpath = os.path.realpath
1 by mbp at sourcefrog
import from baz patch-364
138
779 by Martin Pool
- better quotefn for windows: use doublequotes for strings with
139
def backup_file(fn):
140
    """Copy a file to a backup.
141
142
    Backups are named in GNU-style, with a ~ suffix.
143
144
    If the file is already a backup, it's not copied.
145
    """
146
    if fn[-1] == '~':
147
        return
148
    bfn = fn + '~'
149
1448 by Robert Collins
revert symlinks correctly
150
    if has_symlinks() and os.path.islink(fn):
151
        target = os.readlink(fn)
152
        os.symlink(target, bfn)
153
        return
779 by Martin Pool
- better quotefn for windows: use doublequotes for strings with
154
    inf = file(fn, 'rb')
155
    try:
156
        content = inf.read()
157
    finally:
158
        inf.close()
159
    
160
    outf = file(bfn, 'wb')
161
    try:
162
        outf.write(content)
163
    finally:
164
        outf.close()
165
1185.1.40 by Robert Collins
Merge what applied of Alexander Belchenko's win32 patch.
166
if os.name == 'nt':
167
    import shutil
168
    rename = shutil.move
169
else:
170
    rename = os.rename
779 by Martin Pool
- better quotefn for windows: use doublequotes for strings with
171
172
1 by mbp at sourcefrog
import from baz patch-364
173
def isdir(f):
174
    """True if f is an accessible directory."""
175
    try:
176
        return S_ISDIR(os.lstat(f)[ST_MODE])
177
    except OSError:
178
        return False
179
180
181
def isfile(f):
182
    """True if f is a regular file."""
183
    try:
184
        return S_ISREG(os.lstat(f)[ST_MODE])
185
    except OSError:
186
        return False
187
1092.2.6 by Robert Collins
symlink support updated to work
188
def islink(f):
189
    """True if f is a symlink."""
190
    try:
191
        return S_ISLNK(os.lstat(f)[ST_MODE])
192
    except OSError:
193
        return False
1 by mbp at sourcefrog
import from baz patch-364
194
485 by Martin Pool
- move commit code into its own module
195
def is_inside(dir, fname):
196
    """True if fname is inside dir.
969 by Martin Pool
- Add less-sucky is_within_any
197
    
198
    The parameters should typically be passed to os.path.normpath first, so
199
    that . and .. and repeated slashes are eliminated, and the separators
200
    are canonical for the platform.
201
    
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
202
    The empty string as a dir name is taken as top-of-tree and matches 
203
    everything.
204
    
1185.1.40 by Robert Collins
Merge what applied of Alexander Belchenko's win32 patch.
205
    >>> is_inside('src', os.path.join('src', 'foo.c'))
969 by Martin Pool
- Add less-sucky is_within_any
206
    True
207
    >>> is_inside('src', 'srccontrol')
208
    False
1185.1.40 by Robert Collins
Merge what applied of Alexander Belchenko's win32 patch.
209
    >>> is_inside('src', os.path.join('src', 'a', 'a', 'a', 'foo.c'))
969 by Martin Pool
- Add less-sucky is_within_any
210
    True
211
    >>> is_inside('foo.c', 'foo.c')
212
    True
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
213
    >>> is_inside('foo.c', '')
214
    False
215
    >>> is_inside('', 'foo.c')
216
    True
485 by Martin Pool
- move commit code into its own module
217
    """
969 by Martin Pool
- Add less-sucky is_within_any
218
    # XXX: Most callers of this can actually do something smarter by 
219
    # looking at the inventory
972 by Martin Pool
- less dodgy is_inside function
220
    if dir == fname:
221
        return True
222
    
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
223
    if dir == '':
224
        return True
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
225
972 by Martin Pool
- less dodgy is_inside function
226
    if dir[-1] != os.sep:
227
        dir += os.sep
1185.1.41 by Robert Collins
massive patch from Alexander Belchenko - many PEP8 fixes, removes unused function uuid
228
972 by Martin Pool
- less dodgy is_inside function
229
    return fname.startswith(dir)
230
485 by Martin Pool
- move commit code into its own module
231
232
def is_inside_any(dir_list, fname):
233
    """True if fname is inside any of given dirs."""
234
    for dirname in dir_list:
235
        if is_inside(dirname, fname):
236
            return True
237
    else:
238
        return False
239
240
1 by mbp at sourcefrog
import from baz patch-364
241
def pumpfile(fromfile, tofile):
242
    """Copy contents of one file to another."""
1185.49.12 by John Arbash Meinel
Changed pumpfile to work on blocks, rather than reading the entire file at once.
243
    BUFSIZE = 32768
244
    while True:
245
        b = fromfile.read(BUFSIZE)
246
        if not b:
247
            break
1185.49.13 by John Arbash Meinel
Removed delayed setup, since it broke some tests. Fixed other small bugs. All tests pass.
248
        tofile.write(b)
1 by mbp at sourcefrog
import from baz patch-364
249
250
251
def sha_file(f):
252
    if hasattr(f, 'tell'):
253
        assert f.tell() == 0
254
    s = sha.new()
320 by Martin Pool
- Compute SHA-1 of files in chunks
255
    BUFSIZE = 128<<10
256
    while True:
257
        b = f.read(BUFSIZE)
258
        if not b:
259
            break
260
        s.update(b)
1 by mbp at sourcefrog
import from baz patch-364
261
    return s.hexdigest()
262
263
1235 by Martin Pool
- split sha_strings into osutils
264
265
def sha_strings(strings):
266
    """Return the sha-1 of concatenation of strings"""
267
    s = sha.new()
268
    map(s.update, strings)
269
    return s.hexdigest()
270
271
1 by mbp at sourcefrog
import from baz patch-364
272
def sha_string(f):
273
    s = sha.new()
274
    s.update(f)
275
    return s.hexdigest()
276
277
124 by mbp at sourcefrog
- check file text for past revisions is correct
278
def fingerprint_file(f):
279
    s = sha.new()
126 by mbp at sourcefrog
Use just one big read to fingerprint files
280
    b = f.read()
281
    s.update(b)
282
    size = len(b)
124 by mbp at sourcefrog
- check file text for past revisions is correct
283
    return {'size': size,
284
            'sha1': s.hexdigest()}
285
286
1 by mbp at sourcefrog
import from baz patch-364
287
def compare_files(a, b):
288
    """Returns true if equal in contents"""
74 by mbp at sourcefrog
compare_files: read in one page at a time rather than
289
    BUFSIZE = 4096
290
    while True:
291
        ai = a.read(BUFSIZE)
292
        bi = b.read(BUFSIZE)
293
        if ai != bi:
294
            return False
295
        if ai == '':
296
            return True
1 by mbp at sourcefrog
import from baz patch-364
297
298
49 by mbp at sourcefrog
fix local-time-offset calculation
299
def local_time_offset(t=None):
300
    """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
301
    # python2.3 localtime() can't take None
183 by mbp at sourcefrog
pychecker fixups
302
    if t == None:
73 by mbp at sourcefrog
fix time.localtime call for python 2.3
303
        t = time.time()
304
        
49 by mbp at sourcefrog
fix local-time-offset calculation
305
    if time.localtime(t).tm_isdst and time.daylight:
8 by mbp at sourcefrog
store committer's timezone in revision and show
306
        return -time.altzone
307
    else:
308
        return -time.timezone
309
310
    
1185.12.24 by Aaron Bentley
Made format_date more flexible
311
def format_date(t, offset=0, timezone='original', date_fmt=None, 
312
                show_offset=True):
1 by mbp at sourcefrog
import from baz patch-364
313
    ## TODO: Perhaps a global option to use either universal or local time?
314
    ## Or perhaps just let people set $TZ?
315
    assert isinstance(t, float)
316
    
8 by mbp at sourcefrog
store committer's timezone in revision and show
317
    if timezone == 'utc':
1 by mbp at sourcefrog
import from baz patch-364
318
        tt = time.gmtime(t)
319
        offset = 0
8 by mbp at sourcefrog
store committer's timezone in revision and show
320
    elif timezone == 'original':
23 by mbp at sourcefrog
format_date: handle revisions with no timezone offset
321
        if offset == None:
322
            offset = 0
16 by mbp at sourcefrog
fix inverted calculation for original timezone -> utc
323
        tt = time.gmtime(t + offset)
12 by mbp at sourcefrog
new --timezone option for bzr log
324
    elif timezone == 'local':
1 by mbp at sourcefrog
import from baz patch-364
325
        tt = time.localtime(t)
49 by mbp at sourcefrog
fix local-time-offset calculation
326
        offset = local_time_offset(t)
12 by mbp at sourcefrog
new --timezone option for bzr log
327
    else:
974.1.26 by aaron.bentley at utoronto
merged mbp@sourcefrog.net-20050817233101-0939da1cf91f2472
328
        raise BzrError("unsupported timezone format %r" % timezone,
329
                       ['options are "utc", "original", "local"'])
1185.12.24 by Aaron Bentley
Made format_date more flexible
330
    if date_fmt is None:
331
        date_fmt = "%a %Y-%m-%d %H:%M:%S"
332
    if show_offset:
333
        offset_str = ' %+03d%02d' % (offset / 3600, (offset / 60) % 60)
334
    else:
335
        offset_str = ''
336
    return (time.strftime(date_fmt, tt) +  offset_str)
1 by mbp at sourcefrog
import from baz patch-364
337
338
339
def compact_date(when):
340
    return time.strftime('%Y%m%d%H%M%S', time.gmtime(when))
341
    
342
343
344
def filesize(f):
345
    """Return size of given open file."""
346
    return os.fstat(f.fileno())[ST_SIZE]
347
1185.1.7 by Robert Collins
Nathaniel McCallums patch for urandom friendliness on aix.
348
# Define rand_bytes based on platform.
349
try:
350
    # Python 2.4 and later have os.urandom,
351
    # but it doesn't work on some arches
352
    os.urandom(1)
1 by mbp at sourcefrog
import from baz patch-364
353
    rand_bytes = os.urandom
1185.1.7 by Robert Collins
Nathaniel McCallums patch for urandom friendliness on aix.
354
except (NotImplementedError, AttributeError):
355
    # If python doesn't have os.urandom, or it doesn't work,
356
    # then try to first pull random data from /dev/urandom
357
    if os.path.exists("/dev/urandom"):
358
        rand_bytes = file('/dev/urandom', 'rb').read
359
    # Otherwise, use this hack as a last resort
360
    else:
361
        # not well seeded, but better than nothing
362
        def rand_bytes(n):
363
            import random
364
            s = ''
365
            while n:
366
                s += chr(random.randint(0, 255))
367
                n -= 1
368
            return s
1 by mbp at sourcefrog
import from baz patch-364
369
370
## TODO: We could later have path objects that remember their list
371
## decomposition (might be too tricksy though.)
372
373
def splitpath(p):
374
    """Turn string into list of parts.
375
376
    >>> splitpath('a')
377
    ['a']
378
    >>> splitpath('a/b')
379
    ['a', 'b']
380
    >>> splitpath('a/./b')
381
    ['a', 'b']
382
    >>> splitpath('a/.b')
383
    ['a', '.b']
384
    >>> splitpath('a/../b')
184 by mbp at sourcefrog
pychecker fixups
385
    Traceback (most recent call last):
1 by mbp at sourcefrog
import from baz patch-364
386
    ...
694 by Martin Pool
- weed out all remaining calls to bailout() and remove the function
387
    BzrError: sorry, '..' not allowed in path
1 by mbp at sourcefrog
import from baz patch-364
388
    """
389
    assert isinstance(p, types.StringTypes)
271 by Martin Pool
- Windows path fixes
390
391
    # split on either delimiter because people might use either on
392
    # Windows
393
    ps = re.split(r'[\\/]', p)
394
395
    rps = []
1 by mbp at sourcefrog
import from baz patch-364
396
    for f in ps:
397
        if f == '..':
694 by Martin Pool
- weed out all remaining calls to bailout() and remove the function
398
            raise BzrError("sorry, %r not allowed in path" % f)
271 by Martin Pool
- Windows path fixes
399
        elif (f == '.') or (f == ''):
400
            pass
401
        else:
402
            rps.append(f)
403
    return rps
1 by mbp at sourcefrog
import from baz patch-364
404
405
def joinpath(p):
406
    assert isinstance(p, list)
407
    for f in p:
183 by mbp at sourcefrog
pychecker fixups
408
        if (f == '..') or (f == None) or (f == ''):
694 by Martin Pool
- weed out all remaining calls to bailout() and remove the function
409
            raise BzrError("sorry, %r not allowed in path" % f)
271 by Martin Pool
- Windows path fixes
410
    return os.path.join(*p)
1 by mbp at sourcefrog
import from baz patch-364
411
412
413
def appendpath(p1, p2):
414
    if p1 == '':
415
        return p2
416
    else:
271 by Martin Pool
- Windows path fixes
417
        return os.path.join(p1, p2)
1 by mbp at sourcefrog
import from baz patch-364
418
    
419
1231 by Martin Pool
- more progress on fetch on top of weaves
420
def split_lines(s):
421
    """Split s into lines, but without removing the newline characters."""
422
    return StringIO(s).readlines()
1391 by Robert Collins
merge from integration
423
424
1185.10.4 by Aaron Bentley
Disabled hardlinks on cygwin, mac OS
425
def hardlinks_good():
1185.10.5 by Aaron Bentley
Fixed hardlinks_good test
426
    return sys.platform not in ('win32', 'cygwin', 'darwin')
1185.10.4 by Aaron Bentley
Disabled hardlinks on cygwin, mac OS
427
1185.1.46 by Robert Collins
Aarons branch --basis patch
428
1185.10.3 by Aaron Bentley
Made copy_multi_immutable create hardlinks opportunistically
429
def link_or_copy(src, dest):
430
    """Hardlink a file, or copy it if it can't be hardlinked."""
1185.10.4 by Aaron Bentley
Disabled hardlinks on cygwin, mac OS
431
    if not hardlinks_good():
1185.10.3 by Aaron Bentley
Made copy_multi_immutable create hardlinks opportunistically
432
        copyfile(src, dest)
433
        return
434
    try:
435
        os.link(src, dest)
436
    except (OSError, IOError), e:
437
        if e.errno != errno.EXDEV:
438
            raise
439
        copyfile(src, dest)
1399.1.4 by Robert Collins
move diff and symlink conditionals into inventory.py from diff.py
440
441
442
def has_symlinks():
443
    if hasattr(os, 'symlink'):
444
        return True
445
    else:
446
        return False
1185.16.38 by Martin Pool
- move contains_whitespace and contains_linebreaks to osutils
447
        
448
449
def contains_whitespace(s):
450
    """True if there are any whitespace characters in s."""
451
    for ch in string.whitespace:
452
        if ch in s:
453
            return True
454
    else:
455
        return False
456
457
458
def contains_linebreaks(s):
459
    """True if there is any vertical whitespace in s."""
460
    for ch in '\f\n\r':
461
        if ch in s:
462
            return True
463
    else:
464
        return False
1457.1.2 by Robert Collins
move branch._relpath into osutils as relpath
465
466
467
def relpath(base, path):
468
    """Return path relative to base, or raise exception.
469
470
    The path may be either an absolute path or a path relative to the
471
    current working directory.
472
473
    os.path.commonprefix (python2.4) has a bad bug that it works just
474
    on string prefixes, assuming that '/u' is a prefix of '/u2'.  This
475
    avoids that problem."""
1185.16.70 by Martin Pool
- improved handling of non-ascii branch names and test
476
    rp = abspath(path)
1457.1.2 by Robert Collins
move branch._relpath into osutils as relpath
477
478
    s = []
479
    head = rp
480
    while len(head) >= len(base):
481
        if head == base:
482
            break
483
        head, tail = os.path.split(head)
484
        if tail:
485
            s.insert(0, tail)
486
    else:
487
        # XXX This should raise a NotChildPath exception, as its not tied
488
        # to branch anymore.
489
        raise NotBranchError("path %r is not within branch %r" % (rp, base))
490
491
    return os.sep.join(s)
1185.33.60 by Martin Pool
Use full terminal width for verbose test output.
492
493
494
495
def terminal_width():
496
    """Return estimated terminal width."""
497
498
    # TODO: Do something smart on Windows?
499
500
    # TODO: Is there anything that gets a better update when the window
501
    # is resized while the program is running? We could use the Python termcap
502
    # library.
503
    try:
504
        return int(os.environ['COLUMNS'])
505
    except (IndexError, KeyError, ValueError):
506
        return 80