blob: e67622b06c5da54e7159e3776bf76879d2dfb9cc [file] [log] [blame]
// 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, 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