blob: 88432f19f6c13a7a3f05d6fb4ea069afcadd1f69 [file] [log] [blame]
#include <stdlib.h>
// 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>
#include <thread> // NOLINT (to silence Google-internal linter)
#include <vector>
#include "src/main/cpp/util/file.h"
#include "src/main/cpp/util/file_platform.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"
#include "absl/strings/match.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);
// Assert that a directory is always untampered with. (We do
// not care about directories' mtimes.)
ASSERT_TRUE(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(IsUntampered(file));
// Set the file's mtime to the future, assert that it's so.
ASSERT_TRUE(SetMtimeToDistantFuture(file));
ASSERT_TRUE(IsUntampered(file));
// Overwrite the file, resetting its mtime, assert that
// IsUntampered notices.
ASSERT_TRUE(WriteFile("world", 5, file));
ASSERT_FALSE(IsUntampered(file));
// Set it to the future again so we can reset it using SetToNow.
ASSERT_TRUE(SetMtimeToDistantFuture(file));
ASSERT_TRUE(IsUntampered(file));
// Assert that SetToNow resets the timestamp.
ASSERT_TRUE(SetMtimeToNow(file));
ASSERT_FALSE(IsUntampered(file));
// Delete the file and assert that we can no longer set or query its mtime.
ASSERT_TRUE(UnlinkPath(file));
ASSERT_FALSE(SetMtimeToNow(file));
ASSERT_FALSE(SetMtimeToDistantFuture(file));
ASSERT_FALSE(IsUntampered(file));
}
TEST(FileTest, TestCreateSiblingTempDir) {
const char* tempdir_cstr = getenv("TEST_TMPDIR");
EXPECT_NE(tempdir_cstr, nullptr);
EXPECT_NE(tempdir_cstr[0], 0);
Path tempdir(tempdir_cstr);
Path input_in_existing = tempdir.GetRelative("other");
Path output_in_existing = CreateSiblingTempDir(input_in_existing);
ASSERT_NE(input_in_existing, output_in_existing);
ASSERT_EQ(input_in_existing.GetParent(), output_in_existing.GetParent());
ASSERT_TRUE(absl::StartsWith(output_in_existing.GetBaseName(),
input_in_existing.GetBaseName() + ".tmp."));
EXPECT_TRUE(PathExists(output_in_existing));
Path missing_dir = tempdir.GetRelative("doesntexistyet");
ASSERT_FALSE(PathExists(missing_dir));
Path input_in_missing = missing_dir.GetRelative("other");
Path output_in_missing = CreateSiblingTempDir(input_in_missing);
ASSERT_NE(input_in_missing, output_in_missing);
ASSERT_EQ(input_in_missing.GetParent(), output_in_missing.GetParent());
ASSERT_TRUE(absl::StartsWith(output_in_missing.GetBaseName(),
input_in_missing.GetBaseName() + ".tmp."));
EXPECT_TRUE(PathExists(output_in_missing));
}
TEST(FileTest, TestRenameDirectory) {
const char* tempdir_cstr = getenv("TEST_TMPDIR");
EXPECT_NE(tempdir_cstr, nullptr);
EXPECT_NE(tempdir_cstr[0], 0);
Path tempdir(tempdir_cstr);
Path dir1 = tempdir.GetRelative("test_rename_dir/dir1");
Path dir2 = tempdir.GetRelative("test_rename_dir/dir2");
EXPECT_TRUE(MakeDirectories(dir1, 0700));
Path file1 = dir1.GetRelative("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 MockDirectoryEntryConsumer : public DirectoryEntryConsumer {
public:
void Consume(const Path& path, bool is_directory) override {
entries[path] = is_directory;
}
std::map<Path, bool> entries;
};
TEST(FileTest, ForEachDirectoryEntryTest) {
const char* tmpdir_cstr = getenv("TEST_TMPDIR");
ASSERT_NE(tmpdir_cstr, nullptr);
Path tmpdir(tmpdir_cstr);
ASSERT_TRUE(PathExists(tmpdir));
// $TEST_TMPDIR/
// foo/
// bar/
// file3.txt
// file1.txt
// file2.txt
Path rootdir = tmpdir.GetRelative("foo");
Path file1 = rootdir.GetRelative("file1.txt");
Path file2 = rootdir.GetRelative("file2.txt");
Path subdir = rootdir.GetRelative("bar");
Path file3 = subdir.GetRelative("file3.txt");
ASSERT_TRUE(MakeDirectories(subdir, 0700));
ASSERT_TRUE(WriteFile("hello", 5, file1));
ASSERT_TRUE(WriteFile("hello", 5, file2));
ASSERT_TRUE(WriteFile("hello", 5, file3));
std::map<Path, bool> expected;
expected[file1] = false;
expected[file2] = false;
expected[subdir] = true;
MockDirectoryEntryConsumer consumer;
ForEachDirectoryEntry(rootdir, &consumer);
EXPECT_EQ(consumer.entries, expected);
}
TEST(FilePosixTest, GetAllFilesUnder) {
const char* tmpdir_cstr = getenv("TEST_TMPDIR");
ASSERT_NE(tmpdir_cstr, nullptr);
Path tmpdir(tmpdir_cstr);
ASSERT_TRUE(PathExists(tmpdir));
Path root = tmpdir.GetRelative("FilePosixTest.GetAllFilesUnder.root");
Path file1 = root.GetRelative("file1");
Path dir1 = root.GetRelative("dir1");
Path dir2 = root.GetRelative("dir2");
Path dir3 = dir1.GetRelative("dir3");
Path file2 = dir1.GetRelative("file2");
Path file3 = dir2.GetRelative("file3");
Path file4 = dir3.GetRelative("file4");
Path file5 = dir3.GetRelative("file5");
ASSERT_TRUE(MakeDirectories(root, 0700));
ASSERT_TRUE(MakeDirectories(dir1, 0700));
ASSERT_TRUE(MakeDirectories(dir2, 0700));
ASSERT_TRUE(MakeDirectories(dir3, 0700));
ASSERT_TRUE(WriteFile("", file1));
ASSERT_TRUE(WriteFile("", file2));
ASSERT_TRUE(WriteFile("", file3));
ASSERT_TRUE(WriteFile("", file4));
ASSERT_TRUE(WriteFile("", file5));
std::vector<Path> result;
GetAllFilesUnder(root, &result);
std::sort(result.begin(), result.end());
std::vector<Path> expected({file4, file5, file2, file3, file1});
EXPECT_EQ(expected, result);
EXPECT_TRUE(RemoveRecursively(root));
}
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);
Path tempdir(tempdir_cstr);
ASSERT_TRUE(PathExists(tempdir));
Path non_existent_dir = tempdir.GetRelative("test_rmr_non_existent");
EXPECT_TRUE(RemoveRecursively(non_existent_dir));
EXPECT_FALSE(PathExists(non_existent_dir));
Path empty_dir = tempdir.GetRelative("test_rmr_empty_dir");
EXPECT_TRUE(MakeDirectories(empty_dir, 0700));
EXPECT_TRUE(RemoveRecursively(empty_dir));
EXPECT_FALSE(PathExists(empty_dir));
Path dir_with_content = tempdir.GetRelative("test_rmr_dir_w_content");
EXPECT_TRUE(MakeDirectories(dir_with_content, 0700));
EXPECT_TRUE(WriteFile("junkdata", 8, dir_with_content.GetRelative("file")));
Path subdir = dir_with_content.GetRelative("dir");
EXPECT_TRUE(MakeDirectories(subdir, 0700));
Path subsubdir = subdir.GetRelative("dir");
EXPECT_TRUE(MakeDirectories(subsubdir, 0700));
EXPECT_TRUE(WriteFile("junkdata", 8, subsubdir.GetRelative("deep_file")));
EXPECT_TRUE(RemoveRecursively(dir_with_content));
EXPECT_FALSE(PathExists(dir_with_content));
Path regular_file = tempdir.GetRelative("test_rmr_regular_file");
EXPECT_TRUE(WriteFile("junkdata", 8, regular_file));
EXPECT_TRUE(RemoveRecursively(regular_file));
EXPECT_FALSE(PathExists(regular_file));
}
} // namespace blaze_util