Source code for statick_tool.plugins.tool.black_tool_plugin

"""Apply black tool and gather results."""
import logging
import re
import subprocess
from typing import List, Match, Optional, Pattern

from statick_tool.issue import Issue
from statick_tool.package import Package
from statick_tool.tool_plugin import ToolPlugin


[docs]class BlackToolPlugin(ToolPlugin): """Apply black tool and gather results."""
[docs] def get_name(self) -> str: """Get name of tool.""" return "black"
[docs] def get_file_types(self) -> List[str]: """Return a list of file types the plugin can scan.""" return ["python_src"]
[docs] def process_files( self, package: Package, level: str, files: List[str], user_flags: List[str] ) -> Optional[List[str]]: """Run tool and gather output.""" flags: List[str] = ["--check"] flags += user_flags total_output: List[str] = [] tool_bin = "black" try: subproc_args = [tool_bin] + flags + files output = subprocess.check_output( subproc_args, stderr=subprocess.STDOUT, universal_newlines=True ) except subprocess.CalledProcessError as ex: # Return code 123 means there was an internal error if ex.returncode == 123: logging.warning("Black encountered internal error") logging.warning("black exception: %s", ex.output) # Return code 1 just means "found problems" elif ex.returncode != 1: logging.warning("Problem %d", ex.returncode) logging.warning("%s exception: %s", self.get_name(), ex.output) return None output = ex.output except OSError as ex: logging.info("Couldn't find %s! (%s)", tool_bin, ex) return None total_output.append(output) logging.debug("%s", total_output) return total_output
[docs] def parse_output( self, total_output: List[str], package: Optional[Package] = None ) -> List[Issue]: """Parse tool output and report issues.""" tool_re_reformat = r"(.+)\s(.+)\s(.+)" parse_reformat: Pattern[str] = re.compile(tool_re_reformat) # example output for this regex: # error: cannot format /home/user/file: INTERNAL ERROR: message tool_re_error = r"\w+:\s(.+):\s(.+):\s(.+):\s(.+)" parse_tool_error: Pattern[str] = re.compile(tool_re_error) # example output for this regex: # error: cannot format /home/user/file: Cannot parse: 1:3: {faulty_line} tool_re_parse_error = r"\w+:\s(.+):\s(.+):\s([0-9]+):([0-9]+):\s(.+)" parse_error: Pattern[str] = re.compile(tool_re_parse_error) issues: List[Issue] = [] for output in total_output: for line in output.splitlines(): if line.startswith("would reformat"): match: Optional[Match[str]] = parse_reformat.match(line) if match: issues.append( Issue( match.group(3), "0", self.get_name(), "format", "3", "would reformat", None, ) ) if line.startswith("error"): match_tool_error: Optional[Match[str]] = parse_tool_error.match( line ) match_parse_error: Optional[Match[str]] = parse_error.match(line) if match_parse_error: issues.append( Issue( match_parse_error.group(1).split(" ")[2].rstrip(":"), match_parse_error.group(3), self.get_name(), "format", "3", match_parse_error.group(2) + " " + match_parse_error.group(5), None, ) ) elif match_tool_error: issues.append( Issue( match_tool_error.group(1).split(" ")[2].rstrip(":"), "0", self.get_name(), "format", "3", match_tool_error.group(3), None, ) ) return issues