|
17 | 17 |
|
18 | 18 | LOG = logging.getLogger(__name__)
|
19 | 19 |
|
| 20 | +SKIP_TOKENS = frozenset([tokenize.NL, tokenize.NEWLINE, tokenize.INDENT, |
| 21 | + tokenize.DEDENT]) |
| 22 | + |
20 | 23 |
|
21 | 24 | class Manager(object):
|
22 | 25 | """Manage the parallelism and checker instances for each plugin and file.
|
@@ -216,8 +219,15 @@ def run_check(self, plugin, **arguments):
|
216 | 219 |
|
217 | 220 | def run_logical_checks(self):
|
218 | 221 | """Run all checks expecting a logical line."""
|
| 222 | + comments, logical_line, mapping = self.processor.build_logical_line() |
| 223 | + if not mapping: |
| 224 | + return |
| 225 | + self.processor.update_state(mapping) |
| 226 | + |
| 227 | + LOG.debug('Logical line: "%s"', logical_line.rstrip()) |
| 228 | + |
219 | 229 | for plugin in self.checks.logical_line_plugins:
|
220 |
| - result = self.run_check(plugin) # , logical_line=logical_line) |
| 230 | + result = self.run_check(plugin, logical_line=logical_line) |
221 | 231 | if result is not None:
|
222 | 232 | column_offset, text = result
|
223 | 233 | self.report(
|
@@ -415,6 +425,45 @@ def visited_new_blank_line(self):
|
415 | 425 | """Note that we visited a new blank line."""
|
416 | 426 | self.blank_lines += 1
|
417 | 427 |
|
| 428 | + def build_logical_line_tokens(self): |
| 429 | + """Build the mapping, comments, and logical line lists.""" |
| 430 | + logical = [] |
| 431 | + comments = [] |
| 432 | + length = 0 |
| 433 | + previous_row = previous_column = mapping = None |
| 434 | + for token_type, text, start, end, line in self.tokens: |
| 435 | + if token_type in SKIP_TOKENS: |
| 436 | + continue |
| 437 | + if not mapping: |
| 438 | + mapping = [(0, start)] |
| 439 | + if token_type == tokenize.COMMENT: |
| 440 | + comments.append(text) |
| 441 | + continue |
| 442 | + if token_type == tokenize.STRING: |
| 443 | + text = utils.mutate_string(text) |
| 444 | + if previous_row: |
| 445 | + (start_row, start_column) = start |
| 446 | + if previous_row != start_row: |
| 447 | + row_index = previous_row - 1 |
| 448 | + column_index = previous_column - 1 |
| 449 | + previous_text = self.lines[row_index][column_index] |
| 450 | + if (previous_text == ',' or |
| 451 | + (previous_text not in '{[(' and |
| 452 | + text not in '}])')): |
| 453 | + text = ' ' + text |
| 454 | + elif previous_column != start_column: |
| 455 | + text = line[previous_column:start_column] + text |
| 456 | + logical.append(text) |
| 457 | + length += len(text) |
| 458 | + mapping.append((length, end)) |
| 459 | + (previous_row, previous_column) = end |
| 460 | + return comments, logical, mapping |
| 461 | + |
| 462 | + def build_logical_line(self): |
| 463 | + """Build a logical line from the current tokens list.""" |
| 464 | + comments, logical, mapping_list = self.build_logical_line_tokens() |
| 465 | + return ''.join(comments), ''.join(logical), mapping_list |
| 466 | + |
418 | 467 | def split_line(self, token):
|
419 | 468 | """Split a physical line's line based on new-lines.
|
420 | 469 |
|
|
0 commit comments