libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
cosmos::DirStream Class Reference

Access directory contents in the file system. More...

#include <DirStream.hxx>

Public Member Functions

 DirStream ()=default
 Creates an object not associated with a directory.
 
 DirStream (const DirFD fd)
 Create a DirStream using the given file descriptor.
 
 DirStream (const DirFD fd, const SysString subpath)
 Open the directory subpath relative to fd.
 
 DirStream (const SysString path)
 Create a DirStream object operating on the directory at the given path location.
 
 DirStream (const DirStream &)=delete
 
DirStreamoperator= (const DirStream &)=delete
 
 ~DirStream ()
 Closes the underlying directory object, if currently open.
 
void close ()
 Close the currently associated directory.
 
void open (const DirFD fd)
 Associate with the directory represented by the given file descriptor.
 
void open (const DirFD dir_fd, const SysString subpath)
 Open the directory subpath relative to fd.
 
void open (const SysString path, const FollowSymlinks follow_links=FollowSymlinks{false})
 Associate with the directory at the given file system path location.
 
auto isOpen () const
 Indicates whether currently a directory is associated with this object.
 
DirFD fd () const
 Return the file descriptor associated with the current DirStream object.
 
DirEntry::DirPos tell () const
 Returns the current position in the directory iteration.
 
void seek (const DirEntry::DirPos pos)
 Adjust the directory iterator to the given position.
 
void rewind ()
 Rewind the directory stream to the beginning.
 
std::optional< DirEntrynextEntry ()
 Returns the next entry in the associated directory.
 

Protected Member Functions

void open (const FileNum fd)
 
void requireOpenStream (const std::string_view context) const
 

Protected Attributes

DIR * m_stream = nullptr
 

Detailed Description

Access directory contents in the file system.

Using this type you can open directories in the file system either by path or by using an already opened Directory file descriptor. The directory contents can then be iterated over.

Note that the directory contents will be returned by the operating system in an undefined order (i.e. not alphabetically or otherwise sorted). Also entries for "." and ".." by convention should show up and often need to be filtered by applications, if necessary.

Definition at line 30 of file DirStream.hxx.

Constructor & Destructor Documentation

◆ DirStream() [1/3]

cosmos::DirStream::DirStream ( const DirFD fd)
inlineexplicit

Create a DirStream using the given file descriptor.

See also
open(DirFD fd)

Definition at line 40 of file DirStream.hxx.

40 {
41 open(fd);
42 }
void open(const DirFD fd)
Associate with the directory represented by the given file descriptor.
Definition DirStream.cxx:67
DirFD fd() const
Return the file descriptor associated with the current DirStream object.
Definition DirStream.cxx:95

◆ DirStream() [2/3]

cosmos::DirStream::DirStream ( const DirFD fd,
const SysString subpath )
inline

Open the directory subpath relative to fd.

Definition at line 45 of file DirStream.hxx.

45 {
46 open(fd, subpath);
47 }

◆ DirStream() [3/3]

cosmos::DirStream::DirStream ( const SysString path)
inlineexplicit

Create a DirStream object operating on the directory at the given path location.

Definition at line 50 of file DirStream.hxx.

50 {
51 open(path);
52 }

◆ ~DirStream()

cosmos::DirStream::~DirStream ( )

Closes the underlying directory object, if currently open.

Definition at line 14 of file DirStream.cxx.

14 {
15 try {
16 close();
17 } catch (const std::exception &ex) {
18 noncritical_error(
19 sprintf("%s: failed to close directory stream", __FUNCTION__),
20 ex);
21 }
22}
void close()
Close the currently associated directory.
Definition DirStream.cxx:24

Member Function Documentation

◆ close()

void cosmos::DirStream::close ( )

Close the currently associated directory.

This will disassociate the DirStream object and further attempts to iterate over directory contents will fail.

If closing causes an error then an exception is thrown, but the state of the DirStream object will be invalidated, to avoid recurring errors trying to close() or reuse the object.

If the object is not currently associated with a directory then a call to this function does nothing.

Definition at line 24 of file DirStream.cxx.

24 {
25 if (!m_stream) {
26 return;
27 }
28
29 auto ret = closedir(m_stream);
30 m_stream = nullptr;
31
32 if (ret == -1) {
33 cosmos_throw (ApiError("closedir()"));
34 }
35}

◆ fd()

DirFD cosmos::DirStream::fd ( ) const

Return the file descriptor associated with the current DirStream object.

The caller must not modify the state of this file descriptor, otherwise further attempts to iterate over directory contents will result in undefined behaviour. The file descriptor will become invalid after close() is invoked.

Definition at line 95 of file DirStream.cxx.

95 {
96 requireOpenStream(__FUNCTION__);
97 auto fd = dirfd(m_stream);
98 DirFD ret{FileNum{fd}};
99
100 if (ret.invalid()) {
101 cosmos_throw (ApiError("dirfd()"));
102 }
103
104 return ret;
105}
FileNum
Primitive file descriptor.
Definition types.hxx:32

◆ isOpen()

auto cosmos::DirStream::isOpen ( ) const
inline

Indicates whether currently a directory is associated with this object.

Definition at line 97 of file DirStream.hxx.

97{ return m_stream != nullptr; }

◆ nextEntry()

std::optional< DirEntry > cosmos::DirStream::nextEntry ( )

Returns the next entry in the associated directory.

Calls to this function are only allowed if isOpen() returns true. The validity of the returned object is tied to the lifetime of the DirStream instance it came from. Also any call to nextEntry() will invalidate previously returned DirEntry instances returned from the same DirStream instance.

When the end of the directory has been reached then nullopt is returned.

Definition at line 107 of file DirStream.cxx.

