23
22
# import system imports here
28
30
#import bzrlib specific imports here
29
31
from bzrlib import (
32
34
import bzrlib.branch
33
from bzrlib.tests import TestCase, TestSkipped
35
from bzrlib.tests import (
36
41
# Files which are listed here will be skipped when testing for Copyright (or
38
COPYRIGHT_EXCEPTIONS = ['bzrlib/lsprof.py']
43
COPYRIGHT_EXCEPTIONS = [
44
'bzrlib/_bencode_py.py',
45
'bzrlib/doc_generate/conf.py',
40
LICENSE_EXCEPTIONS = ['bzrlib/lsprof.py']
49
LICENSE_EXCEPTIONS = [
50
'bzrlib/_bencode_py.py',
51
'bzrlib/doc_generate/conf.py',
41
54
# Technically, 'bzrlib/lsprof.py' should be 'bzrlib/util/lsprof.py',
42
55
# (we do not check bzrlib/util/, since that is code bundled from elsewhere)
43
56
# but for compatibility with previous releases, we don't want to move it.
58
# sphinx_conf is semi-autogenerated.
46
61
class TestSourceHelper(TestCase):
48
63
def source_file_name(self, package):
49
64
"""Return the path of the .py file for package."""
65
if getattr(sys, "frozen", None) is not None:
66
raise TestSkipped("can't test sources in frozen distributions.")
50
67
path = package.__file__
51
68
if path[-1] in 'co':
249
277
self.fail('\n'.join(help_text))
251
def test_no_tabs(self):
252
"""bzrlib source files should not contain any tab characters."""
279
def _push_file(self, dict_, fname, line_no):
280
if fname not in dict_:
281
dict_[fname] = [line_no]
283
dict_[fname].append(line_no)
285
def _format_message(self, dict_, message):
286
files = ["%s: %s" % (f, ', '.join([str(i+1) for i in lines]))
287
for f, lines in dict_.items()]
289
return message + '\n\n %s' % ('\n '.join(files))
291
def test_coding_style(self):
292
"""Check if bazaar code conforms to some coding style conventions.
294
Currently we assert that the following is not present:
297
* no newline at end of files
299
Print how many files have
300
* trailing white space
301
* lines longer than 79 chars
305
illegal_newlines = {}
307
no_newline_at_eof = []
308
for fname, text in self.get_source_file_contents(
309
extensions=('.py', '.pyx')):
310
if not self.is_our_code(fname):
312
lines = text.splitlines(True)
313
last_line_no = len(lines) - 1
314
for line_no, line in enumerate(lines):
316
self._push_file(tabs, fname, line_no)
317
if not line.endswith('\n') or line.endswith('\r\n'):
318
if line_no != last_line_no: # not no_newline_at_eof
319
self._push_file(illegal_newlines, fname, line_no)
320
if line.endswith(' \n'):
321
self._push_file(trailing_ws, fname, line_no)
323
self._push_file(long_lines, fname, line_no)
324
if not lines[-1].endswith('\n'):
325
no_newline_at_eof.append(fname)
328
problems.append(self._format_message(tabs,
329
'Tab characters were found in the following source files.'
330
'\nThey should either be replaced by "\\t" or by spaces:'))
332
print ("There are %i lines with trailing white space in %i files."
333
% (sum([len(lines) for f, lines in trailing_ws.items()]),
336
problems.append(self._format_message(illegal_newlines,
337
'Non-unix newlines were found in the following source files:'))
339
print ("There are %i lines longer than 79 characters in %i files."
340
% (sum([len(lines) for f, lines in long_lines.items()]),
342
if no_newline_at_eof:
343
no_newline_at_eof.sort()
344
problems.append("The following source files doesn't have a "
345
"newline at the end:"
347
% ('\n '.join(no_newline_at_eof)))
349
self.fail('\n\n'.join(problems))
351
def test_no_asserts(self):
352
"""bzr shouldn't use the 'assert' statement."""
353
# assert causes too much variation between -O and not, and tends to
354
# give bad errors to the user
356
# scan down through x for assert statements, report any problems
357
# this is a bit cheesy; it may get some false positives?
358
if x[0] == symbol.assert_stmt:
360
elif x[0] == token.NAME:
361
# can't search further down
364
if sub and search(sub):
368
assert_re = re.compile(r'\bassert\b')
255
369
for fname, text in self.get_source_file_contents():
256
if '/util/' in fname or '/plugins/' in fname:
259
incorrect.append(fname)
262
self.fail('Tab characters were found in the following source files.'
263
'\nThey should either be replaced by "\\t" or by spaces:'
265
% ('\n '.join(incorrect)))
370
if not self.is_our_code(fname):
372
if not assert_re.search(text):
374
ast = parser.ast2tuple(parser.suite(text))
376
badfiles.append(fname)
379
"these files contain an assert statement and should not:\n%s"
380
% '\n'.join(badfiles))
382
def test_extension_exceptions(self):
383
"""Extension functions should propagate exceptions.
385
Either they should return an object, have an 'except' clause, or have a
386
"# cannot_raise" to indicate that we've audited them and defined them as not
389
both_exc_and_no_exc = []
391
class_re = re.compile(r'^(cdef\s+)?(public\s+)?'
392
r'(api\s+)?class (\w+).*:', re.MULTILINE)
393
extern_class_re = re.compile(r'## extern cdef class (\w+)',
395
except_re = re.compile(r'cdef\s+' # start with cdef
396
r'([\w *]*?)\s*' # this is the return signature
397
r'(\w+)\s*\(' # the function name
398
r'[^)]*\)\s*' # parameters
399
r'(.*)\s*:' # the except clause
400
r'\s*(#\s*cannot[- _]raise)?' # cannot raise comment
402
for fname, text in self.get_source_file_contents(
403
extensions=('.pyx',)):
404
known_classes = set([m[-1] for m in class_re.findall(text)])
405
known_classes.update(extern_class_re.findall(text))
406
cdefs = except_re.findall(text)
407
for sig, func, exc_clause, no_exc_comment in cdefs:
408
if sig.startswith('api '):
410
if not sig or sig in known_classes:
412
if 'nogil' in exc_clause:
413
exc_clause = exc_clause.replace('nogil', '').strip()
414
if exc_clause and no_exc_comment:
415
both_exc_and_no_exc.append((fname, func))
416
if sig != 'object' and not (exc_clause or no_exc_comment):
417
missing_except.append((fname, func))
419
if both_exc_and_no_exc:
420
error_msg.append('The following functions had "cannot raise" comments'
421
' but did have an except clause set:')
422
for fname, func in both_exc_and_no_exc:
423
error_msg.append('%s:%s' % (fname, func))
424
error_msg.extend(('', ''))
426
error_msg.append('The following functions have fixed return types,'
427
' but no except clause.')
428
error_msg.append('Either add an except or append "# cannot_raise".')
429
for fname, func in missing_except:
430
error_msg.append('%s:%s' % (fname, func))
431
error_msg.extend(('', ''))
433
self.fail('\n'.join(error_msg))