Toolbox snapshot
The Reactive C++ Toolbox
Loading...
Searching...
No Matches
Stream.hpp
Go to the documentation of this file.
1// The Reactive C++ Toolbox.
2// Copyright (C) 2013-2019 Swirly Cloud Limited
3// Copyright (C) 2022 Reactive Markets Limited
4//
5// Licensed under the Apache License, Version 2.0 (the "License");
6// you may not use this file except in compliance with the License.
7// You may obtain a copy of the License at
8//
9// http://www.apache.org/licenses/LICENSE-2.0
10//
11// Unless required by applicable law or agreed to in writing, software
12// distributed under the License is distributed on an "AS IS" BASIS,
13// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14// See the License for the specific language governing permissions and
15// limitations under the License.
16
17#ifndef TOOLBOX_UTIL_STREAM_HPP
18#define TOOLBOX_UTIL_STREAM_HPP
19
23
24namespace toolbox {
25inline namespace util {
26
27namespace detail {
28struct ResetState {};
29TOOLBOX_API std::ostream& operator<<(std::ostream& os, ResetState) noexcept;
30} // namespace detail
31
34constexpr detail::ResetState reset_state{};
35
37template <std::size_t MaxN>
38class OStream final : public OStreamBase<OStream<MaxN>> {
39 public:
41 explicit OStream(std::nullptr_t) noexcept
42 : storage_(nullptr)
43 {
44 }
45
48 : storage_(std::move(storage))
49 {
50 }
51
53 OStream() : storage_(make_storage()) {}
54 ~OStream() = default;
55
56 // Copy.
57 OStream(const OStream&) = delete;
58 OStream& operator=(const OStream&) = delete;
59
60 // Move.
61 OStream(OStream&&) = delete;
63
65 const char* data() const noexcept { return static_cast<const char*>(storage_.get()); }
66 bool empty() const noexcept { return bytes_written_ == 0u; }
67 std::size_t size() const noexcept { return bytes_written_; }
68 std::string_view str() const noexcept { return std::string_view{data(), size()}; }
69
70 // returns false if overflowed or output error.
71 // return true otherwise.
72 explicit operator bool() const noexcept {
73 return !badbit_;
74 }
75
78 {
80 storage_.swap(storage);
81 reset();
82 return storage;
83 }
84
87 {
88 storage_ = std::move(storage);
89 reset();
90 }
91
94 {
95 bytes_written_ = 0u;
96 badbit_ = false;
97 }
98
99 private:
101
102 std::size_t available() noexcept
103 {
104 if (storage_ != nullptr) [[likely]] {
105 return MaxN - bytes_written_;
106 } else {
107 return 0u;
108 }
109 }
110
111 char* wptr() noexcept
112 {
113 assert(storage_ != nullptr);
114 void* buf_base = storage_.get();
115 return static_cast<char*>(buf_base) + bytes_written_;
116 }
117
118 // Required by CRTP base (BasicOStream)
119 char* do_prepare_space(std::size_t num_bytes)
120 {
121 if (num_bytes <= available()) [[likely]] {
122 return wptr();
123 } else {
124 return nullptr;
125 }
126 }
127
128 // Required by CRTP base (BasicOStream)
129 void do_relinquish_space(std::size_t consumed_num_bytes)
130 {
131 bytes_written_ += consumed_num_bytes;
132 assert(bytes_written_ <= MaxN);
133 }
134
135 // Required by CRTP base (BasicOStream)
136 void do_set_badbit()
137 {
138 badbit_ = true;
139 }
140
141 StoragePtr<MaxN> storage_;
142 std::size_t bytes_written_{0u};
143 bool badbit_{false};
144};
145
146template <std::size_t MaxN>
147class OStaticStream final : public OStreamBase<OStaticStream<MaxN>> {
148 public:
149 OStaticStream() = default;
150 ~OStaticStream() = default;
151
152 // Copy.
153 OStaticStream(const OStaticStream&) = delete;
155
156 // Move.
159
160 const char* data() const noexcept { return buf_; }
161 bool empty() const noexcept { return bytes_written_ == 0u; }
162 std::size_t size() const noexcept { return std::min(bytes_written_, MaxN); }
163 std::string_view str() const noexcept { return std::string_view{buf_, size()}; }
164
165 // returns false if overflowed or output error.
166 // return true otherwise.
167 explicit operator bool() const {
168 return !badbit_;
169 }
170
173 {
174 bytes_written_ = 0;
175 badbit_ = false;
176 };
177
178 private:
180
181 std::size_t available() noexcept
182 {
183 return BufSize - bytes_written_;
184 }
185
186 char* wptr() noexcept
187 {
188 return buf_ + bytes_written_;
189 }
190
191 // Required by CRTP base (BasicOStream)
192 char* do_prepare_space(std::size_t num_bytes)
193 {
194 if (num_bytes <= available()) [[likely]] {
195 return wptr();
196 } else {
197 return nullptr;
198 }
199 }
200
201 // Required by CRTP base (BasicOStream)
202 void do_relinquish_space(std::size_t consumed_num_bytes)
203 {
204 std::size_t new_written_count = bytes_written_ + consumed_num_bytes;
205 if (new_written_count > MaxN) [[unlikely]] {
206 badbit_ = true;
207 } else {
208 bytes_written_ = new_written_count;
209 }
210 }
211
212 // Required by CRTP base (BasicOStream)
213 void do_set_badbit()
214 {
215 badbit_ = true;
216 }
217
218 // size the buffer slightly bigger than requested as it allows
219 // for higher performance outputting in OStreamBase. However, this
220 // is an internal detail, not visible to user via API.
221 static constexpr std::size_t BufSize
223
224 char buf_[BufSize];
225 std::size_t bytes_written_{0u};
226 bool badbit_{false};
227};
228
229// Similar to std::ostream_iterator, but this works with any "Streamable" type.
230template <class StreamT>
231 requires Streamable<StreamT>
233 public:
234 OStreamIterator() = delete;
235
236 explicit OStreamIterator(StreamT& os, const char* delim = nullptr) noexcept
237 : os_(&os)
238 , delim_(delim)
239 {
240 }
241
242 template <class T>
243 OStreamIterator& operator=(const T& value)
244 {
245 *os_ << value;
246 if (delim_) [[unlikely]] {
247 *os_ << delim_;
248 }
249 return *this;
250 }
251
252 OStreamIterator& operator*() noexcept { return *this; }
254 OStreamIterator& operator++(int) noexcept { return *this; }
255
256 // required by std::output_iterator concept
258
259 private:
260 // Pointer (instead of reference) because this class needs to be assignable
261 // to satisfy std::output_iterator concept (references can't be assigned).
262 StreamT* os_;
263 const char* delim_{nullptr};
264};
265
266// similar to std::experimental::ostream_joiner, but this works with any "Streamable" type.
267template <class StreamT, class DelimT>
268 requires Streamable<StreamT>
270 public:
271 OStreamJoiner() = delete;
272 ~OStreamJoiner() = default;
273
275 : os_(os)
276 {
277 }
278
280 noexcept(std::is_nothrow_move_constructible_v<DelimT>)
281 : os_(&os)
282 , delim_(std::move(delim))
283 {
284 }
285
286 template <class T>
287 OStreamJoiner& operator=(const T& value)
288 {
289 if (!first_) {
290 *os_ << delim_;
291 }
292 *os_ << value;
293 first_ = false;
294 return *this;
295 }
296
297 OStreamJoiner& operator*() noexcept { return *this; }
298 OStreamJoiner& operator++() noexcept { return *this; }
299 OStreamJoiner& operator++(int) noexcept { return *this; }
300
301 // required by std::output_iterator concept
303
304 private:
305 // Pointer (instead of reference) because this class needs to be assignable
306 // to satisfy std::output_iterator concept (references can't be assigned).
307 StreamT* os_;
308 DelimT delim_;
309 bool first_{true};
310};
311
312static_assert(std::output_iterator<OStreamIterator<OStream<4096>>, const char&>, "");
313static_assert(std::output_iterator<OStreamJoiner<OStream<4096>, int>, const char&>, "");
314
315} // namespace util
316} // namespace toolbox
317
318#endif // TOOLBOX_UTIL_STREAM_HPP
#define TOOLBOX_API
Definition Config.h:39
OStaticStream & operator=(OStaticStream &&)=delete
OStaticStream(OStaticStream &&)=delete
const char * data() const noexcept
Definition Stream.hpp:160
OStaticStream & operator=(const OStaticStream &)=delete
bool empty() const noexcept
Definition Stream.hpp:161
std::size_t size() const noexcept
Definition Stream.hpp:162
void reset() noexcept
Reset the current position back to the beginning of the buffer.
Definition Stream.hpp:172
std::string_view str() const noexcept
Definition Stream.hpp:163
OStaticStream(const OStaticStream &)=delete
OStreamIterator(StreamT &os, const char *delim=nullptr) noexcept
Definition Stream.hpp:236
OStreamIterator & operator*() noexcept
Definition Stream.hpp:252
OStreamIterator & operator=(const T &value)
Definition Stream.hpp:243
OStreamIterator & operator++(int) noexcept
Definition Stream.hpp:254
OStreamIterator & operator++() noexcept
Definition Stream.hpp:253
OStreamJoiner & operator=(const T &value)
Definition Stream.hpp:287
OStreamJoiner & operator++() noexcept
Definition Stream.hpp:298
OStreamJoiner(StreamT &os) noexcept
Definition Stream.hpp:274
OStreamJoiner(StreamT &os, DelimT delim) noexcept(std::is_nothrow_move_constructible_v< DelimT >)
Definition Stream.hpp:279
OStreamJoiner & operator++(int) noexcept
Definition Stream.hpp:299
OStreamJoiner & operator*() noexcept
Definition Stream.hpp:297
OStream uses a dynamic storage acquired from the custom allocator.
Definition Stream.hpp:38
OStream(std::nullptr_t) noexcept
Constructor for initialising the StreamBuf with a null buffer.
Definition Stream.hpp:41
static StoragePtr< MaxN > make_storage()
Definition Stream.hpp:64
void reset() noexcept
Reset the current position back to the beginning of the buffer.
Definition Stream.hpp:93
StoragePtr< MaxN > release_storage() noexcept
Release the managed storage.
Definition Stream.hpp:77
OStream(OStream &&)=delete
OStream & operator=(OStream &&)=delete
std::string_view str() const noexcept
Definition Stream.hpp:68
OStream(StoragePtr< MaxN > storage) noexcept
Constructor for initialising the StreamBuf with pre-allocated storage.
Definition Stream.hpp:47
std::size_t size() const noexcept
Definition Stream.hpp:67
void set_storage(StoragePtr< MaxN > storage) noexcept
Update the internal storage and clear i/o state.
Definition Stream.hpp:86
OStream()
Default constructor allocates required storage.
Definition Stream.hpp:53
bool empty() const noexcept
Definition Stream.hpp:66
OStream & operator=(const OStream &)=delete
const char * data() const noexcept
Definition Stream.hpp:65
OStream(const OStream &)=delete
STL namespace.
StreamT & operator<<(StreamT &os, PutPercentiles pp)
Definition Utility.hpp:68
constexpr detail::ResetState reset_state
Definition Stream.hpp:34
std::unique_ptr< void, detail::StorageDeleter< SizeN > > StoragePtr
Definition Storage.hpp:34
constexpr auto bind() noexcept
Definition Slot.hpp:97