Matthew Treinish | 8b37289 | 2012-12-07 17:13:16 -0500 | [diff] [blame] | 1 | #!/usr/bin/env python |
| 2 | # vim: tabstop=4 shiftwidth=4 softtabstop=4 |
| 3 | |
| 4 | # Copyright (c) 2012, Cloudscaling |
| 5 | # All Rights Reserved. |
| 6 | # |
| 7 | # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| 8 | # not use this file except in compliance with the License. You may obtain |
| 9 | # a copy of the License at |
| 10 | # |
| 11 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 12 | # |
| 13 | # Unless required by applicable law or agreed to in writing, software |
| 14 | # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| 15 | # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| 16 | # License for the specific language governing permissions and limitations |
| 17 | # under the License. |
| 18 | |
| 19 | """tempest HACKING file compliance testing |
| 20 | |
| 21 | built on top of pep8.py |
| 22 | """ |
| 23 | |
| 24 | import fnmatch |
| 25 | import inspect |
| 26 | import logging |
| 27 | import os |
| 28 | import re |
| 29 | import subprocess |
| 30 | import sys |
| 31 | import tokenize |
| 32 | import warnings |
| 33 | |
| 34 | import pep8 |
| 35 | |
| 36 | # Don't need this for testing |
| 37 | logging.disable('LOG') |
| 38 | |
| 39 | #N1xx comments |
| 40 | #N2xx except |
| 41 | #N3xx imports |
| 42 | #N4xx docstrings |
| 43 | #N5xx dictionaries/lists |
| 44 | #N6xx calling methods |
| 45 | #N7xx localization |
| 46 | #N8xx git commit messages |
| 47 | |
| 48 | IMPORT_EXCEPTIONS = ['sqlalchemy', 'migrate'] |
| 49 | DOCSTRING_TRIPLE = ['"""', "'''"] |
| 50 | VERBOSE_MISSING_IMPORT = os.getenv('HACKING_VERBOSE_MISSING_IMPORT', 'False') |
| 51 | |
| 52 | |
| 53 | # Monkey patch broken excluded filter in pep8 |
| 54 | # See https://github.com/jcrocholl/pep8/pull/111 |
| 55 | def excluded(self, filename): |
| 56 | """ |
| 57 | Check if options.exclude contains a pattern that matches filename. |
| 58 | """ |
| 59 | basename = os.path.basename(filename) |
| 60 | return any((pep8.filename_match(filename, self.options.exclude, |
| 61 | default=False), |
| 62 | pep8.filename_match(basename, self.options.exclude, |
| 63 | default=False))) |
| 64 | |
| 65 | |
| 66 | def input_dir(self, dirname): |
| 67 | """Check all files in this directory and all subdirectories.""" |
| 68 | dirname = dirname.rstrip('/') |
| 69 | if self.excluded(dirname): |
| 70 | return 0 |
| 71 | counters = self.options.report.counters |
| 72 | verbose = self.options.verbose |
| 73 | filepatterns = self.options.filename |
| 74 | runner = self.runner |
| 75 | for root, dirs, files in os.walk(dirname): |
| 76 | if verbose: |
| 77 | print('directory ' + root) |
| 78 | counters['directories'] += 1 |
| 79 | for subdir in sorted(dirs): |
| 80 | if self.excluded(os.path.join(root, subdir)): |
| 81 | dirs.remove(subdir) |
| 82 | for filename in sorted(files): |
| 83 | # contain a pattern that matches? |
| 84 | if ((pep8.filename_match(filename, filepatterns) and |
| 85 | not self.excluded(filename))): |
| 86 | runner(os.path.join(root, filename)) |
| 87 | |
| 88 | |
| 89 | def is_import_exception(mod): |
| 90 | return (mod in IMPORT_EXCEPTIONS or |
| 91 | any(mod.startswith(m + '.') for m in IMPORT_EXCEPTIONS)) |
| 92 | |
| 93 | |
| 94 | def import_normalize(line): |
| 95 | # convert "from x import y" to "import x.y" |
| 96 | # handle "from x import y as z" to "import x.y as z" |
| 97 | split_line = line.split() |
| 98 | if ("import" in line and line.startswith("from ") and "," not in line and |
| 99 | split_line[2] == "import" and split_line[3] != "*" and |
| 100 | split_line[1] != "__future__" and |
| 101 | (len(split_line) == 4 or |
| 102 | (len(split_line) == 6 and split_line[4] == "as"))): |
| 103 | return "import %s.%s" % (split_line[1], split_line[3]) |
| 104 | else: |
| 105 | return line |
| 106 | |
| 107 | |
| 108 | def tempest_todo_format(physical_line): |
| 109 | """Check for 'TODO()'. |
| 110 | |
| 111 | tempest HACKING guide recommendation for TODO: |
| 112 | Include your name with TODOs as in "#TODO(termie)" |
| 113 | N101 |
| 114 | """ |
| 115 | pos = physical_line.find('TODO') |
| 116 | pos1 = physical_line.find('TODO(') |
| 117 | pos2 = physical_line.find('#') # make sure it's a comment |
| 118 | if (pos != pos1 and pos2 >= 0 and pos2 < pos): |
| 119 | return pos, "TEMPEST N101: Use TODO(NAME)" |
| 120 | |
| 121 | |
| 122 | def tempest_except_format(logical_line): |
| 123 | """Check for 'except:'. |
| 124 | |
| 125 | tempest HACKING guide recommends not using except: |
| 126 | Do not write "except:", use "except Exception:" at the very least |
| 127 | N201 |
| 128 | """ |
| 129 | if logical_line.startswith("except:"): |
| 130 | yield 6, "TEMPEST N201: no 'except:' at least use 'except Exception:'" |
| 131 | |
| 132 | |
| 133 | def tempest_except_format_assert(logical_line): |
| 134 | """Check for 'assertRaises(Exception'. |
| 135 | |
| 136 | tempest HACKING guide recommends not using assertRaises(Exception...): |
| 137 | Do not use overly broad Exception type |
| 138 | N202 |
| 139 | """ |
| 140 | if logical_line.startswith("self.assertRaises(Exception"): |
| 141 | yield 1, "TEMPEST N202: assertRaises Exception too broad" |
| 142 | |
| 143 | |
| 144 | def tempest_one_import_per_line(logical_line): |
| 145 | """Check for import format. |
| 146 | |
| 147 | tempest HACKING guide recommends one import per line: |
| 148 | Do not import more than one module per line |
| 149 | |
| 150 | Examples: |
| 151 | BAD: from tempest.common.rest_client import RestClient, RestClientXML |
| 152 | N301 |
| 153 | """ |
| 154 | pos = logical_line.find(',') |
| 155 | parts = logical_line.split() |
| 156 | if (pos > -1 and (parts[0] == "import" or |
| 157 | parts[0] == "from" and parts[2] == "import") and |
| 158 | not is_import_exception(parts[1])): |
| 159 | yield pos, "TEMPEST N301: one import per line" |
| 160 | |
| 161 | _missingImport = set([]) |
| 162 | |
| 163 | |
| 164 | def tempest_import_module_only(logical_line): |
| 165 | """Check for import module only. |
| 166 | |
| 167 | tempest HACKING guide recommends importing only modules: |
| 168 | Do not import objects, only modules |
| 169 | N302 import only modules |
| 170 | N303 Invalid Import |
| 171 | N304 Relative Import |
| 172 | """ |
| 173 | def importModuleCheck(mod, parent=None, added=False): |
| 174 | """ |
| 175 | If can't find module on first try, recursively check for relative |
| 176 | imports |
| 177 | """ |
| 178 | current_path = os.path.dirname(pep8.current_file) |
| 179 | try: |
| 180 | with warnings.catch_warnings(): |
| 181 | warnings.simplefilter('ignore', DeprecationWarning) |
| 182 | valid = True |
| 183 | if parent: |
| 184 | if is_import_exception(parent): |
| 185 | return |
| 186 | parent_mod = __import__(parent, globals(), locals(), |
| 187 | [mod], -1) |
| 188 | valid = inspect.ismodule(getattr(parent_mod, mod)) |
| 189 | else: |
| 190 | __import__(mod, globals(), locals(), [], -1) |
| 191 | valid = inspect.ismodule(sys.modules[mod]) |
| 192 | if not valid: |
| 193 | if added: |
| 194 | sys.path.pop() |
| 195 | added = False |
| 196 | return logical_line.find(mod), ("TEMPEST N304: No " |
| 197 | "relative imports. " |
| 198 | "'%s' is a relative " |
| 199 | "import" |
| 200 | % logical_line) |
| 201 | return logical_line.find(mod), ("TEMPEST N302: import only" |
| 202 | " modules. '%s' does not " |
| 203 | "import a module" |
| 204 | % logical_line) |
| 205 | |
| 206 | except (ImportError, NameError) as exc: |
| 207 | if not added: |
| 208 | added = True |
| 209 | sys.path.append(current_path) |
| 210 | return importModuleCheck(mod, parent, added) |
| 211 | else: |
| 212 | name = logical_line.split()[1] |
| 213 | if name not in _missingImport: |
| 214 | if VERBOSE_MISSING_IMPORT != 'False': |
| 215 | print >> sys.stderr, ("ERROR: import '%s' in %s " |
| 216 | "failed: %s" % |
| 217 | (name, pep8.current_file, exc)) |
| 218 | _missingImport.add(name) |
| 219 | added = False |
| 220 | sys.path.pop() |
| 221 | return |
| 222 | |
| 223 | except AttributeError: |
| 224 | # Invalid import |
| 225 | return logical_line.find(mod), ("TEMPEST N303: Invalid import, " |
| 226 | "AttributeError raised") |
| 227 | |
| 228 | # convert "from x import y" to " import x.y" |
| 229 | # convert "from x import y as z" to " import x.y" |
| 230 | import_normalize(logical_line) |
| 231 | split_line = logical_line.split() |
| 232 | |
| 233 | if (logical_line.startswith("import ") and "," not in logical_line and |
| 234 | (len(split_line) == 2 or |
| 235 | (len(split_line) == 4 and split_line[2] == "as"))): |
| 236 | mod = split_line[1] |
| 237 | rval = importModuleCheck(mod) |
| 238 | if rval is not None: |
| 239 | yield rval |
| 240 | |
| 241 | # TODO(jogo) handle "from x import *" |
| 242 | |
| 243 | #TODO(jogo): import template: N305 |
| 244 | |
| 245 | |
| 246 | def tempest_import_alphabetical(logical_line, line_number, lines): |
| 247 | """Check for imports in alphabetical order. |
| 248 | |
| 249 | Tempest HACKING guide recommendation for imports: |
| 250 | imports in human alphabetical order |
| 251 | N306 |
| 252 | """ |
| 253 | # handle import x |
| 254 | # use .lower since capitalization shouldn't dictate order |
| 255 | split_line = import_normalize(logical_line.strip()).lower().split() |
| 256 | split_previous = import_normalize(lines[ |
| 257 | line_number - 2]).strip().lower().split() |
| 258 | # with or without "as y" |
| 259 | length = [2, 4] |
| 260 | if (len(split_line) in length and len(split_previous) in length and |
| 261 | split_line[0] == "import" and split_previous[0] == "import"): |
| 262 | if split_line[1] < split_previous[1]: |
| 263 | yield (0, "TEMPEST N306: imports not in alphabetical order" |
| 264 | " (%s, %s)" |
| 265 | % (split_previous[1], split_line[1])) |
| 266 | |
| 267 | |
| 268 | def tempest_docstring_start_space(physical_line): |
| 269 | """Check for docstring not start with space. |
| 270 | |
| 271 | tempest HACKING guide recommendation for docstring: |
| 272 | Docstring should not start with space |
| 273 | N401 |
| 274 | """ |
| 275 | pos = max([physical_line.find(i) for i in DOCSTRING_TRIPLE]) # start |
| 276 | if (pos != -1 and len(physical_line) > pos + 1): |
| 277 | if (physical_line[pos + 3] == ' '): |
| 278 | return (pos, "TEMPEST N401: one line docstring should not start" |
| 279 | " with a space") |
| 280 | |
| 281 | |
| 282 | def tempest_docstring_one_line(physical_line): |
| 283 | """Check one line docstring end. |
| 284 | |
| 285 | tempest HACKING guide recommendation for one line docstring: |
| 286 | A one line docstring looks like this and ends in a period. |
| 287 | N402 |
| 288 | """ |
| 289 | pos = max([physical_line.find(i) for i in DOCSTRING_TRIPLE]) # start |
| 290 | end = max([physical_line[-4:-1] == i for i in DOCSTRING_TRIPLE]) # end |
| 291 | if (pos != -1 and end and len(physical_line) > pos + 4): |
| 292 | if (physical_line[-5] != '.'): |
Sean Dague | 97449cc | 2013-01-04 14:38:26 -0500 | [diff] [blame^] | 293 | return pos, "N402: one line docstring needs a period" |
Matthew Treinish | 8b37289 | 2012-12-07 17:13:16 -0500 | [diff] [blame] | 294 | |
| 295 | |
| 296 | def tempest_docstring_multiline_end(physical_line): |
| 297 | """Check multi line docstring end. |
| 298 | |
| 299 | Tempest HACKING guide recommendation for docstring: |
| 300 | Docstring should end on a new line |
| 301 | N403 |
| 302 | """ |
| 303 | pos = max([physical_line.find(i) for i in DOCSTRING_TRIPLE]) # start |
| 304 | if (pos != -1 and len(physical_line) == pos): |
| 305 | if (physical_line[pos + 3] == ' '): |
| 306 | return (pos, "TEMPEST N403: multi line docstring end on new line") |
| 307 | |
| 308 | |
Sean Dague | 97449cc | 2013-01-04 14:38:26 -0500 | [diff] [blame^] | 309 | def tempest_no_test_docstring(physical_line, previous_logical, filename): |
| 310 | """Check that test_ functions don't have docstrings |
| 311 | |
| 312 | This ensure we get better results out of tempest, instead |
| 313 | of them being hidden behind generic descriptions of the |
| 314 | functions. |
| 315 | |
| 316 | N404 |
| 317 | """ |
| 318 | if "tempest/test" in filename: |
| 319 | pos = max([physical_line.find(i) for i in DOCSTRING_TRIPLE]) |
| 320 | if pos != -1: |
| 321 | if previous_logical.startswith("def test_"): |
| 322 | return (pos, "TEMPEST N404: test functions must " |
| 323 | "not have doc strings") |
| 324 | |
| 325 | |
Matthew Treinish | 8b37289 | 2012-12-07 17:13:16 -0500 | [diff] [blame] | 326 | FORMAT_RE = re.compile("%(?:" |
| 327 | "%|" # Ignore plain percents |
| 328 | "(\(\w+\))?" # mapping key |
| 329 | "([#0 +-]?" # flag |
| 330 | "(?:\d+|\*)?" # width |
| 331 | "(?:\.\d+)?" # precision |
| 332 | "[hlL]?" # length mod |
| 333 | "\w))") # type |
| 334 | |
| 335 | |
| 336 | class LocalizationError(Exception): |
| 337 | pass |
| 338 | |
| 339 | |
| 340 | def check_i18n(): |
| 341 | """Generator that checks token stream for localization errors. |
| 342 | |
| 343 | Expects tokens to be ``send``ed one by one. |
| 344 | Raises LocalizationError if some error is found. |
| 345 | """ |
| 346 | while True: |
| 347 | try: |
| 348 | token_type, text, _, _, line = yield |
| 349 | except GeneratorExit: |
| 350 | return |
| 351 | if (token_type == tokenize.NAME and text == "_" and |
| 352 | not line.startswith('def _(msg):')): |
| 353 | |
| 354 | while True: |
| 355 | token_type, text, start, _, _ = yield |
| 356 | if token_type != tokenize.NL: |
| 357 | break |
| 358 | if token_type != tokenize.OP or text != "(": |
| 359 | continue # not a localization call |
| 360 | |
| 361 | format_string = '' |
| 362 | while True: |
| 363 | token_type, text, start, _, _ = yield |
| 364 | if token_type == tokenize.STRING: |
| 365 | format_string += eval(text) |
| 366 | elif token_type == tokenize.NL: |
| 367 | pass |
| 368 | else: |
| 369 | break |
| 370 | |
| 371 | if not format_string: |
| 372 | raise LocalizationError(start, |
| 373 | "TEMEPST N701: Empty localization " |
| 374 | "string") |
| 375 | if token_type != tokenize.OP: |
| 376 | raise LocalizationError(start, |
| 377 | "TEMPEST N701: Invalid localization " |
| 378 | "call") |
| 379 | if text != ")": |
| 380 | if text == "%": |
| 381 | raise LocalizationError(start, |
| 382 | "TEMPEST N702: Formatting " |
| 383 | "operation should be outside" |
| 384 | " of localization method call") |
| 385 | elif text == "+": |
| 386 | raise LocalizationError(start, |
| 387 | "TEMPEST N702: Use bare string " |
| 388 | "concatenation instead of +") |
| 389 | else: |
| 390 | raise LocalizationError(start, |
| 391 | "TEMPEST N702: Argument to _ must" |
| 392 | " be just a string") |
| 393 | |
| 394 | format_specs = FORMAT_RE.findall(format_string) |
| 395 | positional_specs = [(key, spec) for key, spec in format_specs |
| 396 | if not key and spec] |
| 397 | # not spec means %%, key means %(smth)s |
| 398 | if len(positional_specs) > 1: |
| 399 | raise LocalizationError(start, |
| 400 | "TEMPEST N703: Multiple positional " |
| 401 | "placeholders") |
| 402 | |
| 403 | |
| 404 | def tempest_localization_strings(logical_line, tokens): |
| 405 | """Check localization in line. |
| 406 | |
| 407 | N701: bad localization call |
| 408 | N702: complex expression instead of string as argument to _() |
| 409 | N703: multiple positional placeholders |
| 410 | """ |
| 411 | |
| 412 | gen = check_i18n() |
| 413 | next(gen) |
| 414 | try: |
| 415 | map(gen.send, tokens) |
| 416 | gen.close() |
| 417 | except LocalizationError as e: |
| 418 | yield e.args |
| 419 | |
| 420 | #TODO(jogo) Dict and list objects |
| 421 | |
| 422 | current_file = "" |
| 423 | |
| 424 | |
| 425 | def readlines(filename): |
| 426 | """Record the current file being tested.""" |
| 427 | pep8.current_file = filename |
| 428 | return open(filename).readlines() |
| 429 | |
| 430 | |
| 431 | def add_tempest(): |
| 432 | """Monkey patch in tempest guidelines. |
| 433 | |
| 434 | Look for functions that start with tempest_ and have arguments |
| 435 | and add them to pep8 module |
| 436 | Assumes you know how to write pep8.py checks |
| 437 | """ |
| 438 | for name, function in globals().items(): |
| 439 | if not inspect.isfunction(function): |
| 440 | continue |
| 441 | args = inspect.getargspec(function)[0] |
| 442 | if args and name.startswith("tempest"): |
| 443 | exec("pep8.%s = %s" % (name, name)) |
| 444 | |
| 445 | |
| 446 | def once_git_check_commit_title(): |
| 447 | """Check git commit messages. |
| 448 | |
| 449 | tempest HACKING recommends not referencing a bug or blueprint |
| 450 | in first line, it should provide an accurate description of the change |
| 451 | N801 |
| 452 | N802 Title limited to 50 chars |
| 453 | """ |
| 454 | #Get title of most recent commit |
| 455 | |
| 456 | subp = subprocess.Popen(['git', 'log', '--no-merges', '--pretty=%s', '-1'], |
| 457 | stdout=subprocess.PIPE) |
| 458 | title = subp.communicate()[0] |
| 459 | if subp.returncode: |
| 460 | raise Exception("git log failed with code %s" % subp.returncode) |
| 461 | |
| 462 | #From https://github.com/openstack/openstack-ci-puppet |
| 463 | # /blob/master/modules/gerrit/manifests/init.pp#L74 |
| 464 | #Changeid|bug|blueprint |
| 465 | git_keywords = (r'(I[0-9a-f]{8,40})|' |
| 466 | '([Bb]ug|[Ll][Pp])[\s\#:]*(\d+)|' |
| 467 | '([Bb]lue[Pp]rint|[Bb][Pp])[\s\#:]*([A-Za-z0-9\\-]+)') |
| 468 | GIT_REGEX = re.compile(git_keywords) |
| 469 | |
| 470 | error = False |
| 471 | #NOTE(jogo) if match regex but over 3 words, acceptable title |
| 472 | if GIT_REGEX.search(title) is not None and len(title.split()) <= 3: |
| 473 | print ("N801: git commit title ('%s') should provide an accurate " |
| 474 | "description of the change, not just a reference to a bug " |
| 475 | "or blueprint" % title.strip()) |
| 476 | error = True |
| 477 | if len(title.decode('utf-8')) > 72: |
| 478 | print ("N802: git commit title ('%s') should be under 50 chars" |
| 479 | % title.strip()) |
| 480 | error = True |
| 481 | return error |
| 482 | |
| 483 | if __name__ == "__main__": |
| 484 | #include tempest path |
| 485 | sys.path.append(os.getcwd()) |
| 486 | #Run once tests (not per line) |
| 487 | once_error = once_git_check_commit_title() |
| 488 | #TEMPEST error codes start with an N |
| 489 | pep8.ERRORCODE_REGEX = re.compile(r'[EWN]\d{3}') |
| 490 | add_tempest() |
| 491 | pep8.current_file = current_file |
| 492 | pep8.readlines = readlines |
| 493 | pep8.StyleGuide.excluded = excluded |
| 494 | pep8.StyleGuide.input_dir = input_dir |
| 495 | try: |
| 496 | pep8._main() |
| 497 | sys.exit(once_error) |
| 498 | finally: |
| 499 | if len(_missingImport) > 0: |
| 500 | print >> sys.stderr, ("%i imports missing in this test environment" |
| 501 | % len(_missingImport)) |