libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
DirStream.hxx
1#pragma once
2
3// C++
4#include <exception>
5#include <optional>
6#include <string_view>
7
8// cosmos
9#include <cosmos/SysString.hxx>
10#include <cosmos/error/ApiError.hxx>
11#include <cosmos/error/UsageError.hxx>
12#include <cosmos/fs/DirEntry.hxx>
13#include <cosmos/fs/DirFD.hxx>
14#include <cosmos/fs/filesystem.hxx>
15#include <cosmos/utils.hxx>
16
17namespace cosmos {
18
20
30class COSMOS_API DirStream {
31public: // functions
32
34 DirStream() = default;
35
37
40 explicit DirStream(const DirFD fd) {
41 open(fd);
42 }
43
45 DirStream(const DirFD fd, const SysString subpath) {
46 open(fd, subpath);
47 }
48
50 explicit DirStream(const SysString path) {
51 open(path);
52 }
53
54 // Prevent copying due to the directory stream ownership.
55 DirStream(const DirStream&) = delete;
56 DirStream& operator=(const DirStream&) = delete;
57
59 ~DirStream();
60
62
73 void close();
74
76
84 void open(const DirFD fd);
85
87 void open(const DirFD dir_fd, const SysString subpath);
88
90
94 void open(const SysString path, const FollowSymlinks follow_links = FollowSymlinks{false});
95
97 auto isOpen() const { return m_stream != nullptr; }
98
100
106 DirFD fd() const;
107
109
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 }
125
127
130 void seek(const DirEntry::DirPos pos) {
131 requireOpenStream("seek");
132
133 ::seekdir(m_stream, to_integral(pos));
134 }
135
137 void rewind() {
138 ::rewinddir(m_stream);
139 }
140
142
152 std::optional<DirEntry> nextEntry();
153
154protected: // functions
155
156 void open(const FileNum fd);
157
158 void requireOpenStream(const std::string_view context) const {
159 if (!isOpen()) {
160 cosmos_throw (UsageError(std::string(context) + " on unassociated DirStream instance"));
161 }
162 }
163
164protected: // data
165
166 DIR *m_stream = nullptr;
167};
168
169} // end ns
170
171// make DirIterator available for range based for loops
172#include <cosmos/fs/DirIterator.hxx>
Specialized exception type used when system APIs fail.
Definition ApiError.hxx:18
DirPos
strong type for representing directory stream positions
Definition DirEntry.hxx:35
A specialized FileDescriptor for directory objects.
Definition DirFD.hxx:17
Access directory contents in the file system.
Definition DirStream.hxx:30
DirStream()=default
Creates an object not associated with a directory.
DirEntry::DirPos tell() const
Returns the current position in the directory iteration.
void rewind()
Rewind the directory stream to the beginning.
DirStream(const DirFD fd, const SysString subpath)
Open the directory subpath relative to fd.
Definition DirStream.hxx:45
void seek(const DirEntry::DirPos pos)
Adjust the directory iterator to the given position.
DirStream(const SysString path)
Create a DirStream object operating on the directory at the given path location.
Definition DirStream.hxx:50
DirStream(const DirFD fd)
Create a DirStream using the given file descriptor.
Definition DirStream.hxx:40
auto isOpen() const
Indicates whether currently a directory is associated with this object.
Definition DirStream.hxx:97
Strong template type to wrap boolean values in a named type.
Definition utils.hxx:50
Exception type for logical usage errors within the application.
FileNum
Primitive file descriptor.
Definition types.hxx:32
Wrapper type around a C-style string for use with system APIs.
Definition SysString.hxx:33