|  | // Copyright 2014 The Bazel Authors. All rights reserved. | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //    http://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  | #include <stdio.h> | 
|  | #include <string.h> | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <map> | 
|  | #include <memory>  // unique_ptr | 
|  | #include <thread>  // NOLINT (to silence Google-internal linter) | 
|  | #include <vector> | 
|  |  | 
|  | #include "src/main/cpp/util/file.h" | 
|  | #include "src/main/cpp/util/path.h" | 
|  | #include "src/main/cpp/util/path_platform.h" | 
|  | #include "src/test/cpp/util/test_util.h" | 
|  | #include "googletest/include/gtest/gtest.h" | 
|  |  | 
|  | namespace blaze_util { | 
|  |  | 
|  | using std::string; | 
|  |  | 
|  | TEST(FileTest, TestSingleThreadedPipe) { | 
|  | std::unique_ptr<IPipe> pipe(CreatePipe()); | 
|  | char buffer[50] = {0}; | 
|  | ASSERT_TRUE(pipe->Send("hello", 5)); | 
|  | int error = -1; | 
|  | ASSERT_EQ(3, pipe->Receive(buffer, 3, &error)); | 
|  | ASSERT_TRUE(pipe->Send(" world", 6)); | 
|  | ASSERT_EQ(5, pipe->Receive(buffer + 3, 5, &error)); | 
|  | ASSERT_EQ(IPipe::SUCCESS, error); | 
|  | ASSERT_EQ(3, pipe->Receive(buffer + 8, 40, &error)); | 
|  | ASSERT_EQ(IPipe::SUCCESS, error); | 
|  | ASSERT_EQ(0, strncmp(buffer, "hello world", 11)); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, TestMultiThreadedPipe) { | 
|  | std::unique_ptr<IPipe> pipe(CreatePipe()); | 
|  | char buffer[50] = {0}; | 
|  | std::thread writer_thread([&pipe]() { | 
|  | ASSERT_TRUE(pipe->Send("hello", 5)); | 
|  | ASSERT_TRUE(pipe->Send(" world", 6)); | 
|  | }); | 
|  |  | 
|  | // Wait for all data to be fully written to the pipe. | 
|  | writer_thread.join(); | 
|  |  | 
|  | int error = -1; | 
|  | ASSERT_EQ(3, pipe->Receive(buffer, 3, &error)); | 
|  | ASSERT_EQ(IPipe::SUCCESS, error); | 
|  | ASSERT_EQ(5, pipe->Receive(buffer + 3, 5, &error)); | 
|  | ASSERT_EQ(IPipe::SUCCESS, error); | 
|  | ASSERT_EQ(3, pipe->Receive(buffer + 8, 40, &error)); | 
|  | ASSERT_EQ(IPipe::SUCCESS, error); | 
|  | ASSERT_EQ(0, strncmp(buffer, "hello world", 11)); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, TestReadFileIntoString) { | 
|  | const char* tempdir = getenv("TEST_TMPDIR"); | 
|  | ASSERT_NE(nullptr, tempdir); | 
|  | ASSERT_NE(0, tempdir[0]); | 
|  |  | 
|  | std::string filename(JoinPath(tempdir, "test.readfile")); | 
|  | AutoFileStream fh(fopen(filename.c_str(), "wt")); | 
|  | EXPECT_TRUE(fh.IsOpen()); | 
|  | ASSERT_EQ(size_t(11), fwrite("hello world", 1, 11, fh)); | 
|  | fh.Close(); | 
|  |  | 
|  | std::string actual; | 
|  | ASSERT_TRUE(ReadFile(filename, &actual)); | 
|  | ASSERT_EQ(std::string("hello world"), actual); | 
|  |  | 
|  | ASSERT_TRUE(ReadFile(filename, &actual, 5)); | 
|  | ASSERT_EQ(std::string("hello"), actual); | 
|  |  | 
|  | ASSERT_TRUE(ReadFile("/dev/null", &actual, 42)); | 
|  | ASSERT_EQ(std::string(""), actual); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, TestReadFileIntoBuffer) { | 
|  | const char* tempdir = getenv("TEST_TMPDIR"); | 
|  | EXPECT_NE(nullptr, tempdir); | 
|  | EXPECT_NE(0, tempdir[0]); | 
|  |  | 
|  | std::string filename(JoinPath(tempdir, "test.readfile")); | 
|  | AutoFileStream fh(fopen(filename.c_str(), "wt")); | 
|  | EXPECT_TRUE(fh.IsOpen()); | 
|  | EXPECT_EQ(size_t(11), fwrite("hello world", 1, 11, fh)); | 
|  | fh.Close(); | 
|  |  | 
|  | char buffer[30]; | 
|  | memset(buffer, 0, 30); | 
|  | ASSERT_TRUE(ReadFile(filename, buffer, 5)); | 
|  | ASSERT_EQ(string("hello"), string(buffer)); | 
|  |  | 
|  | memset(buffer, 0, 30); | 
|  | ASSERT_TRUE(ReadFile(filename, buffer, 30)); | 
|  | ASSERT_EQ(string("hello world"), string(buffer)); | 
|  |  | 
|  | buffer[0] = 'x'; | 
|  | ASSERT_TRUE(ReadFile("/dev/null", buffer, 42)); | 
|  | ASSERT_EQ('x', buffer[0]); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, TestWriteFile) { | 
|  | const char* tempdir = getenv("TEST_TMPDIR"); | 
|  | ASSERT_NE(nullptr, tempdir); | 
|  | ASSERT_NE(0, tempdir[0]); | 
|  |  | 
|  | std::string filename(JoinPath(tempdir, "test.writefile")); | 
|  |  | 
|  | ASSERT_TRUE(WriteFile("hello", 3, filename)); | 
|  |  | 
|  | char buf[6] = {0}; | 
|  | AutoFileStream fh(fopen(filename.c_str(), "rt")); | 
|  | EXPECT_TRUE(fh.IsOpen()); | 
|  | fflush(fh); | 
|  | ASSERT_EQ(size_t(3), fread(buf, 1, 5, fh)); | 
|  | fh.Close(); | 
|  | ASSERT_EQ(std::string(buf), std::string("hel")); | 
|  |  | 
|  | ASSERT_TRUE(WriteFile("hello", 5, filename)); | 
|  | fh = fopen(filename.c_str(), "rt"); | 
|  | EXPECT_TRUE(fh.IsOpen()); | 
|  | memset(buf, 0, 6); | 
|  | ASSERT_EQ(size_t(5), fread(buf, 1, 5, fh)); | 
|  | fh.Close(); | 
|  | ASSERT_EQ(std::string(buf), std::string("hello")); | 
|  |  | 
|  | ASSERT_TRUE(WriteFile("hello", 5, "/dev/null")); | 
|  | ASSERT_EQ(0, remove(filename.c_str())); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, TestLargeFileWrite) { | 
|  | // Buffer over the write limit (2,147,479,552 for Linux, INT32_MAX for MacOS). | 
|  | const size_t size = 4000000000; | 
|  | std::unique_ptr<char[]> buffer(new char[size]); | 
|  | std::fill(buffer.get(), buffer.get() + size, '\0'); | 
|  |  | 
|  | ASSERT_TRUE(WriteFile(buffer.get(), size, "/dev/null")); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, TestMtimeHandling) { | 
|  | const char* tempdir_cstr = getenv("TEST_TMPDIR"); | 
|  | ASSERT_NE(tempdir_cstr, nullptr); | 
|  | ASSERT_NE(tempdir_cstr[0], 0); | 
|  | Path tempdir(tempdir_cstr); | 
|  |  | 
|  | std::unique_ptr<IFileMtime> mtime(CreateFileMtime()); | 
|  | // Assert that a directory is always untampered with. (We do | 
|  | // not care about directories' mtimes.) | 
|  | ASSERT_TRUE(mtime->IsUntampered(tempdir)); | 
|  | // Create a new file, assert its mtime is not in the future. | 
|  | Path file = tempdir.GetRelative("foo.txt"); | 
|  | ASSERT_TRUE(WriteFile("hello", 5, file)); | 
|  | ASSERT_FALSE(mtime->IsUntampered(file)); | 
|  | // Set the file's mtime to the future, assert that it's so. | 
|  | ASSERT_TRUE(mtime->SetToDistantFuture(file)); | 
|  | ASSERT_TRUE(mtime->IsUntampered(file)); | 
|  | // Overwrite the file, resetting its mtime, assert that | 
|  | // IsUntampered notices. | 
|  | ASSERT_TRUE(WriteFile("world", 5, file)); | 
|  | ASSERT_FALSE(mtime->IsUntampered(file)); | 
|  | // Set it to the future again so we can reset it using SetToNow. | 
|  | ASSERT_TRUE(mtime->SetToDistantFuture(file)); | 
|  | ASSERT_TRUE(mtime->IsUntampered(file)); | 
|  | // Assert that SetToNow resets the timestamp. | 
|  | ASSERT_TRUE(mtime->SetToNow(file)); | 
|  | ASSERT_FALSE(mtime->IsUntampered(file)); | 
|  | // Delete the file and assert that we can no longer set or query its mtime. | 
|  | ASSERT_TRUE(UnlinkPath(file)); | 
|  | ASSERT_FALSE(mtime->SetToNow(file)); | 
|  | ASSERT_FALSE(mtime->SetToDistantFuture(file)); | 
|  | ASSERT_FALSE(mtime->IsUntampered(file)); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, TestCreateTempDir) { | 
|  | const char* tempdir_cstr = getenv("TEST_TMPDIR"); | 
|  | EXPECT_NE(tempdir_cstr, nullptr); | 
|  | EXPECT_NE(tempdir_cstr[0], 0); | 
|  | string tempdir(tempdir_cstr); | 
|  | string tmpdir(tempdir_cstr); | 
|  |  | 
|  | string prefix_in_existing_dir(JoinPath(tempdir, "foo.")); | 
|  | string result_in_existing_dir(CreateTempDir(prefix_in_existing_dir)); | 
|  | ASSERT_NE(result_in_existing_dir, prefix_in_existing_dir); | 
|  | ASSERT_EQ(0, result_in_existing_dir.find(prefix_in_existing_dir)); | 
|  | EXPECT_TRUE(PathExists(result_in_existing_dir)); | 
|  |  | 
|  | string base_dir(JoinPath(tempdir, "doesntexistyet")); | 
|  | ASSERT_FALSE(PathExists(base_dir)); | 
|  | string prefix_in_new_dir(JoinPath(base_dir, "foo.")); | 
|  | string result_in_new_dir(CreateTempDir(prefix_in_new_dir)); | 
|  | ASSERT_NE(result_in_new_dir, prefix_in_new_dir); | 
|  | ASSERT_EQ(0, result_in_new_dir.find(prefix_in_new_dir)); | 
|  | EXPECT_TRUE(PathExists(result_in_new_dir)); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, TestRenameDirectory) { | 
|  | const char* tempdir_cstr = getenv("TEST_TMPDIR"); | 
|  | EXPECT_NE(tempdir_cstr, nullptr); | 
|  | EXPECT_NE(tempdir_cstr[0], 0); | 
|  | string tempdir(tempdir_cstr); | 
|  |  | 
|  | string dir1(JoinPath(tempdir, "test_rename_dir/dir1")); | 
|  | string dir2(JoinPath(tempdir, "test_rename_dir/dir2")); | 
|  | EXPECT_TRUE(MakeDirectories(dir1, 0700)); | 
|  | string file1(JoinPath(dir1, "file1.txt")); | 
|  | EXPECT_TRUE(WriteFile("hello", 5, file1)); | 
|  |  | 
|  | ASSERT_EQ(RenameDirectory(dir1, dir2), kRenameDirectorySuccess); | 
|  | ASSERT_EQ(RenameDirectory(dir1, dir2), kRenameDirectoryFailureOtherError); | 
|  | EXPECT_TRUE(MakeDirectories(dir1, 0700)); | 
|  | EXPECT_TRUE(WriteFile("hello", 5, file1)); | 
|  | ASSERT_EQ(RenameDirectory(dir2, dir1), kRenameDirectoryFailureNotEmpty); | 
|  | } | 
|  |  | 
|  | class CollectingDirectoryEntryConsumer : public DirectoryEntryConsumer { | 
|  | public: | 
|  | CollectingDirectoryEntryConsumer(const string& _rootname) | 
|  | : rootname(_rootname) {} | 
|  |  | 
|  | void Consume(const string& name, bool is_directory) override { | 
|  | // Strip the path prefix up to the `rootname` to ease testing on all | 
|  | // platforms. | 
|  | size_t index = name.rfind(rootname); | 
|  | string key = (index == string::npos) ? name : name.substr(index); | 
|  | // Replace backslashes with forward slashes (necessary on Windows only). | 
|  | std::replace(key.begin(), key.end(), '\\', '/'); | 
|  | entries[key] = is_directory; | 
|  | } | 
|  |  | 
|  | const string rootname; | 
|  | std::map<string, bool> entries; | 
|  | }; | 
|  |  | 
|  | TEST(FileTest, ForEachDirectoryEntryTest) { | 
|  | string tmpdir(getenv("TEST_TMPDIR")); | 
|  | EXPECT_FALSE(tmpdir.empty()); | 
|  | // Create a directory structure: | 
|  | //   $TEST_TMPDIR/ | 
|  | //      foo/ | 
|  | //        bar/ | 
|  | //          file3.txt | 
|  | //        file1.txt | 
|  | //        file2.txt | 
|  | string rootdir(JoinPath(tmpdir, "foo")); | 
|  | string file1(JoinPath(rootdir, "file1.txt")); | 
|  | string file2(JoinPath(rootdir, "file2.txt")); | 
|  | string subdir(JoinPath(rootdir, "bar")); | 
|  | string file3(JoinPath(subdir, "file3.txt")); | 
|  |  | 
|  | EXPECT_TRUE(MakeDirectories(subdir, 0700)); | 
|  | EXPECT_TRUE(WriteFile("hello", 5, file1)); | 
|  | EXPECT_TRUE(WriteFile("hello", 5, file2)); | 
|  | EXPECT_TRUE(WriteFile("hello", 5, file3)); | 
|  |  | 
|  | std::map<string, bool> expected; | 
|  | expected["foo/file1.txt"] = false; | 
|  | expected["foo/file2.txt"] = false; | 
|  | expected["foo/bar"] = true; | 
|  |  | 
|  | CollectingDirectoryEntryConsumer consumer("foo"); | 
|  | ForEachDirectoryEntry(rootdir, &consumer); | 
|  | ASSERT_EQ(consumer.entries, expected); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, IsDevNullTest) { | 
|  | ASSERT_TRUE(IsDevNull("/dev/null")); | 
|  | ASSERT_FALSE(IsDevNull("dev/null")); | 
|  | ASSERT_FALSE(IsDevNull("/dev/nul")); | 
|  | ASSERT_FALSE(IsDevNull("/dev/nulll")); | 
|  | ASSERT_FALSE(IsDevNull((char *) nullptr)); | 
|  | ASSERT_FALSE(IsDevNull("")); | 
|  | } | 
|  |  | 
|  | TEST(FileTest, TestRemoveRecursively) { | 
|  | const char* tempdir_cstr = getenv("TEST_TMPDIR"); | 
|  | ASSERT_NE(tempdir_cstr, nullptr); | 
|  | string tempdir(tempdir_cstr); | 
|  | ASSERT_TRUE(PathExists(tempdir)); | 
|  |  | 
|  | string non_existent_dir(JoinPath(tempdir, "test_rmr_non_existent")); | 
|  | EXPECT_TRUE(RemoveRecursively(non_existent_dir)); | 
|  | EXPECT_FALSE(PathExists(non_existent_dir)); | 
|  |  | 
|  | string empty_dir(JoinPath(tempdir, "test_rmr_empty_dir")); | 
|  | EXPECT_TRUE(MakeDirectories(empty_dir, 0700)); | 
|  | EXPECT_TRUE(RemoveRecursively(empty_dir)); | 
|  | EXPECT_FALSE(PathExists(empty_dir)); | 
|  |  | 
|  | string dir_with_content(JoinPath(tempdir, "test_rmr_dir_w_content")); | 
|  | EXPECT_TRUE(MakeDirectories(dir_with_content, 0700)); | 
|  | EXPECT_TRUE(WriteFile("junkdata", 8, JoinPath(dir_with_content, "file"))); | 
|  | string subdir = JoinPath(dir_with_content, "dir"); | 
|  | EXPECT_TRUE(MakeDirectories(subdir, 0700)); | 
|  | string subsubdir = JoinPath(subdir, "dir"); | 
|  | EXPECT_TRUE(MakeDirectories(subsubdir, 0700)); | 
|  | EXPECT_TRUE(WriteFile("junkdata", 8, JoinPath(subsubdir, "deep_file"))); | 
|  | EXPECT_TRUE(RemoveRecursively(dir_with_content)); | 
|  | EXPECT_FALSE(PathExists(dir_with_content)); | 
|  |  | 
|  | string regular_file(JoinPath(tempdir, "test_rmr_regular_file")); | 
|  | EXPECT_TRUE(WriteFile("junkdata", 8, regular_file)); | 
|  | EXPECT_TRUE(RemoveRecursively(regular_file)); | 
|  | EXPECT_FALSE(PathExists(regular_file)); | 
|  | } | 
|  |  | 
|  | }  // namespace blaze_util |