Source code for cpp_linter.clang_tools
from concurrent.futures import ProcessPoolExecutor, as_completed
import json
from pathlib import Path
import re
import subprocess
from typing import Optional, List, Dict, Tuple, cast
import shutil
from ..common_fs import FileObj, FileIOTimeout
from ..common_fs.file_filter import TidyFileFilter, FormatFileFilter
from ..loggers import start_log_group, end_log_group, worker_log_init, logger
from .clang_tidy import run_clang_tidy, TidyAdvice
from .clang_format import run_clang_format, FormatAdvice
from ..cli import Args
[docs]
def assemble_version_exec(tool_name: str, specified_version: str) -> Optional[str]:
"""Assembles the command to the executable of the given clang tool based on given
version information.
:param tool_name: The name of the clang tool to be executed.
:param specified_version: The version number or the installed path to a version of
the tool's executable.
"""
semver = specified_version.split(".")
exe_path = None
if semver and semver[0].isdigit(): # version info is not a path
# let's assume the exe is in the PATH env var
exe_path = shutil.which(f"{tool_name}-{specified_version}")
elif specified_version: # treat value as a path to binary executable
exe_path = shutil.which(tool_name, path=specified_version)
if exe_path is not None:
return exe_path
return shutil.which(tool_name)
def _run_on_single_file(
file: FileObj,
log_lvl: int,
tidy_cmd: Optional[str],
db_json: Optional[List[Dict[str, str]]],
format_cmd: Optional[str],
format_filter: Optional[FormatFileFilter],
tidy_filter: Optional[TidyFileFilter],
args: Args,
) -> Tuple[str, str, Optional[TidyAdvice], Optional[FormatAdvice]]:
log_stream = worker_log_init(log_lvl)
filename = Path(file.name).as_posix()
format_advice = None
if format_cmd is not None and (
format_filter is None or format_filter.is_source_or_ignored(file.name)
):
try:
format_advice = run_clang_format(
command=format_cmd,
file_obj=file,
style=args.style,
lines_changed_only=args.lines_changed_only,
format_review=args.format_review,
)
except FileIOTimeout: # pragma: no cover
logger.error(
"Failed to read or write contents of %s when running clang-format",
filename,
)
except OSError: # pragma: no cover
logger.error(
"Failed to open the file %s when running clang-format", filename
)
tidy_note = None
if tidy_cmd is not None and (
tidy_filter is None or tidy_filter.is_source_or_ignored(file.name)
):
try:
tidy_note = run_clang_tidy(
command=tidy_cmd,
file_obj=file,
checks=args.tidy_checks,
lines_changed_only=args.lines_changed_only,
database=args.database,
extra_args=args.extra_arg,
db_json=db_json,
tidy_review=args.tidy_review,
style=args.style,
)
except FileIOTimeout: # pragma: no cover
logger.error(
"Failed to Read/Write contents of %s when running clang-tidy", filename
)
except OSError: # pragma: no cover
logger.error("Failed to open the file %s when running clang-tidy", filename)
return file.name, log_stream.getvalue(), tidy_note, format_advice
VERSION_PATTERN = re.compile(r"version\s(\d+\.\d+\.\d+)")
def _capture_tool_version(cmd: str) -> str:
"""Get version number from output for executable used."""
version_out = subprocess.run(
[cmd, "--version"], capture_output=True, check=True, text=True
)
matched = VERSION_PATTERN.search(version_out.stdout)
if matched is None: # pragma: no cover
raise RuntimeError(
f"Failed to get version numbers from `{cmd} --version` output"
)
ver = cast(str, matched.group(1))
logger.info("`%s --version`: %s", cmd, ver)
return ver
class ClangVersions:
def __init__(self) -> None:
self.tidy: Optional[str] = None
self.format: Optional[str] = None
[docs]
def capture_clang_tools_output(files: List[FileObj], args: Args) -> ClangVersions:
"""Execute and capture all output from clang-tidy and clang-format. This aggregates
results in the :attr:`~cpp_linter.Globals.OUTPUT`.
:param files: A list of files to analyze.
:param args: A namespace of parsed args from the :doc:`CLI <../cli_args>`.
"""
tidy_cmd, format_cmd = (None, None)
tidy_filter, format_filter = (None, None)
clang_versions = ClangVersions()
if args.style: # if style is an empty value, then clang-format is skipped
format_cmd = assemble_version_exec("clang-format", args.version)
if format_cmd is None: # pragma: no cover
raise FileNotFoundError("clang-format executable was not found")
clang_versions.format = _capture_tool_version(format_cmd)
format_filter = FormatFileFilter(
extensions=args.extensions,
ignore_value=args.ignore_format,
)
if args.tidy_checks != "-*":
# if all checks are disabled, then clang-tidy is skipped
tidy_cmd = assemble_version_exec("clang-tidy", args.version)
if tidy_cmd is None: # pragma: no cover
raise FileNotFoundError("clang-tidy executable was not found")
clang_versions.tidy = _capture_tool_version(tidy_cmd)
tidy_filter = TidyFileFilter(
extensions=args.extensions,
ignore_value=args.ignore_tidy,
)
db_json: Optional[List[Dict[str, str]]] = None
if args.database:
db = Path(args.database)
if not db.is_absolute():
args.database = str(db.resolve())
db_path = (db / "compile_commands.json").resolve()
if db_path.exists():
db_json = json.loads(db_path.read_text(encoding="utf-8"))
with ProcessPoolExecutor(args.jobs) as executor:
log_lvl = logger.getEffectiveLevel()
futures = [
executor.submit(
_run_on_single_file,
file,
log_lvl=log_lvl,
tidy_cmd=tidy_cmd,
db_json=db_json,
format_cmd=format_cmd,
format_filter=format_filter,
tidy_filter=tidy_filter,
args=args,
)
for file in files
]
# temporary cache of parsed notifications for use in log commands
for future in as_completed(futures):
file_name, logs, tidy_advice, format_advice = future.result()
start_log_group(f"Performing checkup on {file_name}")
print(logs, flush=True)
end_log_group()
if tidy_advice or format_advice:
for file in files:
if file.name == file_name:
if tidy_advice:
file.tidy_advice = tidy_advice
if format_advice:
file.format_advice = format_advice
break
else: # pragma: no cover
raise ValueError(f"Failed to find {file_name} in list of files.")
return clang_versions