libwreport  3.29
sys.h
1 #ifndef WREPORT_SYS_H
2 #define WREPORT_SYS_H
3 
11 #include <string>
12 #include <memory>
13 #include <iterator>
14 #include <sys/types.h>
15 #include <sys/stat.h>
16 #include <sys/time.h>
17 #include <sys/resource.h>
18 #include <unistd.h>
19 #include <dirent.h>
20 #include <fcntl.h>
21 
22 namespace wreport {
23 namespace sys {
24 
30 std::unique_ptr<struct stat> stat(const std::string& pathname);
31 
36 void stat(const std::string& pathname, struct stat& st);
37 
43 bool isdir(const std::string& pathname);
44 
46 bool isblk(const std::string& pathname);
47 
49 bool ischr(const std::string& pathname);
50 
52 bool isfifo(const std::string& pathname);
53 
55 bool islnk(const std::string& pathname);
56 
58 bool isreg(const std::string& pathname);
59 
61 bool issock(const std::string& pathname);
62 
64 time_t timestamp(const std::string& file);
65 
67 time_t timestamp(const std::string& file, time_t def);
68 
70 size_t size(const std::string& file);
71 
73 size_t size(const std::string& file, size_t def);
74 
76 ino_t inode(const std::string& file);
77 
79 ino_t inode(const std::string& file, ino_t def);
80 
82 bool access(const std::string& s, int m);
83 
85 bool exists(const std::string& s);
86 
88 std::string getcwd();
89 
91 void chdir(const std::string& dir);
92 
94 void chroot(const std::string& dir);
95 
97 mode_t umask(mode_t mask);
98 
100 std::string abspath(const std::string& pathname);
101 
107 class MMap
108 {
109  void* addr;
110  size_t length;
111 
112 public:
113  MMap(const MMap&) = delete;
114  MMap(MMap&&);
115  MMap(void* addr, size_t length);
116  ~MMap();
117 
118  MMap& operator=(const MMap&) = delete;
119  MMap& operator=(MMap&&);
120 
121  size_t size() const { return length; }
122 
123  void munmap();
124 
125  template<typename T>
126  operator const T*() const { return reinterpret_cast<const T*>(addr); }
127 
128  template<typename T>
129  operator T*() const { return reinterpret_cast<T*>(addr); }
130 };
131 
144 {
145 protected:
146  int fd = -1;
147 
148 public:
149  FileDescriptor();
151  FileDescriptor(int fd);
152  virtual ~FileDescriptor();
153 
154  // We can copy at the FileDescriptor level because the destructor does not
155  // close fd
156  FileDescriptor(const FileDescriptor& o) = default;
157  FileDescriptor& operator=(const FileDescriptor& o) = default;
158 
166  [[noreturn]] virtual void throw_error(const char* desc);
167 
175  [[noreturn]] virtual void throw_runtime_error(const char* desc);
176 
178  bool is_open() const;
179 
185  void close();
186 
187  void fstat(struct stat& st);
188  void fchmod(mode_t mode);
189 
190  void futimens(const struct ::timespec ts[2]);
191 
192  void fsync();
193  void fdatasync();
194 
195  int dup();
196 
197  size_t read(void* buf, size_t count);
198 
206  bool read_all_or_retry(void* buf, size_t count);
207 
212  void read_all_or_throw(void* buf, size_t count);
213 
214  size_t write(const void* buf, size_t count);
215 
216  template<typename Container>
217  size_t write(const Container& c)
218  {
219  return write(c.data(), c.size() * sizeof(Container::value_type));
220  }
221 
223  void write_all_or_retry(const void* buf, size_t count);
224 
225  template<typename Container>
226  void write_all_or_retry(const Container& c)
227  {
228  write_all_or_retry(c.data(), c.size() * sizeof(typename Container::value_type));
229  }
230 
235  void write_all_or_throw(const void* buf, size_t count);
236 
237  template<typename Container>
238  void write_all_or_throw(const Container& c)
239  {
240  write_all_or_throw(c.data(), c.size() * sizeof(typename Container::value_type));
241  }
242 
243  off_t lseek(off_t offset, int whence=SEEK_SET);
244 
245  size_t pread(void* buf, size_t count, off_t offset);
246  size_t pwrite(const void* buf, size_t count, off_t offset);
247 
248  template<typename Container>
249  size_t pwrite(const Container& c, off_t offset)
250  {
251  return pwrite(c.data(), c.size() * sizeof(typename Container::value_type), offset);
252  }
253 
254  void ftruncate(off_t length);
255 
256  MMap mmap(size_t length, int prot, int flags, off_t offset=0);
257 
264  bool ofd_setlk(struct ::flock&);
265 
275  bool ofd_setlkw(struct ::flock&, bool retry_on_signal=true);
276 
282  bool ofd_getlk(struct ::flock&);
283 
290  void sendfile(FileDescriptor& out_fd, off_t offset, size_t count);
291 
293  int getfl();
294 
296  void setfl(int flags);
297 
298  operator int() const { return fd; }
299 };
300 
301 
306 {
307 protected:
308  FileDescriptor fd;
309  struct ::timespec ts[2];
310 
311 public:
314 };
315 
316 
317 
322 {
323 protected:
324  std::string pathname;
325 
326 public:
327  NamedFileDescriptor(int fd, const std::string& pathname);
330 
331  // We can copy at the NamedFileDescriptor level because the destructor does not
332  // close fd
333  NamedFileDescriptor(const NamedFileDescriptor& o) = default;
334  NamedFileDescriptor& operator=(const NamedFileDescriptor& o) = default;
335 
336  [[noreturn]] virtual void throw_error(const char* desc);
337  [[noreturn]] virtual void throw_runtime_error(const char* desc);
338 
340  const std::string& name() const { return pathname; }
341 };
342 
343 
348 {
349  using NamedFileDescriptor::NamedFileDescriptor;
350 
353 
362 
363  ManagedNamedFileDescriptor& operator=(const ManagedNamedFileDescriptor&) = delete;
365 };
366 
367 
372 {
376  struct iterator : public std::iterator<std::input_iterator_tag, struct dirent>
377  {
378  Path* path = nullptr;
379  DIR* dir = nullptr;
380  struct dirent* cur_entry = nullptr;
381 
382  // End iterator
383  iterator();
384  // Start iteration on dir
385  iterator(Path& dir);
386  iterator(iterator&) = delete;
387  iterator(iterator&& o)
388  : dir(o.dir), cur_entry(o.cur_entry)
389  {
390  o.dir = nullptr;
391  o.cur_entry = nullptr;
392  }
393  ~iterator();
394  iterator& operator=(iterator&) = delete;
395  iterator& operator=(iterator&&) = delete;
396 
397  bool operator==(const iterator& i) const;
398  bool operator!=(const iterator& i) const;
399  struct dirent& operator*() const { return *cur_entry; }
400  struct dirent* operator->() const { return cur_entry; }
401  void operator++();
402 
404  bool isdir() const;
405 
407  bool isblk() const;
408 
410  bool ischr() const;
411 
413  bool isfifo() const;
414 
416  bool islnk() const;
417 
419  bool isreg() const;
420 
422  bool issock() const;
423 
425  Path open_path(int flags=0) const;
426  };
427 
428  using ManagedNamedFileDescriptor::ManagedNamedFileDescriptor;
429 
433  Path(const char* pathname, int flags=0, mode_t mode=0777);
437  Path(const std::string& pathname, int flags=0, mode_t mode=0777);
441  Path(Path& parent, const char* pathname, int flags=0, mode_t mode=0777);
442  Path(const Path&) = delete;
443  Path(Path&&) = default;
444  Path& operator=(const Path&) = delete;
445  Path& operator=(Path&&) = default;
446 
448  void open(int flags, mode_t mode=0777);
449 
450  DIR* fdopendir();
451 
454 
457 
458  int openat(const char* pathname, int flags, mode_t mode=0777);
459 
461  int openat_ifexists(const char* pathname, int flags, mode_t mode=0777);
462 
463  bool faccessat(const char* pathname, int mode, int flags=0);
464 
465  void fstatat(const char* pathname, struct stat& st);
466 
468  bool fstatat_ifexists(const char* pathname, struct stat& st);
469 
471  void lstatat(const char* pathname, struct stat& st);
472 
474  bool lstatat_ifexists(const char* pathname, struct stat& st);
475 
476  void unlinkat(const char* pathname);
477 
478  void mkdirat(const char* pathname, mode_t mode=0777);
479 
481  void rmdirat(const char* pathname);
482 
483  void symlinkat(const char* target, const char* linkpath);
484 
485  std::string readlinkat(const char* pathname);
486 
492  void rmtree();
493 
494  static std::string mkdtemp(const std::string& prefix);
495  static std::string mkdtemp(const char* prefix);
496  static std::string mkdtemp(char* pathname_template);
497 };
498 
499 
504 {
505 public:
506  using ManagedNamedFileDescriptor::ManagedNamedFileDescriptor;
507 
508  File(File&&) = default;
509  File(const File&) = delete;
510 
514  File(const std::string& pathname);
515 
517  File(const std::string& pathname, int flags, mode_t mode=0777);
518 
519  File& operator=(const File&) = delete;
520  File& operator=(File&&) = default;
521 
523  void open(int flags, mode_t mode=0777);
524 
529  bool open_ifexists(int flags, mode_t mode=0777);
530 
531  static File mkstemp(const std::string& prefix);
532  static File mkstemp(const char* prefix);
533  static File mkstemp(char* pathname_template);
534 };
535 
536 
542 class Tempfile : public File
543 {
544 protected:
545  bool m_unlink_on_exit = true;
546 
547 public:
548  Tempfile();
549  Tempfile(const std::string& prefix);
550  Tempfile(const char* prefix);
551  ~Tempfile();
552 
554  void unlink_on_exit(bool val);
555 
557  void unlink();
558 };
559 
560 
567 class Tempdir : public Path
568 {
569 protected:
570  bool m_rmtree_on_exit = true;
571 
572 public:
573  Tempdir();
574  Tempdir(const std::string& prefix);
575  Tempdir(const char* prefix);
576  ~Tempdir();
577 
579  void rmtree_on_exit(bool val);
580 };
581 
582 
584 std::string read_file(const std::string &file);
585 
592 void write_file(const std::string& file, const std::string& data, mode_t mode=0777);
593 
600 void write_file(const std::string& file, const void* data, size_t size, mode_t mode=0777);
601 
611 void write_file_atomically(const std::string& file, const std::string& data, mode_t mode=0777);
612 
622 void write_file_atomically(const std::string& file, const void* data, size_t size, mode_t mode=0777);
623 
624 #if 0
625 // Create a temporary directory based on a template.
626 std::string mkdtemp(std::string templ);
627 
630 void mkFilePath(const std::string& file);
631 #endif
632 
638 bool unlink_ifexists(const std::string& file);
639 
645 bool rename_ifexists(const std::string& src, const std::string& dst);
646 
655 bool mkdir_ifmissing(const char* pathname, mode_t mode=0777);
656 
657 bool mkdir_ifmissing(const std::string& pathname, mode_t mode=0777);
658 
665 bool makedirs(const std::string& pathname, mode_t=0777);
666 
674 std::string which(const std::string& name);
675 
677 void unlink(const std::string& pathname);
678 
680 void rmdir(const std::string& pathname);
681 
683 void rmtree(const std::string& pathname);
684 
690 bool rmtree_ifexists(const std::string& pathname);
691 
698 void rename(const std::string& src_pathname, const std::string& dst_pathname);
699 
703 void touch(const std::string& pathname, time_t ts);
704 
708 void clock_gettime(::clockid_t clk_id, struct ::timespec& ts);
709 
713 unsigned long long timesec_elapsed(const struct ::timespec& begin, const struct ::timespec& until);
714 
718 struct Clock
719 {
720  ::clockid_t clk_id;
721  struct ::timespec ts;
722 
726  Clock(::clockid_t clk_id);
727 
732  unsigned long long elapsed();
733 };
734 
739 void getrlimit(int resource, struct ::rlimit& rlim);
741 
743 void setrlimit(int resource, const struct ::rlimit& rlim);
744 
747 {
748  int resource;
749  struct ::rlimit orig;
750 
751  OverrideRlimit(int resource, rlim_t rlim);
752  ~OverrideRlimit();
753 
755  void set(rlim_t rlim);
756 };
757 
758 }
759 }
760 
761 #endif
wreport::sys::OverrideRlimit::set
void set(rlim_t rlim)
Change the limit value again.
wreport::sys::FileDescriptor::throw_error
virtual void throw_error(const char *desc)
Throw an exception based on errno and the given message.
wreport::sys::Clock::elapsed
unsigned long long elapsed()
Return the number of nanoseconds elapsed since the last time ts was updated.
wreport::sys::Path::iterator::islnk
bool islnk() const
wreport::sys::Path::iterator::issock
bool issock() const
wreport::sys::FileDescriptor::ofd_getlk
bool ofd_getlk(struct ::flock &)
Open file description locks F_OFD_GETLK operation.
wreport::sys::NamedFileDescriptor::name
const std::string & name() const
Return the file pathname.
Definition: sys.h:340
wreport::sys::Path::rmdirat
void rmdirat(const char *pathname)
unlinkat with the AT_REMOVEDIR flag set
wreport::sys::Path::begin
iterator begin()
Begin iterator on all directory entries.
wreport::sys::FileDescriptor::read_all_or_throw
void read_all_or_throw(void *buf, size_t count)
Read all the data into buf, throwing runtime_error in case of a partial read.
wreport::sys::Path
Wrap a path on the file system opened with O_PATH.
Definition: sys.h:372
wreport::sys::FileDescriptor::sendfile
void sendfile(FileDescriptor &out_fd, off_t offset, size_t count)
Call sendfile with this file as in_fd, falling back on write if it is not available.
wreport::sys::Path::iterator::isblk
bool isblk() const
wreport::sys::Path::lstatat
void lstatat(const char *pathname, struct stat &st)
fstatat with the AT_SYMLINK_NOFOLLOW flag set
wreport::sys::Path::iterator::ischr
bool ischr() const
wreport::sys::File::open
void open(int flags, mode_t mode=0777)
Wrapper around open(2)
wreport::sys::FileDescriptor::write_all_or_retry
void write_all_or_retry(const void *buf, size_t count)
Write all the data in buf, retrying partial writes.
wreport::sys::Path::Path
Path(const char *pathname, int flags=0, mode_t mode=0777)
Open the given pathname with flags | O_PATH.
wreport::sys::Tempdir::rmtree_on_exit
void rmtree_on_exit(bool val)
Change the rmtree-on-exit behaviour.
wreport::sys::Path::iterator::open_path
Path open_path(int flags=0) const
Return a Path object for this entry.
wreport::sys::Path::iterator::isfifo
bool isfifo() const
wreport::sys::ManagedNamedFileDescriptor
File descriptor that gets automatically closed in the object destructor.
Definition: sys.h:348
wreport::sys::Path::Path
Path(const std::string &pathname, int flags=0, mode_t mode=0777)
Open the given pathname with flags | O_PATH.
wreport::sys::FileDescriptor::setfl
void setfl(int flags)
Set open flags for the file.
wreport::sys::OverrideRlimit
Override a soft resource limit during the lifetime of the object.
Definition: sys.h:747
wreport::sys::ManagedNamedFileDescriptor::~ManagedNamedFileDescriptor
~ManagedNamedFileDescriptor()
The destructor closes the file descriptor, but does not check errors on ::close().
wreport::sys::Tempfile
Open a temporary file.
Definition: sys.h:543
wreport::sys::Path::iterator::isreg
bool isreg() const
wreport::sys::Path::iterator
Iterator for directory entries.
Definition: sys.h:377
wreport::sys::File::open_ifexists
bool open_ifexists(int flags, mode_t mode=0777)
Wrap open(2) and return false instead of throwing an exception if open fails with ENOENT.
wreport::sys::File::File
File(const std::string &pathname)
Create an unopened File object for the given pathname.
wreport::sys::FileDescriptor::throw_runtime_error
virtual void throw_runtime_error(const char *desc)
Throw a runtime_error unrelated from errno.
wreport::sys::FileDescriptor::getfl
int getfl()
Get open flags for the file.
wreport::sys::Path::rmtree
void rmtree()
Delete the directory pointed to by this Path, with all its contents.
wreport::sys::Path::open
void open(int flags, mode_t mode=0777)
Wrapper around open(2) with flags | O_PATH.
wreport::sys::Path::end
iterator end()
End iterator on all directory entries.
wreport::sys::MMap
Wraps a mmapped memory area, unmapping it on destruction.
Definition: sys.h:108
wreport::sys::Tempfile::unlink_on_exit
void unlink_on_exit(bool val)
Change the unlink-on-exit behaviour.
wreport::sys::Path::iterator::isdir
bool isdir() const
wreport::sys::PreserveFileTimes
RAII mechanism to save restore file times at the end of some file operations.
Definition: sys.h:306
wreport::sys::NamedFileDescriptor
File descriptor with a name.
Definition: sys.h:322
wreport::sys::Tempfile::unlink
void unlink()
Unlink the file right now.
wreport::sys::NamedFileDescriptor::throw_error
virtual void throw_error(const char *desc)
Throw an exception based on errno and the given message.
wreport::sys::FileDescriptor
Common operations on file descriptors.
Definition: sys.h:144
wreport::sys::FileDescriptor::ofd_setlk
bool ofd_setlk(struct ::flock &)
Open file description locks F_OFD_SETLK operation.
wreport::sys::Path::openat_ifexists
int openat_ifexists(const char *pathname, int flags, mode_t mode=0777)
Same as openat, but returns -1 if the file does not exist.
wreport::sys::FileDescriptor::ofd_setlkw
bool ofd_setlkw(struct ::flock &, bool retry_on_signal=true)
Open file description locks F_OFD_SETLKW operation.
wreport::sys::Path::Path
Path(Path &parent, const char *pathname, int flags=0, mode_t mode=0777)
Open the given pathname calling parent.openat, with flags | O_PATH.
wreport::sys::FileDescriptor::write_all_or_throw
void write_all_or_throw(const void *buf, size_t count)
Write all the data in buf, throwing runtime_error in case of a partial write.
wreport::sys::Path::lstatat_ifexists
bool lstatat_ifexists(const char *pathname, struct stat &st)
lstatat, but in case of ENOENT returns false instead of throwing
wreport::sys::FileDescriptor::read_all_or_retry
bool read_all_or_retry(void *buf, size_t count)
Read count bytes into bufr, retrying partial reads, stopping at EOF.
wreport::sys::FileDescriptor::close
void close()
Close the file descriptor, setting its value to -1.
wreport::sys::Tempdir
Open a temporary directory.
Definition: sys.h:568
wreport::sys::File
File in the file system.
Definition: sys.h:504
wreport::sys::FileDescriptor::is_open
bool is_open() const
Check if the file descriptor is open (that is, if it is not -1)
wreport::sys::NamedFileDescriptor::throw_runtime_error
virtual void throw_runtime_error(const char *desc)
Throw a runtime_error unrelated from errno.
wreport
String functions.
Definition: benchmark.h:13
wreport::sys::Path::fstatat_ifexists
bool fstatat_ifexists(const char *pathname, struct stat &st)
fstatat, but in case of ENOENT returns false instead of throwing
wreport::sys::File::File
File(const std::string &pathname, int flags, mode_t mode=0777)
Wrapper around open(2)
wreport::sys::Clock::Clock
Clock(::clockid_t clk_id)
Initialize ts with the value of the given clock.
wreport::sys::Clock
Access to clock_gettime.
Definition: sys.h:719