• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* Copyright 2015 The TensorFlow Authors. All Rights Reserved.
2 
3 Licensed under the Apache License, Version 2.0 (the "License");
4 you may not use this file except in compliance with the License.
5 You may obtain a copy of the License at
6 
7     http://www.apache.org/licenses/LICENSE-2.0
8 
9 Unless required by applicable law or agreed to in writing, software
10 distributed under the License is distributed on an "AS IS" BASIS,
11 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 See the License for the specific language governing permissions and
13 limitations under the License.
14 ==============================================================================*/
15 
16 #include "tensorflow/core/platform/file_system.h"
17 
18 #include <sys/stat.h>
19 
20 #include "tensorflow/core/lib/core/status_test_util.h"
21 #include "tensorflow/core/platform/null_file_system.h"
22 #include "tensorflow/core/platform/path.h"
23 #include "tensorflow/core/platform/str_util.h"
24 #include "tensorflow/core/platform/strcat.h"
25 #include "tensorflow/core/platform/test.h"
26 
27 namespace tensorflow {
28 
29 static const char* const kPrefix = "ipfs://solarsystem";
30 
31 // A file system that has Planets, Satellites and Sub Satellites. Sub satellites
32 // cannot have children further.
33 class InterPlanetaryFileSystem : public NullFileSystem {
34  public:
35   TF_USE_FILESYSTEM_METHODS_WITH_NO_TRANSACTION_SUPPORT;
36 
FileExists(const string & fname,TransactionToken * token)37   Status FileExists(const string& fname, TransactionToken* token) override {
38     string parsed_path;
39     ParsePath(fname, &parsed_path);
40     if (BodyExists(parsed_path)) {
41       return OkStatus();
42     }
43     return Status(tensorflow::error::NOT_FOUND, "File does not exist");
44   }
45 
46   // Adds the dir to the parent's children list and creates an entry for itself.
CreateDir(const string & dirname,TransactionToken * token)47   Status CreateDir(const string& dirname, TransactionToken* token) override {
48     string parsed_path;
49     ParsePath(dirname, &parsed_path);
50     // If the directory already exists, throw an error.
51     if (celestial_bodies_.find(parsed_path) != celestial_bodies_.end()) {
52       return Status(tensorflow::error::ALREADY_EXISTS,
53                     "dirname already exists.");
54     }
55     std::vector<string> split_path = str_util::Split(parsed_path, '/');
56     // If the path is too long then we don't support it.
57     if (split_path.size() > 3) {
58       return Status(tensorflow::error::INVALID_ARGUMENT, "Bad dirname");
59     }
60     if (split_path.empty()) {
61       return OkStatus();
62     }
63     if (split_path.size() == 1) {
64       celestial_bodies_[""].insert(parsed_path);
65       celestial_bodies_.insert(
66           std::pair<string, std::set<string>>(parsed_path, {}));
67       return OkStatus();
68     }
69     if (split_path.size() == 2) {
70       if (!BodyExists(split_path[0])) {
71         return Status(tensorflow::error::FAILED_PRECONDITION,
72                       "Base dir not created");
73       }
74       celestial_bodies_[split_path[0]].insert(split_path[1]);
75       celestial_bodies_.insert(
76           std::pair<string, std::set<string>>(parsed_path, {}));
77       return OkStatus();
78     }
79     if (split_path.size() == 3) {
80       const string& parent_path = this->JoinPath(split_path[0], split_path[1]);
81       if (!BodyExists(parent_path)) {
82         return Status(tensorflow::error::FAILED_PRECONDITION,
83                       "Base dir not created");
84       }
85       celestial_bodies_[parent_path].insert(split_path[2]);
86       celestial_bodies_.insert(
87           std::pair<string, std::set<string>>(parsed_path, {}));
88       return OkStatus();
89     }
90     return Status(tensorflow::error::FAILED_PRECONDITION, "Failed to create");
91   }
92 
IsDirectory(const string & dirname,TransactionToken * token)93   Status IsDirectory(const string& dirname, TransactionToken* token) override {
94     string parsed_path;
95     ParsePath(dirname, &parsed_path);
96     // Simulate evil_directory has bad permissions by throwing a LOG(FATAL)
97     if (parsed_path == "evil_directory") {
98       LOG(FATAL) << "evil_directory cannot be accessed";
99     }
100     std::vector<string> split_path = str_util::Split(parsed_path, '/');
101     if (split_path.size() > 2) {
102       return Status(tensorflow::error::FAILED_PRECONDITION, "Not a dir");
103     }
104     if (celestial_bodies_.find(parsed_path) != celestial_bodies_.end()) {
105       return OkStatus();
106     }
107     return Status(tensorflow::error::FAILED_PRECONDITION, "Not a dir");
108   }
109 
GetChildren(const string & dir,TransactionToken * token,std::vector<string> * result)110   Status GetChildren(const string& dir, TransactionToken* token,
111                      std::vector<string>* result) override {
112     TF_RETURN_IF_ERROR(IsDirectory(dir, nullptr));
113     string parsed_path;
114     ParsePath(dir, &parsed_path);
115     result->insert(result->begin(), celestial_bodies_[parsed_path].begin(),
116                    celestial_bodies_[parsed_path].end());
117     return OkStatus();
118   }
119 
120  private:
BodyExists(const string & name)121   bool BodyExists(const string& name) {
122     return celestial_bodies_.find(name) != celestial_bodies_.end();
123   }
124 
ParsePath(const string & name,string * parsed_path)125   void ParsePath(const string& name, string* parsed_path) {
126     StringPiece scheme, host, path;
127     this->ParseURI(name, &scheme, &host, &path);
128     ASSERT_EQ(scheme, "ipfs");
129     ASSERT_EQ(host, "solarsystem");
130     absl::ConsumePrefix(&path, "/");
131     *parsed_path = string(path);
132   }
133 
134   std::map<string, std::set<string>> celestial_bodies_ = {
135       std::pair<string, std::set<string>>(
136           "", {"Mercury", "Venus", "Earth", "Mars", "Jupiter", "Saturn",
137                "Uranus", "Neptune"}),
138       std::pair<string, std::set<string>>("Mercury", {}),
139       std::pair<string, std::set<string>>("Venus", {}),
140       std::pair<string, std::set<string>>("Earth", {"Moon"}),
141       std::pair<string, std::set<string>>("Mars", {}),
142       std::pair<string, std::set<string>>("Jupiter",
143                                           {"Europa", "Io", "Ganymede"}),
144       std::pair<string, std::set<string>>("Saturn", {}),
145       std::pair<string, std::set<string>>("Uranus", {}),
146       std::pair<string, std::set<string>>("Neptune", {}),
147       std::pair<string, std::set<string>>("Earth/Moon", {}),
148       std::pair<string, std::set<string>>("Jupiter/Europa", {}),
149       std::pair<string, std::set<string>>("Jupiter/Io", {}),
150       std::pair<string, std::set<string>>("Jupiter/Ganymede", {})};
151 };
152 
153 // Returns all the matched entries as a comma separated string removing the
154 // common prefix of BaseDir().
Match(InterPlanetaryFileSystem * ipfs,const string & suffix_pattern)155 string Match(InterPlanetaryFileSystem* ipfs, const string& suffix_pattern) {
156   std::vector<string> results;
157   Status s = ipfs->GetMatchingPaths(ipfs->JoinPath(kPrefix, suffix_pattern),
158                                     nullptr, &results);
159   if (!s.ok()) {
160     return s.ToString();
161   } else {
162     std::vector<StringPiece> trimmed_results;
163     std::sort(results.begin(), results.end());
164     for (const string& result : results) {
165       StringPiece trimmed_result(result);
166       EXPECT_TRUE(
167           absl::ConsumePrefix(&trimmed_result, strings::StrCat(kPrefix, "/")));
168       trimmed_results.push_back(trimmed_result);
169     }
170     return absl::StrJoin(trimmed_results, ",");
171   }
172 }
173 
TEST(InterPlanetaryFileSystemTest,IPFSMatch)174 TEST(InterPlanetaryFileSystemTest, IPFSMatch) {
175   InterPlanetaryFileSystem ipfs;
176   EXPECT_EQ(Match(&ipfs, "thereisnosuchfile"), "");
177   EXPECT_EQ(Match(&ipfs, "*"),
178             "Earth,Jupiter,Mars,Mercury,Neptune,Saturn,Uranus,Venus");
179   // Returns Jupiter's moons.
180   EXPECT_EQ(Match(&ipfs, "Jupiter/*"),
181             "Jupiter/Europa,Jupiter/Ganymede,Jupiter/Io");
182   // Returns Jupiter's and Earth's moons.
183   EXPECT_EQ(Match(&ipfs, "*/*"),
184             "Earth/Moon,Jupiter/Europa,Jupiter/Ganymede,Jupiter/Io");
185   TF_EXPECT_OK(ipfs.CreateDir(ipfs.JoinPath(kPrefix, "Planet0"), nullptr));
186   TF_EXPECT_OK(ipfs.CreateDir(ipfs.JoinPath(kPrefix, "Planet1"), nullptr));
187   EXPECT_EQ(Match(&ipfs, "Planet[0-1]"), "Planet0,Planet1");
188   EXPECT_EQ(Match(&ipfs, "Planet?"), "Planet0,Planet1");
189 }
190 
TEST(InterPlanetaryFileSystemTest,MatchSimple)191 TEST(InterPlanetaryFileSystemTest, MatchSimple) {
192   InterPlanetaryFileSystem ipfs;
193   TF_EXPECT_OK(ipfs.CreateDir(ipfs.JoinPath(kPrefix, "match-00"), nullptr));
194   TF_EXPECT_OK(ipfs.CreateDir(ipfs.JoinPath(kPrefix, "match-0a"), nullptr));
195   TF_EXPECT_OK(ipfs.CreateDir(ipfs.JoinPath(kPrefix, "match-01"), nullptr));
196   TF_EXPECT_OK(ipfs.CreateDir(ipfs.JoinPath(kPrefix, "match-aaa"), nullptr));
197 
198   EXPECT_EQ(Match(&ipfs, "match-*"), "match-00,match-01,match-0a,match-aaa");
199   EXPECT_EQ(Match(&ipfs, "match-0[0-9]"), "match-00,match-01");
200   EXPECT_EQ(Match(&ipfs, "match-?[0-9]"), "match-00,match-01");
201   EXPECT_EQ(Match(&ipfs, "match-?a*"), "match-0a,match-aaa");
202   EXPECT_EQ(Match(&ipfs, "match-??"), "match-00,match-01,match-0a");
203 }
204 
205 // Create 2 directories abcd and evil_directory. Look for abcd and make sure
206 // that evil_directory isn't accessed.
TEST(InterPlanetaryFileSystemTest,MatchOnlyNeeded)207 TEST(InterPlanetaryFileSystemTest, MatchOnlyNeeded) {
208   InterPlanetaryFileSystem ipfs;
209   TF_EXPECT_OK(ipfs.CreateDir(ipfs.JoinPath(kPrefix, "abcd"), nullptr));
210   TF_EXPECT_OK(
211       ipfs.CreateDir(ipfs.JoinPath(kPrefix, "evil_directory"), nullptr));
212 
213   EXPECT_EQ(Match(&ipfs, "abcd"), "abcd");
214 }
215 
TEST(InterPlanetaryFileSystemTest,MatchDirectory)216 TEST(InterPlanetaryFileSystemTest, MatchDirectory) {
217   InterPlanetaryFileSystem ipfs;
218   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
219       ipfs.JoinPath(kPrefix, "match-00/abc/x"), nullptr));
220   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
221       ipfs.JoinPath(kPrefix, "match-0a/abc/x"), nullptr));
222   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
223       ipfs.JoinPath(kPrefix, "match-01/abc/x"), nullptr));
224   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
225       ipfs.JoinPath(kPrefix, "match-aaa/abc/x"), nullptr));
226 
227   EXPECT_EQ(Match(&ipfs, "match-*/abc/x"),
228             "match-00/abc/x,match-01/abc/x,match-0a/abc/x,match-aaa/abc/x");
229   EXPECT_EQ(Match(&ipfs, "match-0[0-9]/abc/x"),
230             "match-00/abc/x,match-01/abc/x");
231   EXPECT_EQ(Match(&ipfs, "match-?[0-9]/abc/x"),
232             "match-00/abc/x,match-01/abc/x");
233   EXPECT_EQ(Match(&ipfs, "match-?a*/abc/x"), "match-0a/abc/x,match-aaa/abc/x");
234   EXPECT_EQ(Match(&ipfs, "match-?[^a]/abc/x"), "match-00/abc/x,match-01/abc/x");
235 }
236 
TEST(InterPlanetaryFileSystemTest,MatchMultipleWildcards)237 TEST(InterPlanetaryFileSystemTest, MatchMultipleWildcards) {
238   InterPlanetaryFileSystem ipfs;
239   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
240       ipfs.JoinPath(kPrefix, "match-00/abc/00"), nullptr));
241   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
242       ipfs.JoinPath(kPrefix, "match-00/abc/01"), nullptr));
243   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
244       ipfs.JoinPath(kPrefix, "match-00/abc/09"), nullptr));
245   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
246       ipfs.JoinPath(kPrefix, "match-01/abc/00"), nullptr));
247   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
248       ipfs.JoinPath(kPrefix, "match-01/abc/04"), nullptr));
249   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
250       ipfs.JoinPath(kPrefix, "match-01/abc/10"), nullptr));
251   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(
252       ipfs.JoinPath(kPrefix, "match-02/abc/00"), nullptr));
253 
254   EXPECT_EQ(Match(&ipfs, "match-0[0-1]/abc/0[0-8]"),
255             "match-00/abc/00,match-00/abc/01,match-01/abc/00,match-01/abc/04");
256 }
257 
TEST(InterPlanetaryFileSystemTest,RecursivelyCreateAlreadyExistingDir)258 TEST(InterPlanetaryFileSystemTest, RecursivelyCreateAlreadyExistingDir) {
259   InterPlanetaryFileSystem ipfs;
260   const string dirname = ipfs.JoinPath(kPrefix, "match-00/abc/00");
261   TF_EXPECT_OK(ipfs.RecursivelyCreateDir(dirname));
262   // We no longer check for recursively creating the directory again because
263   // `ipfs.IsDirectory` is badly implemented, fixing it will break other tests
264   // in this suite and we already test creating the directory again in
265   // env_test.cc as well as in the modular filesystem tests.
266 }
267 
TEST(InterPlanetaryFileSystemTest,HasAtomicMove)268 TEST(InterPlanetaryFileSystemTest, HasAtomicMove) {
269   InterPlanetaryFileSystem ipfs;
270   const string dirname = io::JoinPath(kPrefix, "match-00/abc/00");
271   bool has_atomic_move;
272   TF_EXPECT_OK(ipfs.HasAtomicMove(dirname, &has_atomic_move));
273   EXPECT_EQ(has_atomic_move, true);
274 }
275 
276 // A simple file system with a root directory and a single file underneath it.
277 class TestFileSystem : public NullFileSystem {
278  public:
279   // Only allow for a single root directory.
IsDirectory(const string & dirname,TransactionToken * token)280   Status IsDirectory(const string& dirname, TransactionToken* token) override {
281     if (dirname == "." || dirname.empty()) {
282       return OkStatus();
283     }
284     return Status(tensorflow::error::FAILED_PRECONDITION, "Not a dir");
285   }
286 
287   // Simulating a FS with a root dir and a single file underneath it.
GetChildren(const string & dir,TransactionToken * token,std::vector<string> * result)288   Status GetChildren(const string& dir, TransactionToken* token,
289                      std::vector<string>* result) override {
290     if (dir == "." || dir.empty()) {
291       result->push_back("test");
292     }
293     return OkStatus();
294   }
295 };
296 
297 // Making sure that ./<pattern> and <pattern> have the same result.
TEST(TestFileSystemTest,RootDirectory)298 TEST(TestFileSystemTest, RootDirectory) {
299   TestFileSystem fs;
300   std::vector<string> results;
301   auto ret = fs.GetMatchingPaths("./te*", nullptr, &results);
302   EXPECT_EQ(1, results.size());
303   EXPECT_EQ("./test", results[0]);
304   ret = fs.GetMatchingPaths("te*", nullptr, &results);
305   EXPECT_EQ(1, results.size());
306   EXPECT_EQ("./test", results[0]);
307 }
308 
309 }  // namespace tensorflow
310