2 * Copyright 2013 Facebook, Inc.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
17 #include "folly/File.h"
21 #include <boost/thread/locks.hpp>
22 #include <glog/logging.h>
23 #include <gtest/gtest.h>
25 #include "folly/Benchmark.h"
26 #include "folly/String.h"
27 #include "folly/Subprocess.h"
28 #include "folly/experimental/io/FsUtil.h"
29 #include "folly/experimental/TestUtil.h"
31 using namespace folly;
32 using namespace folly::test;
35 void expectWouldBlock(ssize_t r) {
36 int savedErrno = errno;
38 EXPECT_EQ(EAGAIN, savedErrno) << errnoStr(errno);
40 void expectOK(ssize_t r) {
41 int savedErrno = errno;
42 EXPECT_LE(0, r) << ": errno=" << errnoStr(errno);
47 // Open a file, ensure it's indeed open for reading
51 EXPECT_NE(-1, f.fd());
52 EXPECT_EQ(1, ::read(f.fd(), &buf, 1));
54 EXPECT_EQ(-1, f.fd());
59 // Wrap a file descriptor, make sure that ownsFd works
60 // We'll test that the file descriptor is closed by closing the writing
61 // end of a pipe and making sure that a non-blocking read from the reading
67 int flags = ::fcntl(p[0], F_GETFL);
69 expectOK(::fcntl(p[0], F_SETFL, flags | O_NONBLOCK));
70 expectWouldBlock(::read(p[0], &buf, 1));
73 EXPECT_EQ(p[1], f.fd());
75 // Ensure that moving the file doesn't close it
78 EXPECT_EQ(p[1], f.fd());
79 File f1(std::move(f));
80 EXPECT_EQ(-1, f.fd());
81 EXPECT_EQ(p[1], f1.fd());
83 expectWouldBlock(::read(p[0], &buf, 1)); // not closed
86 EXPECT_EQ(p[1], f.fd());
88 ssize_t r = ::read(p[0], &buf, 1); // eof
94 #define EXPECT_CONTAINS(haystack, needle) \
95 EXPECT_NE(::std::string::npos, ::folly::StringPiece(haystack).find(needle)) \
96 << "Haystack: '" << haystack << "'\nNeedle: '" << needle << "'";
98 TEST(File, UsefulError) {
100 File("does_not_exist.txt", 0, 0666);
101 } catch (const std::runtime_error& e) {
102 EXPECT_CONTAINS(e.what(), "does_not_exist.txt");
103 EXPECT_CONTAINS(e.what(), "0666");
108 File temp = File::temporary();
110 EXPECT_TRUE(bool(temp));
118 if (File file = File::temporary()) {
124 EXPECT_FALSE(bool(File()));
128 if (File notOpened = File()) {
134 typedef std::unique_lock<File> Lock;
135 typedef boost::shared_lock<File> SharedLock;
137 // Find out where we are.
138 static constexpr size_t pathLength = 2048;
139 char buf[pathLength + 1];
140 int r = readlink("/proc/self/exe", buf, pathLength);
144 fs::path helper(buf);
145 helper.remove_filename();
146 helper /= "file_test_lock_helper";
148 TemporaryFile tempFile;
149 File f(tempFile.fd());
151 enum LockMode { EXCLUSIVE, SHARED };
152 auto testLock = [&] (LockMode mode, bool expectedSuccess) {
154 Subprocess({helper.native(),
155 mode == SHARED ? "-s" : "-x",
156 tempFile.path().native()}).wait();
157 EXPECT_TRUE(ret.exited());
159 EXPECT_EQ(expectedSuccess ? 0 : 42, ret.exitStatus());
163 // Make sure nothing breaks and things compile.
173 Lock lock(f, std::defer_lock);
174 EXPECT_TRUE(lock.try_lock());
178 SharedLock lock(f, boost::defer_lock);
179 EXPECT_TRUE(lock.try_lock());
185 testLock(EXCLUSIVE, false);
191 testLock(SHARED, false);
197 testLock(EXCLUSIVE, false);
200 // S does not block S
203 testLock(SHARED, true);