homebrew-core/Formula/cppcheck.rb

128 lines
4.5 KiB
Ruby

class Cppcheck < Formula
desc "Static analysis of C and C++ code"
homepage "https://sourceforge.net/projects/cppcheck/"
url "https://github.com/danmar/cppcheck/archive/2.10.tar.gz"
sha256 "785dcbf711048dfe43ae920b6eff2eeebb4a096e88188a40e173ca4c030f57c3"
license "GPL-3.0-or-later"
head "https://github.com/danmar/cppcheck.git", branch: "main"
bottle do
sha256 arm64_ventura: "4faeb9800a997294538ec6417ac280a37dd28f15fc76bf134d681c9d988ad6a4"
sha256 arm64_monterey: "883aa5ee44bda447bc44c05e99b05be3db6cf909099b0ad86b4b52cd32691181"
sha256 arm64_big_sur: "77056140afe653c2de557bc87cf7692e40737801e3785fae3d5ec8d93b492a67"
sha256 ventura: "7d0b5f7f74d8e2f43e0d7cd6854a6916722482901e58f09e4adb5998dca818e1"
sha256 monterey: "3b22dcfe10ab8e4e3f637edcced509f646f4022bc905236dd845ca28ffe8e1d2"
sha256 big_sur: "921e1731bcd827b989ef6007bab1630dddfb6a0e611d03c1eee0aea2d9cc3f2e"
sha256 x86_64_linux: "819340ec31495b038d94a275e44c4033646b17c41f9799b0a46a2c0a6f7fcc6c"
end
depends_on "cmake" => :build
depends_on "python@3.11" => [:build, :test]
depends_on "pcre"
depends_on "tinyxml2"
uses_from_macos "libxml2"
def python3
which("python3.11")
end
def install
args = std_cmake_args + %W[
-DHAVE_RULES=ON
-DUSE_MATCHCOMPILER=ON
-DUSE_BUNDLED_TINYXML2=OFF
-DENABLE_OSS_FUZZ=OFF
-DPYTHON_EXECUTABLE=#{python3}
]
system "cmake", "-S", ".", "-B", "build", *args
system "cmake", "--build", "build"
system "cmake", "--install", "build"
# Move the python addons to the cppcheck pkgshare folder
(pkgshare/"addons").install Dir.glob("addons/*.py")
end
test do
# Execution test with an input .cpp file
test_cpp_file = testpath/"test.cpp"
test_cpp_file.write <<~EOS
#include <iostream>
using namespace std;
int main()
{
cout << "Hello World!" << endl;
return 0;
}
class Example
{
public:
int GetNumber() const;
explicit Example(int initialNumber);
private:
int number;
};
Example::Example(int initialNumber)
{
number = initialNumber;
}
EOS
system "#{bin}/cppcheck", test_cpp_file
# Test the "out of bounds" check
test_cpp_file_check = testpath/"testcheck.cpp"
test_cpp_file_check.write <<~EOS
int main()
{
char a[10];
a[10] = 0;
return 0;
}
EOS
output = shell_output("#{bin}/cppcheck #{test_cpp_file_check} 2>&1")
assert_match "out of bounds", output
# Test the addon functionality: sampleaddon.py imports the cppcheckdata python
# module and uses it to parse a cppcheck dump into an OOP structure. We then
# check the correct number of detected tokens and function names.
addons_dir = pkgshare/"addons"
cppcheck_module = "#{name}data"
expect_token_count = 55
expect_function_names = "main,GetNumber,Example"
assert_parse_message = "Error: sampleaddon.py: failed: can't parse the #{name} dump."
sample_addon_file = testpath/"sampleaddon.py"
sample_addon_file.write <<~EOS
#!/usr/bin/env #{python3}
"""A simple test addon for #{name}, prints function names and token count"""
import sys
from importlib import machinery, util
# Manually import the '#{cppcheck_module}' module
spec = machinery.PathFinder().find_spec("#{cppcheck_module}", ["#{addons_dir}"])
cpp_check_data = util.module_from_spec(spec)
spec.loader.exec_module(cpp_check_data)
for arg in sys.argv[1:]:
# Parse the dump file generated by #{name}
configKlass = cpp_check_data.parsedump(arg)
if len(configKlass.configurations) == 0:
sys.exit("#{assert_parse_message}") # Parse failure
fConfig = configKlass.configurations[0]
# Pick and join the function names in a string, separated by ','
detected_functions = ','.join(fn.name for fn in fConfig.functions)
detected_token_count = len(fConfig.tokenlist)
# Print the function names on the first line and the token count on the second
print("%s\\n%s" %(detected_functions, detected_token_count))
EOS
system "#{bin}/cppcheck", "--dump", test_cpp_file
test_cpp_file_dump = "#{test_cpp_file}.dump"
assert_predicate testpath/test_cpp_file_dump, :exist?
output = shell_output("#{python3} #{sample_addon_file} #{test_cpp_file_dump}")
assert_match "#{expect_function_names}\n#{expect_token_count}", output
end
end