blob: 093841e20adfab4e1ff75fe469ac9827c5b2f3a6 [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 "src/main/cpp/util/file.h"
#include "src/main/cpp/util/file_platform.h"
#include "gtest/gtest.h"
namespace blaze_util {
using std::map;
using std::string;
using std::vector;
TEST(FileTest, TestSingleThreadedPipe) {
std::unique_ptr<IPipe> pipe(CreatePipe());
char buffer[50] = {0};
ASSERT_TRUE(pipe.get()->Send("hello", 5));
ASSERT_EQ(3, pipe.get()->Receive(buffer, 3));
ASSERT_TRUE(pipe.get()->Send(" world", 6));
ASSERT_EQ(5, pipe.get()->Receive(buffer + 3, 5));
ASSERT_EQ(3, pipe.get()->Receive(buffer + 8, 40));
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.get()->Send("hello", 5));
ASSERT_TRUE(pipe.get()->Send(" world", 6));
});
// Wait for all data to be fully written to the pipe.
writer_thread.join();
ASSERT_EQ(3, pipe.get()->Receive(buffer, 3));
ASSERT_EQ(5, pipe.get()->Receive(buffer + 3, 5));
ASSERT_EQ(3, pipe.get()->Receive(buffer + 8, 40));
ASSERT_EQ(0, strncmp(buffer, "hello world", 11));
}
TEST(FileTest, TestReadFile) {
const char* tempdir = getenv("TEST_TMPDIR");
ASSERT_NE(nullptr, tempdir);
ASSERT_NE(0, tempdir[0]);
std::string filename(JoinPath(tempdir, "test.readfile"));
FILE* fh = fopen(filename.c_str(), "wt");
ASSERT_NE(nullptr, fh);
ASSERT_EQ(11, fwrite("hello world", 1, 11, fh));
fclose(fh);
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, 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};
FILE* fh = fopen(filename.c_str(), "rt");
fflush(fh);
ASSERT_NE(nullptr, fh);
ASSERT_EQ(3, fread(buf, 1, 5, fh));
fclose(fh);
ASSERT_EQ(std::string(buf), std::string("hel"));
ASSERT_TRUE(WriteFile("hello", 5, filename));
fh = fopen(filename.c_str(), "rt");
ASSERT_NE(nullptr, fh);
memset(buf, 0, 6);
ASSERT_EQ(5, fread(buf, 1, 5, fh));
fclose(fh);
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);
string tempdir(tempdir_cstr);
std::unique_ptr<IFileMtime> mtime(CreateFileMtime());
bool actual = false;
ASSERT_TRUE(mtime.get()->GetIfInDistantFuture(tempdir, &actual));
ASSERT_FALSE(actual);
// Create a new file, assert its mtime is not in the future.
string file(JoinPath(tempdir, "foo.txt"));
ASSERT_TRUE(WriteFile("hello", 5, file));
ASSERT_TRUE(mtime.get()->GetIfInDistantFuture(file, &actual));
ASSERT_FALSE(actual);
// Set the file's mtime to the future, assert that it's so.
ASSERT_TRUE(mtime.get()->SetToDistantFuture(file));
ASSERT_TRUE(mtime.get()->GetIfInDistantFuture(file, &actual));
ASSERT_TRUE(actual);
// Overwrite the file, resetting its mtime, assert that GetIfInDistantFuture
// notices.
ASSERT_TRUE(WriteFile("world", 5, file));
ASSERT_TRUE(mtime.get()->GetIfInDistantFuture(file, &actual));
ASSERT_FALSE(actual);
// Set it to the future again so we can reset it using SetToNow.
ASSERT_TRUE(mtime.get()->SetToDistantFuture(file));
ASSERT_TRUE(mtime.get()->GetIfInDistantFuture(file, &actual));
ASSERT_TRUE(actual);
// Assert that SetToNow resets the timestamp.
ASSERT_TRUE(mtime.get()->SetToNow(file));
ASSERT_TRUE(mtime.get()->GetIfInDistantFuture(file, &actual));
ASSERT_FALSE(actual);
// Delete the file and assert that we can no longer set or query its mtime.
ASSERT_TRUE(UnlinkPath(file));
ASSERT_FALSE(mtime.get()->SetToNow(file));
ASSERT_FALSE(mtime.get()->SetToDistantFuture(file));
ASSERT_FALSE(mtime.get()->GetIfInDistantFuture(file, &actual));
}
class CollectingDirectoryEntryConsumer : public DirectoryEntryConsumer {
public:
void Consume(const std::string& name, bool is_directory) override {
// use just base name for easy comparison and no hassle with path separators
// for Windows' sake (test runs on every platform)
entries[Basename(name)] = is_directory;
}
map<string, bool> entries;
};
TEST(FileTest, ForEachDirectoryEntryTest) {
string rootdir(JoinPath(getenv("TEST_TMPDIR"), "foo"));
string file1(JoinPath(rootdir, "file1.txt"));
string file2(JoinPath(rootdir, "file2.txt"));
string subdir(JoinPath(rootdir, "dir1"));
string file3(JoinPath(subdir, "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));
map<string, bool> expected;
expected["file1.txt"] = false;
expected["file2.txt"] = false;
expected["dir1"] = true;
CollectingDirectoryEntryConsumer consumer;
ForEachDirectoryEntry(rootdir, &consumer);
ASSERT_EQ(consumer.entries, expected);
vector<string> actual2;
GetAllFilesUnder(rootdir, &actual2);
std::sort(actual2.begin(), actual2.end());
vector<string> expected2;
vector<string> unixstyle_actual;
// normalize path separators for Windows' sake (test runs on every platform)
for (auto i : actual2) {
std::replace(i.begin(), i.end(), '\\', '/');
unixstyle_actual.push_back(i);
}
for (auto i : {file3, file1, file2}) {
std::replace(i.begin(), i.end(), '\\', '/');
expected2.push_back(i);
}
ASSERT_EQ(unixstyle_actual, expected2);
}
} // namespace blaze_util