libdcp
filesystem.cc
1 /*
2  Copyright (C) 2012-2021 Carl Hetherington <cth@carlh.net>
3 
4  This file is part of libdcp.
5 
6  libdcp is free software; you can redistribute it and/or modify
7  it under the terms of the GNU General Public License as published by
8  the Free Software Foundation; either version 2 of the License, or
9  (at your option) any later version.
10 
11  libdcp is distributed in the hope that it will be useful,
12  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14  GNU General Public License for more details.
15 
16  You should have received a copy of the GNU General Public License
17  along with libdcp. If not, see <http://www.gnu.org/licenses/>.
18 
19  In addition, as a special exception, the copyright holders give
20  permission to link the code of portions of this program with the
21  OpenSSL library under certain conditions as described in each
22  individual source file, and distribute linked combinations
23  including the two.
24 
25  You must obey the GNU General Public License in all respects
26  for all of the code used other than OpenSSL. If you modify
27  file(s) with this exception, you may extend this exception to your
28  version of the file(s), but you are not obligated to do so. If you
29  do not wish to do so, delete this exception statement from your
30  version. If you delete this exception statement from all source
31  files in the program, then also delete it here.
32 */
33 
34 
35 #include "filesystem.h"
36 #include <boost/algorithm/string.hpp>
37 
38 
39 bool
40 dcp::filesystem::exists(boost::filesystem::path const& path)
41 {
42  return boost::filesystem::exists(dcp::filesystem::fix_long_path(path));
43 }
44 
45 
46 bool
47 dcp::filesystem::exists(boost::filesystem::path const& path, boost::system::error_code& ec)
48 {
49  return boost::filesystem::exists(dcp::filesystem::fix_long_path(path), ec);
50 }
51 
52 
53 bool
54 dcp::filesystem::is_directory(boost::filesystem::path const& path)
55 {
56  return boost::filesystem::is_directory(dcp::filesystem::fix_long_path(path));
57 }
58 
59 
60 bool
61 dcp::filesystem::is_empty(boost::filesystem::path const& path)
62 {
63  return boost::filesystem::is_empty(dcp::filesystem::fix_long_path(path));
64 }
65 
66 
67 bool
68 dcp::filesystem::is_regular_file(boost::filesystem::path const& path)
69 {
70  return boost::filesystem::is_regular_file(dcp::filesystem::fix_long_path(path));
71 }
72 
73 
74 bool
75 dcp::filesystem::create_directory(boost::filesystem::path const& path)
76 {
77  return boost::filesystem::create_directory(dcp::filesystem::fix_long_path(path));
78 }
79 
80 
81 bool
82 dcp::filesystem::create_directory(boost::filesystem::path const& path, boost::system::error_code& ec)
83 {
84  return boost::filesystem::create_directory(dcp::filesystem::fix_long_path(path), ec);
85 }
86 
87 
88 void
89 dcp::filesystem::copy(boost::filesystem::path const& from, boost::filesystem::path const& to)
90 {
91  boost::filesystem::copy(dcp::filesystem::fix_long_path(from), dcp::filesystem::fix_long_path(to));
92 }
93 
94 
95 void
96 dcp::filesystem::copy_file(boost::filesystem::path const& from, boost::filesystem::path const& to)
97 {
98  boost::filesystem::copy_file(dcp::filesystem::fix_long_path(from), dcp::filesystem::fix_long_path(to));
99 }
100 
101 
102 void
103 dcp::filesystem::copy_file(boost::filesystem::path const& from, boost::filesystem::path const& to, boost::system::error_code& ec)
104 {
105  boost::filesystem::copy_file(dcp::filesystem::fix_long_path(from), dcp::filesystem::fix_long_path(to), ec);
106 }
107 
108 
109 void
110 dcp::filesystem::copy_file(boost::filesystem::path const& from, boost::filesystem::path const& to, CopyOptions option)
111 {
112 #ifdef LIBDCP_HAVE_COPY_OPTIONS
113  auto const options = option == CopyOptions::OVERWRITE_EXISTING ? boost::filesystem::copy_options::overwrite_existing : boost::filesystem::copy_options::none;
114 #else
115  auto const options = option == CopyOptions::OVERWRITE_EXISTING ? boost::filesystem::copy_option::overwrite_if_exists : boost::filesystem::copy_option::none;
116 #endif
117 
118  boost::filesystem::copy_file(dcp::filesystem::fix_long_path(from), dcp::filesystem::fix_long_path(to), options);
119 }
120 
121 
122 bool
123 dcp::filesystem::create_directories(boost::filesystem::path const& path)
124 {
125  return boost::filesystem::create_directories(dcp::filesystem::fix_long_path(path));
126 }
127 
128 
129 bool
130 dcp::filesystem::create_directories(boost::filesystem::path const& path, boost::system::error_code& ec)
131 {
132  return boost::filesystem::create_directories(dcp::filesystem::fix_long_path(path), ec);
133 }
134 
135 
136 boost::filesystem::path
137 dcp::filesystem::absolute(boost::filesystem::path const& path)
138 {
139  return dcp::filesystem::unfix_long_path(boost::filesystem::absolute(dcp::filesystem::fix_long_path(path)));
140 }
141 
142 
143 boost::filesystem::path
144 dcp::filesystem::canonical(boost::filesystem::path const& path)
145 {
146  return dcp::filesystem::unfix_long_path(boost::filesystem::canonical(dcp::filesystem::fix_long_path(path)));
147 }
148 
149 
150 boost::filesystem::path
151 dcp::filesystem::weakly_canonical(boost::filesystem::path const& path)
152 {
153 #ifdef DCPOMATIC_HAVE_WEAKLY_CANONICAL
154  return dcp::filesystem::unfix_long_path(boost::filesystem::weakly_canonical(dcp::filesystem::fix_long_path(path)));
155 #else
156  boost::filesystem::path complete(boost::filesystem::system_complete(dcp::filesystem::fix_long_path(path)));
157  boost::filesystem::path result;
158  for (auto part: complete) {
159  if (part == "..") {
160  boost::system::error_code ec;
161  if (boost::filesystem::is_symlink(result, ec) || result.filename() == "..") {
162  result /= part;
163  } else {
164  result = result.parent_path();
165  }
166  } else if (part != ".") {
167  result /= part;
168  }
169  }
170 
171  return dcp::filesystem::unfix_long_path(result.make_preferred());
172 #endif
173 }
174 
175 
176 bool
177 dcp::filesystem::remove(boost::filesystem::path const& path)
178 {
179  return boost::filesystem::remove(dcp::filesystem::fix_long_path(path));
180 }
181 
182 
183 bool
184 dcp::filesystem::remove(boost::filesystem::path const& path, boost::system::error_code& ec)
185 {
186  return boost::filesystem::remove(dcp::filesystem::fix_long_path(path), ec);
187 }
188 
189 
190 uintmax_t
191 dcp::filesystem::remove_all(boost::filesystem::path const& path)
192 {
193  return boost::filesystem::remove_all(dcp::filesystem::fix_long_path(path));
194 }
195 
196 
197 uintmax_t
198 dcp::filesystem::remove_all(boost::filesystem::path const& path, boost::system::error_code& ec)
199 {
200  return boost::filesystem::remove_all(dcp::filesystem::fix_long_path(path), ec);
201 }
202 
203 
204 uintmax_t
205 dcp::filesystem::file_size(boost::filesystem::path const& path)
206 {
207  return boost::filesystem::file_size(dcp::filesystem::fix_long_path(path));
208 }
209 
210 
211 uintmax_t
212 dcp::filesystem::file_size(boost::filesystem::path const& path, boost::system::error_code& ec)
213 {
214  return boost::filesystem::file_size(dcp::filesystem::fix_long_path(path), ec);
215 }
216 
217 
218 boost::filesystem::path
219 dcp::filesystem::current_path()
220 {
221  return dcp::filesystem::unfix_long_path(boost::filesystem::current_path());
222 }
223 
224 
225 void
226 dcp::filesystem::current_path(boost::filesystem::path const& path)
227 {
228  boost::filesystem::current_path(dcp::filesystem::fix_long_path(path));
229 }
230 
231 
232 void
233 dcp::filesystem::create_hard_link(boost::filesystem::path const& from, boost::filesystem::path const& to)
234 {
235  boost::filesystem::create_hard_link(dcp::filesystem::fix_long_path(from), dcp::filesystem::fix_long_path(to));
236 }
237 
238 
239 void
240 dcp::filesystem::create_hard_link(boost::filesystem::path const& from, boost::filesystem::path const& to, boost::system::error_code& ec)
241 {
242  boost::filesystem::create_hard_link(dcp::filesystem::fix_long_path(from), dcp::filesystem::fix_long_path(to), ec);
243 }
244 
245 
246 void
247 dcp::filesystem::create_symlink(boost::filesystem::path const& from, boost::filesystem::path const& to, boost::system::error_code& ec)
248 {
249  boost::filesystem::create_symlink(dcp::filesystem::fix_long_path(from), dcp::filesystem::fix_long_path(to), ec);
250 }
251 
252 
253 std::string
254 dcp::filesystem::extension(boost::filesystem::path const& path)
255 {
256  return dcp::filesystem::fix_long_path(path).extension().string();
257 }
258 
259 
260 boost::filesystem::space_info
261 dcp::filesystem::space(boost::filesystem::path const& path)
262 {
263  return boost::filesystem::space(dcp::filesystem::fix_long_path(path));
264 }
265 
266 
267 std::time_t
268 dcp::filesystem::last_write_time(boost::filesystem::path const& path)
269 {
270  return boost::filesystem::last_write_time(dcp::filesystem::fix_long_path(path));
271 }
272 
273 
274 std::time_t
275 dcp::filesystem::last_write_time(boost::filesystem::path const& path, boost::system::error_code& ec)
276 {
277  return boost::filesystem::last_write_time(dcp::filesystem::fix_long_path(path), ec);
278 }
279 
280 
281 uintmax_t
282 dcp::filesystem::hard_link_count(boost::filesystem::path const& path)
283 {
284  return boost::filesystem::hard_link_count(dcp::filesystem::fix_long_path(path));
285 }
286 
287 
288 void
289 dcp::filesystem::rename(boost::filesystem::path const& old_path, boost::filesystem::path const& new_path)
290 {
291  boost::filesystem::rename(dcp::filesystem::fix_long_path(old_path), dcp::filesystem::fix_long_path(new_path));
292 }
293 
294 
295 void
296 dcp::filesystem::rename(boost::filesystem::path const& old_path, boost::filesystem::path const& new_path, boost::system::error_code& ec)
297 {
298  boost::filesystem::rename(dcp::filesystem::fix_long_path(old_path), dcp::filesystem::fix_long_path(new_path), ec);
299 }
300 
301 
302 /* We don't really need this but let's add it for completeness */
303 boost::filesystem::path
304 dcp::filesystem::change_extension(boost::filesystem::path const& path, std::string const& new_extension)
305 {
306 #ifdef LIBDCP_HAVE_REPLACE_EXTENSION
307  auto copy = path;
308  copy.replace_extension(new_extension);
309  return copy;
310 #else
311  return boost::filesystem::change_extension(path, new_extension);
312 #endif
313 }
314 
315 
316 #ifdef DCPOMATIC_WINDOWS
317 
318 dcp::filesystem::directory_iterator::directory_iterator(boost::filesystem::path const& path)
319  : _wrapped(dcp::filesystem::fix_long_path(path))
320 {
321 
322 }
323 
324 
325 dcp::filesystem::directory_iterator::directory_iterator(boost::filesystem::path const& path, boost::system::error_code& ec)
326  : _wrapped(dcp::filesystem::fix_long_path(path), ec)
327 {
328 
329 }
330 
331 
332 boost::filesystem::path
333 dcp::filesystem::directory_entry::path() const
334 {
335  return dcp::filesystem::unfix_long_path(_path);
336 }
337 
338 
339 dcp::filesystem::directory_entry::operator boost::filesystem::path const &() const
340 {
341  return dcp::filesystem::unfix_long_path(_path);
342 }
343 
344 
345 dcp::filesystem::recursive_directory_iterator::recursive_directory_iterator(boost::filesystem::path const& path)
346  : _wrapped(dcp::filesystem::fix_long_path(path))
347 {
348 
349 }
350 
351 #else
352 
353 dcp::filesystem::directory_iterator::directory_iterator(boost::filesystem::path const& path)
354  : _wrapped(path)
355 {
356 
357 }
358 
359 
360 dcp::filesystem::directory_iterator::directory_iterator(boost::filesystem::path const& path, boost::system::error_code& ec)
361  : _wrapped(path, ec)
362 {
363 
364 }
365 
366 
367 boost::filesystem::path
368 dcp::filesystem::directory_entry::path() const
369 {
370  return _path;
371 }
372 
373 
374 dcp::filesystem::directory_entry::operator boost::filesystem::path const &() const
375 {
376  return _path;
377 }
378 
379 
380 dcp::filesystem::recursive_directory_iterator::recursive_directory_iterator(boost::filesystem::path const& path)
381  : _wrapped(path)
382 {
383 
384 }
385 
386 #endif
387 
388 
389 dcp::filesystem::directory_entry::directory_entry(boost::filesystem::path const& path)
390  : _path(path)
391 {
392 
393 }
394 
395 
397 dcp::filesystem::directory_iterator::operator++()
398 {
399  ++_wrapped;
400  return *this;
401 }
402 
403 
405 dcp::filesystem::directory_iterator::operator*() const
406 {
407  _entry = dcp::filesystem::directory_entry(*_wrapped);
408  return _entry;
409 }
410 
411 
413 dcp::filesystem::directory_iterator::operator->() const
414 {
415  _entry = dcp::filesystem::directory_entry(_wrapped->path());
416  return &_entry;
417 }
418 
419 bool
420 dcp::filesystem::directory_iterator::operator!=(dcp::filesystem::directory_iterator const& other) const
421 {
422  return _wrapped != other._wrapped;
423 }
424 
425 
427 dcp::filesystem::begin(dcp::filesystem::directory_iterator const& iter)
428 {
429  return iter;
430 }
431 
432 
434 dcp::filesystem::end(dcp::filesystem::directory_iterator const&)
435 {
437 }
438 
439 
441 dcp::filesystem::recursive_directory_iterator::operator++()
442 {
443  ++_wrapped;
444  return *this;
445 }
446 
447 
448 bool
449 dcp::filesystem::recursive_directory_iterator::operator!=(dcp::filesystem::recursive_directory_iterator const& other) const
450 {
451  return _wrapped != other._wrapped;
452 }
453 
454 
456 dcp::filesystem::recursive_directory_iterator::operator*() const
457 {
458  _entry = dcp::filesystem::directory_entry(_wrapped->path());
459  return _entry;
460 }
461 
462 
464 dcp::filesystem::recursive_directory_iterator::operator->() const
465 {
466  _entry = dcp::filesystem::directory_entry(_wrapped->path());
467  return &_entry;
468 }
469 
470 
472 dcp::filesystem::begin(dcp::filesystem::recursive_directory_iterator const& iter)
473 {
474  return iter;
475 }
476 
477 
479 dcp::filesystem::end(dcp::filesystem::recursive_directory_iterator const&)
480 {
482 }
483 
484 
499 boost::filesystem::path
500 dcp::filesystem::fix_long_path(boost::filesystem::path long_path)
501 {
502 #ifdef LIBDCP_WINDOWS
503  using namespace boost::filesystem;
504 
505  if (boost::algorithm::starts_with(long_path.string(), "\\\\")) {
506  /* This could mean it starts with \\ (i.e. a SMB path) or \\?\ (a long path)
507  * or a variety of other things... anyway, we'll leave it alone.
508  */
509  return long_path;
510  }
511 
512  /* We have to make the path canonical but we can't call canonical() on the long path
513  * as it will fail. So we'll sort of do it ourselves (possibly badly).
514  */
515  path fixed = "\\\\?\\";
516  if (long_path.is_absolute()) {
517  fixed += long_path.make_preferred();
518  } else {
519  fixed += filesystem::current_path() / long_path.make_preferred();
520  }
521  return fixed;
522 #else
523  return long_path;
524 #endif
525 }
526 
527 
528 boost::filesystem::path
529 dcp::filesystem::unfix_long_path(boost::filesystem::path long_path)
530 {
531 #ifdef LIBDCP_WINDOWS
532  if (boost::algorithm::starts_with(long_path.string(), "\\\\?\\")) {
533  return long_path.string().substr(4);
534  }
535 #endif
536  return long_path;
537 }
Definition: filesystem.h:92
Namespace for everything in libdcp.
Definition: array_data.h:50