| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- #include "toolchain/install/busybox_info.h"
- #include <gmock/gmock.h>
- #include <gtest/gtest.h>
- #include <cstdlib>
- #include <filesystem>
- #include <fstream>
- #include <optional>
- #include <system_error>
- #include "common/check.h"
- #include "common/filesystem.h"
- #include "llvm/ADT/ScopeExit.h"
- #include "llvm/Support/FileSystem.h"
- namespace Carbon {
- namespace {
- using ::testing::Eq;
- class BusyboxInfoTest : public ::testing::Test {
- public:
- explicit BusyboxInfoTest()
- : dir_(std::move(*Filesystem::MakeTmpDir())), path_(dir_.path()) {
- // Most tests need the running binary for `MakeBusyboxFile`.
- static int static_for_main_addr;
- running_binary_ = llvm::sys::fs::getMainExecutable("busybox_info_test",
- &static_for_main_addr);
- }
- // Creates a synthetic install tree to test a batch of interactions.
- // Optionally accepts a symlink target for the busybox in the install tree.
- // Returns the input prefix for easy use.
- auto MakeInstallTree(std::filesystem::path prefix,
- std::optional<std::filesystem::path> busybox_target = {})
- -> std::filesystem::path {
- Filesystem::Dir prefix_dir = *dir_.CreateDirectories(prefix);
- Filesystem::Dir lib_carbon = *prefix_dir.CreateDirectories("lib/carbon");
- if (busybox_target) {
- lib_carbon.Symlink("carbon-busybox", busybox_target->native()).Check();
- } else {
- lib_carbon.Symlink("carbon-busybox", running_binary_).Check();
- }
- Filesystem::Dir llvm_bin = *lib_carbon.CreateDirectories("llvm/bin");
- llvm_bin.Symlink("clang++", "clang").Check();
- llvm_bin.Symlink("clang", "../../carbon-busybox").Check();
- Filesystem::Dir bin = *prefix_dir.OpenDir("bin", Filesystem::CreateNew);
- bin.Symlink("carbon", "../lib/carbon/carbon-busybox").Check();
- return path_ / prefix;
- }
- // Helper function to change the working directory and check for an error.
- auto ChangeWorkingDir(std::filesystem::path p) -> void {
- std::error_code ec;
- std::filesystem::current_path(p, ec);
- CARBON_CHECK(!ec, "Error changing working directory: {0}", ec.message());
- }
- // The path to the running binary, `busybox_info_test`. This is provided
- // because `GetExecutablePath` can fall back to it.
- std::string running_binary_;
- Filesystem::RemovingDir dir_;
- std::filesystem::path path_;
- };
- TEST_F(BusyboxInfoTest, Direct) {
- dir_.Symlink("carbon-busybox", running_binary_).Check();
- auto info = GetBusyboxInfo((path_ / "carbon-busybox").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "carbon-busybox"));
- EXPECT_THAT(info->mode, Eq(std::nullopt));
- }
- TEST_F(BusyboxInfoTest, SymlinkInCurrentDirectory) {
- dir_.Symlink("carbon-busybox", running_binary_).Check();
- dir_.Symlink("carbon", "carbon-busybox").Check();
- auto info = GetBusyboxInfo((path_ / "carbon").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "carbon-busybox"));
- EXPECT_THAT(info->mode, Eq(std::nullopt));
- }
- TEST_F(BusyboxInfoTest, SymlinkInCurrentDirectoryWithDot) {
- dir_.Symlink("carbon-busybox", running_binary_).Check();
- dir_.Symlink("carbon", "./carbon-busybox").Check();
- auto info = GetBusyboxInfo((path_ / "carbon").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "./carbon-busybox"));
- EXPECT_THAT(info->mode, Eq(std::nullopt));
- }
- TEST_F(BusyboxInfoTest, ExtraSymlink) {
- dir_.Symlink("carbon-busybox", running_binary_).Check();
- dir_.Symlink("c", "carbon-busybox").Check();
- dir_.Symlink("carbon", "c").Check();
- auto info = GetBusyboxInfo((path_ / "carbon").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "carbon-busybox"));
- EXPECT_THAT(info->mode, Eq(std::nullopt));
- }
- TEST_F(BusyboxInfoTest, OriginalSymlinkNameFormsMode) {
- dir_.Symlink("carbon-busybox", running_binary_).Check();
- dir_.Symlink("carbon", "carbon-busybox").Check();
- dir_.Symlink("clang", "carbon").Check();
- dir_.Symlink("clang++", "clang").Check();
- auto info = GetBusyboxInfo((path_ / "clang").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "carbon-busybox"));
- EXPECT_THAT(info->mode, Eq("clang"));
- info = GetBusyboxInfo((path_ / "clang++").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "carbon-busybox"));
- EXPECT_THAT(info->mode, Eq("clang++"));
- }
- TEST_F(BusyboxInfoTest, BusyboxIsSymlinkToNowhere) {
- dir_.Symlink("carbon-busybox", "nonexistent").Check();
- auto info = GetBusyboxInfo((path_ / "carbon-busybox").c_str());
- ASSERT_FALSE(info.ok());
- EXPECT_THAT(info.error().message(),
- Eq(llvm::formatv("expected carbon-busybox symlink at `{0}`",
- running_binary_)
- .str()));
- }
- TEST_F(BusyboxInfoTest, BusyboxIsWrongFile) {
- // This has the correct name, but it doesn't map back to the running binary
- // and so is ignored.
- dir_.WriteFileFromString("carbon-busybox", "stub").Check();
- auto info = GetBusyboxInfo((path_ / "carbon-busybox").c_str());
- ASSERT_FALSE(info.ok());
- EXPECT_THAT(info.error().message(),
- Eq(llvm::formatv("expected carbon-busybox symlink at `{0}`",
- running_binary_)
- .str()));
- }
- TEST_F(BusyboxInfoTest, RelativeSymlink) {
- Filesystem::Dir d1 = *dir_.OpenDir("dir1", Filesystem::CreateNew);
- d1.Symlink("carbon-busybox", running_binary_).Check();
- Filesystem::Dir d2 = *dir_.OpenDir("dir2", Filesystem::CreateNew);
- d2.Symlink("carbon", "../dir1/carbon-busybox").Check();
- auto info = GetBusyboxInfo((path_ / "dir2/carbon").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "dir2/../dir1/carbon-busybox"));
- EXPECT_THAT(info->mode, Eq(std::nullopt));
- }
- TEST_F(BusyboxInfoTest, AbsoluteSymlink) {
- Filesystem::Dir d1 = *dir_.OpenDir("dir1", Filesystem::CreateNew);
- d1.Symlink("carbon-busybox", running_binary_).Check();
- Filesystem::Dir d2 = *dir_.OpenDir("dir2", Filesystem::CreateNew);
- ASSERT_TRUE(path_.is_absolute());
- d2.Symlink("carbon", (path_ / "dir1/carbon-busybox")).Check();
- auto info = GetBusyboxInfo((path_ / "dir2/carbon").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "dir1/carbon-busybox"));
- EXPECT_THAT(info->mode, Eq(std::nullopt));
- }
- TEST_F(BusyboxInfoTest, NotBusyboxFile) {
- dir_.WriteFileFromString("file", "stub").Check();
- auto info = GetBusyboxInfo((path_ / "file").c_str());
- EXPECT_FALSE(info.ok());
- }
- TEST_F(BusyboxInfoTest, NotBusyboxSymlink) {
- dir_.WriteFileFromString("file", "stub").Check();
- dir_.Symlink("carbon", "file").Check();
- auto info = GetBusyboxInfo((path_ / "carbon").c_str());
- EXPECT_FALSE(info.ok());
- }
- TEST_F(BusyboxInfoTest, LayerSymlinksInstallTree) {
- dir_.Symlink("actual-busybox", running_binary_).Check();
- // Create a facsimile of the install prefix with even the busybox as a
- // symlink. Also include potential relative sibling symlinks like `clang++` to
- // `clang`.
- auto prefix = MakeInstallTree("test_prefix", (path_ / "actual-busybox"));
- auto info = GetBusyboxInfo((prefix / "bin/carbon").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(prefix / "lib/carbon/carbon-busybox"));
- EXPECT_THAT(info->mode, Eq(std::nullopt));
- info = GetBusyboxInfo((prefix / "lib/carbon/llvm/bin/clang").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(prefix / "lib/carbon/carbon-busybox"));
- EXPECT_THAT(info->mode, Eq("clang"));
- info = GetBusyboxInfo((prefix / "lib/carbon/llvm/bin/clang++").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(prefix / "lib/carbon/carbon-busybox"));
- EXPECT_THAT(info->mode, Eq("clang++"));
- }
- TEST_F(BusyboxInfoTest, RunWithinInstallTree) {
- dir_.Symlink("actual-busybox", running_binary_).Check();
- // Create a facsimile of the install prefix with even the busybox as a
- // symlink. Also include potential relative sibling symlinks like `clang++` to
- // `clang`.
- auto prefix = MakeInstallTree("test_prefix", (path_ / "actual-busybox"));
- std::error_code ec;
- auto orig_cwd = std::filesystem::current_path(ec);
- CARBON_CHECK(!ec, "Error getting working directory: {0}", ec.message());
- auto restore_cwd = llvm::scope_exit([&] {
- std::filesystem::current_path(orig_cwd, ec);
- CARBON_CHECK(!ec, "Error changing working directory: {0}", ec.message());
- });
- // First test using a working-directory relative `./bin/carbon` path.
- ChangeWorkingDir(prefix);
- auto info = GetBusyboxInfo("./bin/carbon");
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq("./lib/carbon/carbon-busybox"));
- EXPECT_THAT(info->mode, Eq(std::nullopt));
- // Also test using a working-directory relative `./bin/clang` path.
- ChangeWorkingDir(prefix / "lib/carbon/llvm");
- info = GetBusyboxInfo("./bin/clang");
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq("../carbon-busybox"));
- EXPECT_THAT(info->mode, Eq("clang"));
- // Include redundant `./` components that should be stripped before we give up
- // and use `../` components.
- info = GetBusyboxInfo("./././bin/clang");
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq("../carbon-busybox"));
- EXPECT_THAT(info->mode, Eq("clang"));
- // Also test using a working-directory relative `./llvm/bin/clang` path.
- ChangeWorkingDir(prefix / "lib/carbon");
- info = GetBusyboxInfo("./llvm/bin/clang");
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq("./carbon-busybox"));
- EXPECT_THAT(info->mode, Eq("clang"));
- // Include redundant `./` components at multiple levels, only one of which we
- // end up needing to strip off.
- info = GetBusyboxInfo("././llvm/././bin/clang");
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq("././carbon-busybox"));
- EXPECT_THAT(info->mode, Eq("clang"));
- }
- TEST_F(BusyboxInfoTest, StopSearchAtFirstSymlinkWithRelativeBusybox) {
- // Some install of Carbon under `opt`.
- std::filesystem::path opt_prefix = MakeInstallTree("opt");
- // A second install, but with its symlinks pointing into the `opt` tree rather
- // than at its busybox.
- {
- Filesystem::Dir lib_carbon = *dir_.CreateDirectories("lib/carbon");
- lib_carbon.Symlink("carbon-busybox", running_binary_).Check();
- Filesystem::Dir bin = *dir_.OpenDir("bin", Filesystem::CreateNew);
- bin.Symlink("carbon", "../opt/bin/carbon").Check();
- Filesystem::Dir llvm_bin = *lib_carbon.CreateDirectories("llvm/bin");
- llvm_bin.Symlink("clang", (opt_prefix / "lib/carbon/llvm/bin/clang"))
- .Check();
- }
- // Starting from the second install uses the relative busybox rather than
- // traversing the symlink further.
- auto info = GetBusyboxInfo((path_ / "bin/carbon").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "lib/carbon/carbon-busybox"));
- info = GetBusyboxInfo((path_ / "lib/carbon/llvm/bin/clang").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "lib/carbon/carbon-busybox"));
- }
- TEST_F(BusyboxInfoTest, RejectSymlinkInUnrelatedInstall) {
- // Add two installs of Carbon nested inside each other in a realistic
- // scenario: `/usr` and `/usr/local`.
- MakeInstallTree("usr");
- std::filesystem::path usr_local = MakeInstallTree("usr/local");
- // Now add a stray symlink directly in `.../usr/local` to the local install.
- //
- // This has the interesting property that both of these "work" and find the
- // same busybox but probably wanted to find different ones:
- // - `.../usr/local/../lib/carbon/carbon-busybox`
- // - `.../usr/bin/../lib/carbon/carbon-busybox`
- Filesystem::Dir usr_local_dir = *dir_.OpenDir("usr/local");
- usr_local_dir.Symlink("carbon", "bin/carbon").Check();
- // Check that the busybox doesn't use the relative busybox in this case, and
- // walks the symlink to find the correct installation.
- auto info = GetBusyboxInfo((usr_local / "carbon").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(usr_local / "lib/carbon/carbon-busybox"));
- // Ensure this works even with intervening `.` directory components.
- usr_local_dir.Symlink("carbon2", "bin/././carbon").Check();
- // Check that the busybox doesn't use the relative busybox in this case, and
- // walks the symlink to find the correct installation.
- info = GetBusyboxInfo((usr_local / "carbon2").c_str());
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(usr_local / "lib/carbon/carbon-busybox"));
- }
- TEST_F(BusyboxInfoTest, EnvBinaryPathOverride) {
- // The test should not have this environment variable set.
- ASSERT_THAT(getenv(Argv0OverrideEnv), Eq(nullptr));
- // Set the environment to our actual busybox.
- dir_.Symlink("carbon-busybox", running_binary_).Check();
- setenv(Argv0OverrideEnv, (path_ / "carbon-busybox").c_str(), /*overwrite=*/1);
- auto info = GetBusyboxInfo("/some/nonexistent/path");
- if (getenv(Argv0OverrideEnv)) {
- unsetenv(Argv0OverrideEnv);
- ADD_FAILURE() << "GetBusyboxInfo should unset Argv0OverrideEnv";
- }
- ASSERT_TRUE(info.ok()) << info.error();
- EXPECT_THAT(info->bin_path, Eq(path_ / "carbon-busybox"));
- EXPECT_THAT(info->mode, Eq(std::nullopt));
- // Make sure that we cleaned up the environment afterward.
- EXPECT_THAT(getenv(Argv0OverrideEnv), Eq(nullptr));
- }
- } // namespace
- } // namespace Carbon
|