X-Git-Url: http://plrg.eecs.uci.edu/git/?a=blobdiff_plain;f=folly%2Fexperimental%2FTestUtil.cpp;h=987d605b3bb1483a4aa5d021516f8add7d01f695;hb=f19e2b86e49ad30f2279262a672be852283644d2;hp=d68235fb1a73469533455255682e3cf90726081f;hpb=34129f8f52e1d144ae38a3ade5c4aa2365b2b0b1;p=folly.git diff --git a/folly/experimental/TestUtil.cpp b/folly/experimental/TestUtil.cpp index d68235fb..987d605b 100644 --- a/folly/experimental/TestUtil.cpp +++ b/folly/experimental/TestUtil.cpp @@ -1,5 +1,5 @@ /* - * Copyright 2013 Facebook, Inc. + * Copyright 2017 Facebook, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -14,14 +14,23 @@ * limitations under the License. */ -#include "folly/experimental/TestUtil.h" +#include -#include #include -#include +#include + +#include -#include "folly/Conv.h" -#include "folly/Exception.h" +#include +#include +#include +#include +#include +#include + +#ifdef _WIN32 +#include // @manual +#endif namespace folly { namespace test { @@ -41,7 +50,7 @@ fs::path generateUniquePath(fs::path path, StringPiece namePrefix) { return path; } -} // namespce +} // namespace TemporaryFile::TemporaryFile(StringPiece namePrefix, fs::path dir, @@ -51,7 +60,7 @@ TemporaryFile::TemporaryFile(StringPiece namePrefix, closeOnDestruction_(closeOnDestruction), fd_(-1), path_(generateUniquePath(std::move(dir), namePrefix)) { - fd_ = open(path_.c_str(), O_RDWR | O_CREAT | O_EXCL, 0644); + fd_ = open(path_.string().c_str(), O_RDWR | O_CREAT | O_EXCL, 0666); checkUnixError(fd_, "open failed"); if (scope_ == Scope::UNLINK_IMMEDIATELY) { @@ -65,16 +74,23 @@ TemporaryFile::TemporaryFile(StringPiece namePrefix, } } +void TemporaryFile::close() { + if (::close(fd_) == -1) { + PLOG(ERROR) << "close failed"; + } + fd_ = -1; +} + const fs::path& TemporaryFile::path() const { CHECK(scope_ != Scope::UNLINK_IMMEDIATELY); DCHECK(!path_.empty()); return path_; } -TemporaryFile::~TemporaryFile() { +void TemporaryFile::reset() { if (fd_ != -1 && closeOnDestruction_) { - if (close(fd_) == -1) { - PLOG(ERROR) << "close failed"; + if (::close(fd_) == -1) { + PLOG(ERROR) << "close failed (fd = " << fd_ << "): "; } } @@ -89,23 +105,127 @@ TemporaryFile::~TemporaryFile() { } } -TemporaryDirectory::TemporaryDirectory(StringPiece namePrefix, - fs::path dir, - Scope scope) - : scope_(scope), - path_(generateUniquePath(std::move(dir), namePrefix)) { - fs::create_directory(path_); +TemporaryFile::~TemporaryFile() { + reset(); +} + +TemporaryDirectory::TemporaryDirectory( + StringPiece namePrefix, + fs::path dir, + Scope scope) + : scope_(scope), + path_(std::make_unique( + generateUniquePath(std::move(dir), namePrefix))) { + fs::create_directory(path()); } TemporaryDirectory::~TemporaryDirectory() { - if (scope_ == Scope::DELETE_ON_DESTRUCTION) { + if (scope_ == Scope::DELETE_ON_DESTRUCTION && path_ != nullptr) { boost::system::error_code ec; - fs::remove_all(path_, ec); + fs::remove_all(path(), ec); if (ec) { LOG(WARNING) << "recursive delete on destruction failed: " << ec; } } } -} // namespace test -} // namespace folly +ChangeToTempDir::ChangeToTempDir() { + orig_ = fs::current_path(); + fs::current_path(path()); +} + +ChangeToTempDir::~ChangeToTempDir() { + if (!orig_.empty()) { + fs::current_path(orig_); + } +} + +namespace detail { + +SavedState disableInvalidParameters() { +#ifdef _WIN32 + SavedState ret; + ret.previousThreadLocalHandler = _set_thread_local_invalid_parameter_handler( + [](const wchar_t*, + const wchar_t*, + const wchar_t*, + unsigned int, + uintptr_t) {}); + ret.previousCrtReportMode = _CrtSetReportMode(_CRT_ASSERT, 0); + return ret; +#else + return SavedState(); +#endif +} + +#ifdef _WIN32 +void enableInvalidParameters(SavedState state) { + _set_thread_local_invalid_parameter_handler( + (_invalid_parameter_handler)state.previousThreadLocalHandler); + _CrtSetReportMode(_CRT_ASSERT, state.previousCrtReportMode); +} +#else +void enableInvalidParameters(SavedState) {} +#endif + +bool hasPCREPatternMatch(StringPiece pattern, StringPiece target) { + return boost::regex_match( + target.begin(), + target.end(), + boost::regex(pattern.begin(), pattern.end()) + ); +} + +bool hasNoPCREPatternMatch(StringPiece pattern, StringPiece target) { + return !hasPCREPatternMatch(pattern, target); +} + +} // namespace detail + +CaptureFD::CaptureFD(int fd, ChunkCob chunk_cob) + : chunkCob_(std::move(chunk_cob)), fd_(fd), readOffset_(0) { + oldFDCopy_ = dup(fd_); + PCHECK(oldFDCopy_ != -1) << "Could not copy FD " << fd_; + + int file_fd = open(file_.path().string().c_str(), O_WRONLY|O_CREAT, 0600); + PCHECK(dup2(file_fd, fd_) != -1) << "Could not replace FD " << fd_ + << " with " << file_fd; + PCHECK(close(file_fd) != -1) << "Could not close " << file_fd; +} + +void CaptureFD::release() { + if (oldFDCopy_ != fd_) { + readIncremental(); // Feed chunkCob_ + PCHECK(dup2(oldFDCopy_, fd_) != -1) << "Could not restore old FD " + << oldFDCopy_ << " into " << fd_; + PCHECK(close(oldFDCopy_) != -1) << "Could not close " << oldFDCopy_; + oldFDCopy_ = fd_; // Make this call idempotent + } +} + +CaptureFD::~CaptureFD() { + release(); +} + +std::string CaptureFD::read() const { + std::string contents; + std::string filename = file_.path().string(); + PCHECK(folly::readFile(filename.c_str(), contents)); + return contents; +} + +std::string CaptureFD::readIncremental() { + std::string filename = file_.path().string(); + // Yes, I know that I could just keep the file open instead. So sue me. + folly::File f(openNoInt(filename.c_str(), O_RDONLY), true); + auto size = size_t(lseek(f.fd(), 0, SEEK_END) - readOffset_); + std::unique_ptr buf(new char[size]); + auto bytes_read = folly::preadFull(f.fd(), buf.get(), size, readOffset_); + PCHECK(ssize_t(size) == bytes_read); + readOffset_ += off_t(size); + chunkCob_(StringPiece(buf.get(), buf.get() + size)); + return std::string(buf.get(), size); +} + +} // namespace test +} // namespace folly