Files
clice/tests/unit/command/command_tests.cpp

580 lines
21 KiB
C++

#include "test/test.h"
#include "command/argument_parser.h"
#include "command/command.h"
#include "support/filesystem.h"
#include "llvm/Support/raw_ostream.h"
namespace clice::testing {
namespace {
using namespace std::literals;
CommandOptions quiet_options() {
CommandOptions options;
options.suppress_logging = true;
return options;
}
#define EXPECT_CONTAINS(haystack, needle) EXPECT_TRUE(llvm::StringRef(haystack).contains(needle))
#define EXPECT_NOT_CONTAINS(haystack, needle) \
EXPECT_FALSE(llvm::StringRef(haystack).contains(needle))
TEST_SUITE(Command) {
void expect_strip(llvm::StringRef argv, llvm::StringRef result) {
CompilationDatabase database;
llvm::StringRef file = "main.cpp";
database.add_command("fake/", file, argv);
ASSERT_EQ(result, print_argv(database.lookup(file, quiet_options()).front().arguments));
};
TEST_CASE(DefaultFilters) {
/// Filter -c, -o and input file.
expect_strip("g++ main.cpp", "g++ main.cpp");
expect_strip("clang++ -c main.cpp", "clang++ main.cpp");
expect_strip("clang++ -o main.o main.cpp", "clang++ main.cpp");
expect_strip("clang++ -c -o main.o main.cpp", "clang++ main.cpp");
expect_strip("cl.exe /c /Fomain.cpp.o main.cpp", "cl.exe main.cpp");
/// Filter PCH related.
/// CMake
expect_strip("g++ -std=gnu++20 -Winvalid-pch -include cmake_pch.hxx -o main.cpp.o -c main.cpp",
"g++ -std=gnu++20 -Winvalid-pch -include cmake_pch.hxx main.cpp");
expect_strip(
"clang++ -Winvalid-pch -Xclang -include-pch -Xclang cmake_pch.hxx.pch -Xclang -include -Xclang cmake_pch.hxx -o main.cpp.o -c main.cpp",
"clang++ -Winvalid-pch -Xclang -include -Xclang cmake_pch.hxx main.cpp");
expect_strip("cl.exe /Yufoo.h /FIfoo.h /Fpfoo.h_v143.pch /c /Fomain.cpp.o main.cpp",
"cl.exe -include foo.h main.cpp");
/// TODO: Test more commands from other build system.
};
TEST_CASE(Reuse) {
CompilationDatabase database;
database.add_command("fake", "test.cpp", "clang++ -std=c++23 test.cpp"sv);
database.add_command("fake", "test2.cpp", "clang++ -std=c++23 test2.cpp"sv);
auto options = quiet_options();
auto command1 = database.lookup("test.cpp", options).front().arguments;
auto command2 = database.lookup("test2.cpp", options).front().arguments;
ASSERT_EQ(command1.size(), 3U);
ASSERT_EQ(command2.size(), 3U);
ASSERT_EQ(command1[0], "clang++"sv);
ASSERT_EQ(command1[1], "-std=c++23"sv);
ASSERT_EQ(command1[2], "test.cpp"sv);
ASSERT_EQ(command1[0], command2[0]);
ASSERT_EQ(command1[1], command2[1]);
ASSERT_EQ(command2[2], "test2.cpp"sv);
};
TEST_CASE(RemoveAppend) {
llvm::SmallVector args = {
"clang++",
"--output=main.o",
"-D",
"A",
"-D",
"B=0",
"main.cpp",
};
CompilationDatabase database;
database.add_command("/fake", "main.cpp", args);
auto options = quiet_options();
llvm::SmallVector<std::string> remove;
llvm::SmallVector<std::string> append;
remove = {"-DA"};
options.remove = remove;
auto result = database.lookup("main.cpp", options).front().arguments;
ASSERT_EQ(print_argv(result), "clang++ -D B=0 main.cpp");
remove = {"-D", "A"};
options.remove = remove;
result = database.lookup("main.cpp", options).front().arguments;
ASSERT_EQ(print_argv(result), "clang++ -D B=0 main.cpp");
remove = {"-DA", "-D", "B=0"};
options.remove = remove;
result = database.lookup("main.cpp", options).front().arguments;
ASSERT_EQ(print_argv(result), "clang++ main.cpp");
remove = {"-D*"};
options.remove = remove;
result = database.lookup("main.cpp", options).front().arguments;
ASSERT_EQ(print_argv(result), "clang++ main.cpp");
remove = {"-D", "*"};
options.remove = remove;
result = database.lookup("main.cpp", options).front().arguments;
ASSERT_EQ(print_argv(result), "clang++ main.cpp");
append = {"-D", "C"};
options.append = append;
result = database.lookup("main.cpp", options).front().arguments;
ASSERT_EQ(print_argv(result), "clang++ -D C main.cpp");
};
TEST_CASE(DefaultFallback) {
/// Lookup for a file not in the CDB should synthesize a default command.
CompilationDatabase database;
/// C++ files get "clang++ -std=c++20 <file>".
auto cpp_results = database.lookup("unknown.cpp");
ASSERT_EQ(cpp_results.size(), 1U);
auto& cpp_ctx = cpp_results.front();
ASSERT_EQ(cpp_ctx.arguments.size(), 3U);
ASSERT_EQ(cpp_ctx.arguments[0], "clang++"sv);
ASSERT_EQ(cpp_ctx.arguments[1], "-std=c++20"sv);
ASSERT_EQ(cpp_ctx.arguments[2], "unknown.cpp"sv);
/// .hpp files also get C++ default.
auto hpp_results = database.lookup("header.hpp");
ASSERT_EQ(hpp_results.front().arguments.size(), 3U);
ASSERT_EQ(hpp_results.front().arguments[0], "clang++"sv);
/// .cc files also get C++ default.
auto cc_results = database.lookup("file.cc");
ASSERT_EQ(cc_results.front().arguments.size(), 3U);
ASSERT_EQ(cc_results.front().arguments[0], "clang++"sv);
/// C files get "clang <file>".
auto c_results = database.lookup("unknown.c");
ASSERT_EQ(c_results.size(), 1U);
auto& c_ctx = c_results.front();
ASSERT_EQ(c_ctx.arguments.size(), 2U);
ASSERT_EQ(c_ctx.arguments[0], "clang"sv);
ASSERT_EQ(c_ctx.arguments[1], "unknown.c"sv);
/// Other extensions also get plain clang.
auto h_results = database.lookup("foo.h");
ASSERT_EQ(h_results.front().arguments.size(), 2U);
ASSERT_EQ(h_results.front().arguments[0], "clang"sv);
};
TEST_CASE(MultiCommand) {
/// A file can have multiple compilation commands (e.g. different configs).
CompilationDatabase database;
database.add_command("fake", "main.cpp", "clang++ -std=c++17 main.cpp"sv);
database.add_command("fake", "main.cpp", "clang++ -std=c++20 main.cpp"sv);
database.add_command("fake", "other.cpp", "clang++ -std=c++23 other.cpp"sv);
auto options = quiet_options();
auto results = database.lookup("main.cpp", options);
ASSERT_EQ(results.size(), 2U);
/// Both commands are present (order depends on insert position).
bool has_17 = false, has_20 = false;
for(auto& ctx: results) {
auto argv = print_argv(ctx.arguments);
if(llvm::StringRef(argv).contains("-std=c++17"))
has_17 = true;
if(llvm::StringRef(argv).contains("-std=c++20"))
has_20 = true;
}
EXPECT_TRUE(has_17);
EXPECT_TRUE(has_20);
/// other.cpp has only one.
auto other = database.lookup("other.cpp", options);
ASSERT_EQ(other.size(), 1U);
};
TEST_CASE(CodegenFilter) {
/// Codegen-only options should be stripped from the canonical command.
CompilationDatabase database;
database.add_command(
"fake",
"main.cpp",
"clang++ -std=c++20 -fPIC -fno-omit-frame-pointer -fstack-protector-strong " "-fdata-sections -ffunction-sections -flto -fcolor-diagnostics -g main.cpp"sv);
auto result = database.lookup("main.cpp", quiet_options()).front().arguments;
auto argv = print_argv(result);
/// -std=c++20 must survive (semantic).
EXPECT_CONTAINS(argv, "-std=c++20");
/// All codegen flags must be stripped.
EXPECT_NOT_CONTAINS(argv, "-fPIC");
EXPECT_NOT_CONTAINS(argv, "-fno-omit-frame-pointer");
EXPECT_NOT_CONTAINS(argv, "-fstack-protector");
EXPECT_NOT_CONTAINS(argv, "-fdata-sections");
EXPECT_NOT_CONTAINS(argv, "-ffunction-sections");
EXPECT_NOT_CONTAINS(argv, "-flto");
EXPECT_NOT_CONTAINS(argv, "-fcolor-diagnostics");
EXPECT_NOT_CONTAINS(argv, "-g");
};
TEST_CASE(DependencyScanFilter) {
/// Dependency scan options should be stripped.
CompilationDatabase database;
database.add_command("fake",
"main.cpp",
"clang++ -std=c++20 -MD -MF main.d -MT main.o main.cpp"sv);
auto result = database.lookup("main.cpp", quiet_options()).front().arguments;
auto argv = print_argv(result);
EXPECT_CONTAINS(argv, "-std=c++20");
EXPECT_NOT_CONTAINS(argv, "-MD");
EXPECT_NOT_CONTAINS(argv, "-MF");
EXPECT_NOT_CONTAINS(argv, "-MT");
EXPECT_NOT_CONTAINS(argv, "main.d");
};
TEST_CASE(ModuleFilter) {
/// Module-related options should be stripped.
expect_strip("clang++ -std=c++20 -fmodule-file=mod.pcm main.cpp",
"clang++ -std=c++20 main.cpp");
expect_strip("clang++ -std=c++20 -fprebuilt-module-path=/tmp main.cpp",
"clang++ -std=c++20 main.cpp");
};
TEST_CASE(UserContentClassification) {
/// -D, -U, -include go to per-file patch; -std=, -W go to canonical.
/// Files with different -D but same -std/-W share canonical.
CompilationDatabase database;
database.add_command("fake", "a.cpp", "clang++ -std=c++20 -Wall -DA=1 -DFOO a.cpp"sv);
database.add_command("fake", "b.cpp", "clang++ -std=c++20 -Wall -DB=2 b.cpp"sv);
auto options = quiet_options();
auto a_argv = print_argv(database.lookup("a.cpp", options).front().arguments);
auto b_argv = print_argv(database.lookup("b.cpp", options).front().arguments);
/// Both must contain canonical flags.
EXPECT_CONTAINS(a_argv, "-std=c++20");
EXPECT_CONTAINS(a_argv, "-Wall");
EXPECT_CONTAINS(b_argv, "-std=c++20");
EXPECT_CONTAINS(b_argv, "-Wall");
/// a.cpp has its own defines.
EXPECT_CONTAINS(a_argv, "-D");
EXPECT_CONTAINS(a_argv, "A=1");
EXPECT_CONTAINS(a_argv, "FOO");
/// b.cpp has its own defines.
EXPECT_CONTAINS(b_argv, "-D");
EXPECT_CONTAINS(b_argv, "B=2");
/// Cross check: a.cpp should not have B=2, b.cpp should not have A=1.
EXPECT_NOT_CONTAINS(a_argv, "B=2");
EXPECT_NOT_CONTAINS(b_argv, "A=1");
};
TEST_CASE(IncludePathAbsolutize) {
/// Relative include paths should be absolutized against the directory.
CompilationDatabase database;
database.add_command("/project/build",
"main.cpp",
"clang++ -Iinclude -isystem sys/inc -iquote ../src main.cpp"sv);
auto result = database.lookup("main.cpp", quiet_options()).front().arguments;
/// Check each argument individually with separator normalization
/// (print_argv escapes backslashes, breaking convert_to_slash on Windows).
auto has_path = [](llvm::ArrayRef<const char*> args, llvm::StringRef needle) {
for(auto* arg: args) {
if(path::convert_to_slash(arg).find(needle.str()) != std::string::npos)
return true;
}
return false;
};
/// Relative paths must be resolved against /project/build.
EXPECT_TRUE(has_path(result, "/project/build/include"));
EXPECT_TRUE(has_path(result, "/project/build/sys/inc"));
/// ../src relative to /project/build → /project/src (or /project/build/../src)
EXPECT_TRUE(has_path(result, "/project/"));
/// Absolute paths should be kept as-is.
CompilationDatabase database2;
database2.add_command("/project/build", "main.cpp", "clang++ -I/usr/include main.cpp"sv);
auto result2 = database2.lookup("main.cpp", quiet_options()).front().arguments;
EXPECT_TRUE(has_path(result2, "/usr/include"));
};
TEST_CASE(SemanticOptionsPreserved) {
/// Flags that affect semantics must survive.
expect_strip("clang++ -std=c++20 -fno-exceptions -fno-rtti -pedantic main.cpp",
"clang++ -std=c++20 -fno-exceptions -fno-rtti -pedantic main.cpp");
expect_strip("clang++ -std=c++20 -Wall -Werror main.cpp",
"clang++ -std=c++20 -Wall -Werror main.cpp");
};
TEST_CASE(LookupSearchConfig) {
CompilationDatabase database;
database.add_command(
"/project",
"main.cpp",
"clang++ -std=c++20 -I/usr/include -isystem /usr/local/include main.cpp"sv);
ASSERT_FALSE(database.has_cached_configs());
auto options = quiet_options();
auto config = database.lookup_search_config("main.cpp", options);
/// Should have search dirs from the command.
EXPECT_FALSE(config.dirs.empty());
/// Second call should hit cache.
EXPECT_TRUE(database.has_cached_configs());
auto config2 = database.lookup_search_config("main.cpp", options);
ASSERT_EQ(config.dirs.size(), config2.dirs.size());
};
TEST_CASE(ResolvePath) {
CompilationDatabase database;
database.add_command("fake", "test/main.cpp", "clang++ test/main.cpp"sv);
/// After add_command, lookup should work and resolve_path via the file in arguments.
auto result = database.lookup("test/main.cpp", quiet_options()).front().arguments;
/// The last argument is the file, resolved from PathPool.
ASSERT_EQ(result.back(), "test/main.cpp"sv);
};
TEST_CASE(MoveSemantics) {
CompilationDatabase db1;
db1.add_command("fake", "main.cpp", "clang++ -std=c++23 main.cpp"sv);
/// Move construct.
CompilationDatabase db2 = std::move(db1);
auto options = quiet_options();
auto result = db2.lookup("main.cpp", options).front().arguments;
ASSERT_EQ(result.size(), 3U);
ASSERT_EQ(result[1], "-std=c++23"sv);
/// Move assign.
CompilationDatabase db3;
db3 = std::move(db2);
result = db3.lookup("main.cpp", options).front().arguments;
ASSERT_EQ(result.size(), 3U);
ASSERT_EQ(result[1], "-std=c++23"sv);
};
/// Write JSON to a temp file, load into a CDB, remove the file.
/// Returns the number of entries loaded.
std::size_t load_json(CompilationDatabase& database, llvm::StringRef json) {
auto path = fs::createTemporaryFile("cdb", "json");
if(!path)
return 0;
{
std::error_code ec;
llvm::raw_fd_ostream out(*path, ec);
if(ec)
return 0;
out << json;
}
auto count = database.load(*path);
llvm::sys::fs::remove(*path);
return count;
}
TEST_CASE(LoadMixedFormats) {
/// "arguments" array and "command" string can coexist in the same CDB.
/// Use relative file paths so that the test works on both Linux and Windows
/// (paths like "/src/a.cpp" are not absolute on Windows — no drive letter).
CompilationDatabase database;
auto count = load_json(database, R"([
{"directory": "/build", "file": "a.cpp",
"arguments": ["clang++", "-std=c++20", "a.cpp"]},
{"directory": "/build", "file": "b.cpp",
"command": "clang++ -std=c++23 b.cpp"}
])");
ASSERT_EQ(count, 2U);
auto options = quiet_options();
auto a = database.lookup(path::join("/build", "a.cpp"), options);
ASSERT_EQ(a.size(), 1U);
EXPECT_CONTAINS(print_argv(a.front().arguments), "-std=c++20");
auto b = database.lookup(path::join("/build", "b.cpp"), options);
ASSERT_EQ(b.size(), 1U);
EXPECT_CONTAINS(print_argv(b.front().arguments), "-std=c++23");
};
TEST_CASE(LoadErrorRecovery) {
/// Bad entries should be skipped; good entries still load.
CompilationDatabase database;
auto count = load_json(database, R"([
{"file": "no_dir.cpp",
"arguments": ["clang++", "no_dir.cpp"]},
{"directory": "/build",
"arguments": ["clang++", "no_file.cpp"]},
{"directory": "/build", "file": "no_args.cpp"},
{"directory": "/build", "file": "good.cpp",
"arguments": ["clang++", "-std=c++20", "good.cpp"]},
42,
{"directory": "/build", "file": "also_good.cpp",
"command": "clang++ -Wall also_good.cpp"}
])");
/// Only the two valid entries should survive.
ASSERT_EQ(count, 2U);
auto options = quiet_options();
auto good = database.lookup(path::join("/build", "good.cpp"), options);
ASSERT_EQ(good.size(), 1U);
EXPECT_CONTAINS(print_argv(good.front().arguments), "-std=c++20");
auto also = database.lookup(path::join("/build", "also_good.cpp"), options);
ASSERT_EQ(also.size(), 1U);
EXPECT_CONTAINS(print_argv(also.front().arguments), "-Wall");
};
TEST_CASE(LoadEmptyCommand) {
/// Whitespace-only or empty "command" should not crash.
CompilationDatabase database;
auto count = load_json(database, R"([
{"directory": "/build", "file": "empty.cpp", "command": ""},
{"directory": "/build", "file": "spaces.cpp", "command": " "},
{"directory": "/build", "file": "ok.cpp",
"command": "clang++ -std=c++20 ok.cpp"}
])");
/// Only the valid entry survives.
ASSERT_EQ(count, 1U);
auto ok = database.lookup(path::join("/build", "ok.cpp"), quiet_options());
ASSERT_EQ(ok.size(), 1U);
EXPECT_CONTAINS(print_argv(ok.front().arguments), "-std=c++20");
};
TEST_CASE(LoadReload) {
/// Second load() replaces all entries from the first.
CompilationDatabase database;
auto file_a = path::join("/build", "a.cpp");
auto file_b = path::join("/build", "b.cpp");
load_json(database, R"([
{"directory": "/build", "file": "a.cpp",
"arguments": ["clang++", "-std=c++17", "a.cpp"]}
])");
auto options = quiet_options();
auto a = database.lookup(file_a, options);
ASSERT_EQ(a.size(), 1U);
EXPECT_CONTAINS(print_argv(a.front().arguments), "-std=c++17");
/// Reload with different content.
auto count = load_json(database, R"([
{"directory": "/build", "file": "b.cpp",
"arguments": ["clang++", "-std=c++23", "b.cpp"]}
])");
ASSERT_EQ(count, 1U);
/// Old entry gone (falls back to default).
auto a2 = database.lookup(file_a, options);
ASSERT_EQ(a2.size(), 1U);
EXPECT_NOT_CONTAINS(print_argv(a2.front().arguments), "-std=c++17");
/// New entry present.
auto b = database.lookup(file_b, options);
ASSERT_EQ(b.size(), 1U);
EXPECT_CONTAINS(print_argv(b.front().arguments), "-std=c++23");
};
TEST_CASE(LoadCommandQuoting) {
/// "command" string with spaces in paths and quoted defines.
CompilationDatabase database;
auto count = load_json(database, R"([
{"directory": "/build", "file": "main.cpp",
"command": "clang++ -std=c++20 \"-DMSG=hello world\" -I\"/path with spaces\" main.cpp"}
])");
ASSERT_EQ(count, 1U);
auto result = database.lookup(path::join("/build", "main.cpp"), quiet_options());
ASSERT_EQ(result.size(), 1U);
auto argv = print_argv(result.front().arguments);
/// The define and include path should be present after shell tokenization.
EXPECT_CONTAINS(argv, "hello world");
EXPECT_CONTAINS(argv, "/path with spaces");
};
TEST_CASE(LoadRelativePath) {
/// load() should resolve relative file paths against directory.
CompilationDatabase database;
auto count = load_json(database, R"([
{"directory": "/project/build", "file": "src/main.cpp",
"arguments": ["clang++", "-std=c++20", "src/main.cpp"]},
{"directory": "/other/build", "file": "src/main.cpp",
"arguments": ["clang++", "-std=c++17", "src/main.cpp"]}
])");
ASSERT_EQ(count, 2U);
auto options = quiet_options();
/// Lookup by the resolved absolute path (use path::join for correct separator).
auto results = database.lookup(path::join("/project/build", "src/main.cpp"), options);
ASSERT_EQ(results.size(), 1U);
EXPECT_CONTAINS(print_argv(results.front().arguments), "-std=c++20");
auto results2 = database.lookup(path::join("/other/build", "src/main.cpp"), options);
ASSERT_EQ(results2.size(), 1U);
EXPECT_CONTAINS(print_argv(results2.front().arguments), "-std=c++17");
/// Relative path lookup should not match (different path_id).
auto results3 = database.lookup("src/main.cpp", options);
ASSERT_EQ(results3.size(), 1U);
/// Falls back to default command since no match.
EXPECT_CONTAINS(print_argv(results3.front().arguments), "clang");
};
TEST_CASE(Module) {
// TODO: revisit module command handling.
}
TEST_CASE(ResourceDir) {
// When query_toolchain is enabled, resource dir is injected automatically.
CompilationDatabase database;
database.add_command("/fake", "main.cpp", "clang++ -std=c++23 test.cpp"sv);
// Without query_toolchain, no resource dir injection.
auto args_no_tc = database.lookup("main.cpp").front().arguments;
ASSERT_EQ(args_no_tc.size(), 3U);
ASSERT_EQ(args_no_tc[0], "clang++"sv);
ASSERT_EQ(args_no_tc[1], "-std=c++23"sv);
ASSERT_EQ(args_no_tc[2], "main.cpp"sv);
// With query_toolchain, resource dir is present in the result.
auto args_tc = database.lookup("main.cpp", {.query_toolchain = true}).front().arguments;
bool has_resource_dir = false;
for(size_t i = 0; i + 1 < args_tc.size(); ++i) {
if(args_tc[i] == "-resource-dir"sv) {
EXPECT_EQ(llvm::StringRef(args_tc[i + 1]), resource_dir());
has_resource_dir = true;
break;
}
}
if(resource_dir().empty()) {
EXPECT_FALSE(has_resource_dir);
} else {
EXPECT_TRUE(has_resource_dir);
}
};
}; // TEST_SUITE(Command)
} // namespace
} // namespace clice::testing