107 {
108 requireOpenStream(__FUNCTION__);
109
110 /*
111 * there's a bit confusion between readdir and readdir_r(). Today on
112 * Linux readdir() is thread-safe between different directory streams
113 * but not thread safe when using the same directory stream in
114 * parallel. The latter is rather peculiar and should not be needed.
115 * Therefore use readdir().
116 */
117
118 // needed to differentiate between end-of-stream and error condition
119 reset_errno();
120 const auto entry = readdir(m_stream);
121
122 if (entry) {
123 return DirEntry{entry};
124 }
125
126 if (is_errno_set()) {
127 cosmos_throw(ApiError("readdir()"));
128 }
129
130 return {};
131}
void reset_errno()
Resets the currently set errno to indicate no error.
Definition errno.hxx:113
bool is_errno_set()
Checks whether currently an errno is set.
Definition errno.hxx:117

◆ open() [1/4]

void cosmos::DirStream::open ( const DirFD dir_fd,
const SysString subpath )

Open the directory subpath relative to fd.

Definition at line 82 of file DirStream.cxx.

82 {
83
84 const OpenFlags flags{OpenFlag::DIRECTORY};
85 auto fd = fs::open_at(dir_fd, subpath, OpenMode::READ_ONLY, flags);
86
87 // ownership is transferred to the stream
88 m_stream = ::fdopendir(to_integral(fd.raw()));
89
90 if (!m_stream) {
91 cosmos_throw (ApiError("fdopendir()"));
92 }
93}
FileNum raw() const
Returns the primitive file descriptor contained in the object.
BitMask< OpenFlag > OpenFlags
Collection of OpenFlag used for opening files.
Definition types.hxx:96

◆ open() [2/4]

void cosmos::DirStream::open ( const DirFD fd)

Associate with the directory represented by the given file descriptor.

The implementation operates on a duplicate of the given file descriptor. You must not modify the file descriptor's state, otherwise the usage of the DirStream object will become undefined.

If the object is already associated with another directory then this previous association will be implicitly close()'d.

Definition at line 67 of file DirStream.cxx.

67 {
68 close();
69
70 auto duplicate = fd.duplicate();
71 open(duplicate.raw());
72}
void duplicate(const FileDescriptor new_fd, const CloseOnExec cloexec=CloseOnExec{true}) const
Get a duplicate file descriptor that will further be known as new_fd.

◆ open() [3/4]

void cosmos::DirStream::open ( const FileNum fd)
protected

Definition at line 74 of file DirStream.cxx.

74 {
75 m_stream = ::fdopendir(to_integral(fd));
76
77 if (!m_stream) {
78 cosmos_throw (ApiError("fdopendir()"));
79 }
80}

◆ open() [4/4]

void cosmos::DirStream::open ( const SysString path,
const FollowSymlinks follow_links = FollowSymlinks{false} )

Associate with the directory at the given file system path location.

If the object is already associated with another directory then this previous association will be implicitly close()'d.

Definition at line 37 of file DirStream.cxx.

37 {
38 close();
39
40 /*
41 * reuse the open(DirFD) logic and only open the file
42 * descriptor before. This allows us to pass needful flags like
43 * O_CLOEXEC. This gives us more control than when using opendir().
44 */
45 auto res = ::open(
46 path.raw(),
47 O_RDONLY | O_CLOEXEC | O_DIRECTORY | (follow_links ? O_NOFOLLOW : 0)
48 );
49
50 DirFD fd{FileNum{res}};
51
52 if (fd.invalid()) {
53 cosmos_throw (ApiError("open(O_DIRECTORY)"));
54 }
55
56 try {
57 open(fd.raw());
58 } catch (...) {
59 // intentionally ignore error conditions here
60 try {
61 fd.close();
62 } catch(...) {}
63 throw;
64 }
65}
void close()
Explicitly close the contained FD.

◆ requireOpenStream()

void cosmos::DirStream::requireOpenStream ( const std::string_view context) const
inlineprotected

Definition at line 158 of file DirStream.hxx.

158 {
159 if (!isOpen()) {
160 cosmos_throw (UsageError(std::string(context) + " on unassociated DirStream instance"));
161 }
162 }
auto isOpen() const
Indicates whether currently a directory is associated with this object.
Definition DirStream.hxx:97

◆ rewind()

void cosmos::DirStream::rewind ( )
inline

Rewind the directory stream to the beginning.

Definition at line 137 of file DirStream.hxx.

137 {
138 ::rewinddir(m_stream);
139 }

◆ seek()

void cosmos::DirStream::seek ( const DirEntry::DirPos pos)
inline

Adjust the directory iterator to the given position.

pos needs to be previously obtained from tell().

Definition at line 130 of file DirStream.hxx.

130 {
131 requireOpenStream("seek");
132
133 ::seekdir(m_stream, to_integral(pos));
134 }

◆ tell()

DirEntry::DirPos cosmos::DirStream::tell ( ) const
inline

Returns the current position in the directory iteration.

The returned value needs to be treated opaque, i.e. no assumptions should be made about it. It can merely be used to seek() at a later point in time.

Definition at line 114 of file DirStream.hxx.

114 {
115 requireOpenStream("tell");
116
117 auto ret = ::telldir(m_stream);
118
119 if (ret == -1) {
120 cosmos_throw (ApiError("telldir()"));
121 }
122
123 return DirEntry::DirPos{ret};
124 }
DirPos
strong type for representing directory stream positions
Definition DirEntry.hxx:35

Member Data Documentation

◆ m_stream

DIR* cosmos::DirStream::m_stream = nullptr
protected

Definition at line 166 of file DirStream.hxx.


The documentation for this class was generated from the following files: