3 # ===- clang-tidy-diff.py - ClangTidy Diff Checker -----------*- python -*--===#
5 # Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
6 # See https://llvm.org/LICENSE.txt for license information.
7 # SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
9 # ===-----------------------------------------------------------------------===#
12 ClangTidy Diff Checker
13 ======================
15 This script reads input from a unified diff, runs clang-tidy on all changed
16 files and outputs clang-tidy warnings in changed lines only. This is useful to
17 detect clang-tidy regressions in the lines touched by a specific patch.
18 Example usage for git/svn users:
20 git diff -U0 HEAD^ | clang-tidy-diff.py -p1
21 svn diff --diff-cmd=diff -x-U0 | \
22 clang-tidy-diff.py -fix -checks=-*,modernize-use-override
29 import multiprocessing
44 is_py2
= sys
.version
[0] == "2"
52 def run_tidy(task_queue
, lock
, timeout
, failed_files
):
55 command
= task_queue
.get()
57 proc
= subprocess
.Popen(
58 command
, stdout
=subprocess
.PIPE
, stderr
=subprocess
.PIPE
61 if timeout
is not None:
62 watchdog
= threading
.Timer(timeout
, proc
.kill
)
65 stdout
, stderr
= proc
.communicate()
66 if proc
.returncode
!= 0:
67 if proc
.returncode
< 0:
68 msg
= "Terminated by signal %d : %s\n" % (
72 stderr
+= msg
.encode("utf-8")
73 failed_files
.append(command
)
76 sys
.stdout
.write(stdout
.decode("utf-8") + "\n")
79 sys
.stderr
.write(stderr
.decode("utf-8") + "\n")
81 except Exception as e
:
83 sys
.stderr
.write("Failed: " + str(e
) + ": ".join(command
) + "\n")
86 if not (timeout
is None or watchdog
is None):
87 if not watchdog
.is_alive():
89 "Terminated by timeout: " + " ".join(command
) + "\n"
92 task_queue
.task_done()
95 def start_workers(max_tasks
, tidy_caller
, arguments
):
96 for _
in range(max_tasks
):
97 t
= threading
.Thread(target
=tidy_caller
, args
=arguments
)
102 def merge_replacement_files(tmpdir
, mergefile
):
103 """Merge all replacement files in a directory into a single file"""
104 # The fixes suggested by clang-tidy >= 4.0.0 are given under
105 # the top level key 'Diagnostics' in the output yaml files
106 mergekey
= "Diagnostics"
108 for replacefile
in glob
.iglob(os
.path
.join(tmpdir
, "*.yaml")):
109 content
= yaml
.safe_load(open(replacefile
, "r"))
111 continue # Skip empty files.
112 merged
.extend(content
.get(mergekey
, []))
115 # MainSourceFile: The key is required by the definition inside
116 # include/clang/Tooling/ReplacementsYaml.h, but the value
117 # is actually never used inside clang-apply-replacements,
118 # so we set it to '' here.
119 output
= {"MainSourceFile": "", mergekey
: merged
}
120 with
open(mergefile
, "w") as out
:
121 yaml
.safe_dump(output
, out
)
124 open(mergefile
, "w").close()
128 parser
= argparse
.ArgumentParser(
129 description
="Run clang-tidy against changed files, and "
130 "output diagnostics only for modified "
134 "-clang-tidy-binary",
136 default
="clang-tidy",
137 help="path to clang-tidy binary",
143 help="strip the smallest prefix containing P slashes",
149 help="custom pattern selecting file paths to check "
150 "(case sensitive, overrides -iregex)",
155 default
=r
".*\.(cpp|cc|c\+\+|cxx|c|cl|h|hpp|m|mm|inc)",
156 help="custom pattern selecting file paths to check "
157 "(case insensitive, overridden by -regex)",
163 help="number of tidy instances to be run in parallel.",
166 "-timeout", type=int, default
=None, help="timeout per each file in seconds."
169 "-fix", action
="store_true", default
=False, help="apply suggested fixes"
173 help="checks filter, when not specified, use clang-tidy " "default",
176 parser
.add_argument("-use-color", action
="store_true", help="Use colors in output")
178 "-path", dest
="build_path", help="Path used to read a compile command database."
183 metavar
="FILE_OR_DIRECTORY",
185 help="A directory or a yaml file to store suggested fixes in, "
186 "which can be applied with clang-apply-replacements. If the "
187 "parameter is a directory, the fixes of each compilation unit are "
188 "stored in individual yaml files in the directory.",
195 help="A directory to store suggested fixes in, which can be applied "
196 "with clang-apply-replacements. The fixes of each compilation unit are "
197 "stored in individual yaml files in the directory.",
204 help="Additional argument to append to the compiler " "command line.",
208 dest
="extra_arg_before",
211 help="Additional argument to prepend to the compiler " "command line.",
217 help="Run clang-tidy in quiet mode",
224 help="Load the specified plugin in clang-tidy.",
230 clang_tidy_args
.extend(argv
[argv
.index("--") :])
231 argv
= argv
[: argv
.index("--")]
233 args
= parser
.parse_args(argv
)
235 # Extract changed lines for each file.
238 for line
in sys
.stdin
:
239 match
= re
.search('^\+\+\+\ "?(.*?/){%s}([^ \t\n"]*)' % args
.p
, line
)
241 filename
= match
.group(2)
245 if args
.regex
is not None:
246 if not re
.match("^%s$" % args
.regex
, filename
):
249 if not re
.match("^%s$" % args
.iregex
, filename
, re
.IGNORECASE
):
252 match
= re
.search("^@@.*\+(\d+)(,(\d+))?", line
)
254 start_line
= int(match
.group(1))
257 line_count
= int(match
.group(3))
260 end_line
= start_line
+ line_count
- 1
261 lines_by_file
.setdefault(filename
, []).append([start_line
, end_line
])
263 if not any(lines_by_file
):
264 print("No relevant changes found.")
267 max_task_count
= args
.j
268 if max_task_count
== 0:
269 max_task_count
= multiprocessing
.cpu_count()
270 max_task_count
= min(len(lines_by_file
), max_task_count
)
272 combine_fixes
= False
273 export_fixes_dir
= None
274 delete_fixes_dir
= False
275 if args
.export_fixes
is not None:
276 # if a directory is given, create it if it does not exist
277 if args
.export_fixes
.endswith(os
.path
.sep
) and not os
.path
.isdir(
280 os
.makedirs(args
.export_fixes
)
282 if not os
.path
.isdir(args
.export_fixes
):
285 "Cannot combine fixes in one yaml file. Either install PyYAML or specify an output directory."
290 if os
.path
.isdir(args
.export_fixes
):
291 export_fixes_dir
= args
.export_fixes
294 export_fixes_dir
= tempfile
.mkdtemp()
295 delete_fixes_dir
= True
297 # Tasks for clang-tidy.
298 task_queue
= queue
.Queue(max_task_count
)
299 # A lock for console output.
300 lock
= threading
.Lock()
302 # List of files with a non-zero return code.
305 # Run a pool of clang-tidy workers.
307 max_task_count
, run_tidy
, (task_queue
, lock
, args
.timeout
, failed_files
)
310 # Form the common args list.
311 common_clang_tidy_args
= []
313 common_clang_tidy_args
.append("-fix")
314 if args
.checks
!= "":
315 common_clang_tidy_args
.append("-checks=" + args
.checks
)
317 common_clang_tidy_args
.append("-quiet")
318 if args
.build_path
is not None:
319 common_clang_tidy_args
.append("-p=%s" % args
.build_path
)
321 common_clang_tidy_args
.append("--use-color")
322 for arg
in args
.extra_arg
:
323 common_clang_tidy_args
.append("-extra-arg=%s" % arg
)
324 for arg
in args
.extra_arg_before
:
325 common_clang_tidy_args
.append("-extra-arg-before=%s" % arg
)
326 for plugin
in args
.plugins
:
327 common_clang_tidy_args
.append("-load=%s" % plugin
)
329 for name
in lines_by_file
:
330 line_filter_json
= json
.dumps(
331 [{"name": name
, "lines": lines_by_file
[name
]}], separators
=(",", ":")
334 # Run clang-tidy on files containing changes.
335 command
= [args
.clang_tidy_binary
]
336 command
.append("-line-filter=" + line_filter_json
)
337 if args
.export_fixes
is not None:
338 # Get a temporary file. We immediately close the handle so clang-tidy can
340 (handle
, tmp_name
) = tempfile
.mkstemp(suffix
=".yaml", dir=export_fixes_dir
)
342 command
.append("-export-fixes=" + tmp_name
)
343 command
.extend(common_clang_tidy_args
)
345 command
.extend(clang_tidy_args
)
347 task_queue
.put(command
)
349 # Application return code
352 # Wait for all threads to be done.
354 # Application return code
360 print("Writing fixes to " + args
.export_fixes
+ " ...")
362 merge_replacement_files(export_fixes_dir
, args
.export_fixes
)
364 sys
.stderr
.write("Error exporting fixes.\n")
365 traceback
.print_exc()
369 shutil
.rmtree(export_fixes_dir
)
370 sys
.exit(return_code
)
373 if __name__
== "__main__":