18#include <seqan3/contrib/parallel/buffer_queue.hpp>
29namespace seqan3::detail
37template <std::ranges::range urng_t>
38class async_input_buffer_view :
public std::ranges::view_interface<async_input_buffer_view<urng_t>>
41 static_assert(std::ranges::input_range<urng_t>,
42 "The range parameter to async_input_buffer_view must be at least a std::ranges::input_range.");
43 static_assert(std::ranges::view<urng_t>,
44 "The range parameter to async_input_buffer_view must model std::ranges::view.");
45 static_assert(std::movable<std::ranges::range_value_t<urng_t>>,
46 "The range parameter to async_input_buffer_view must have a value_type that is std::movable.");
48 std::constructible_from<std::ranges::range_value_t<urng_t>,
50 "The range parameter to async_input_buffer_view must have a value_type that is constructible by a moved "
51 "value of its reference type.");
54 using urng_iterator_type = std::ranges::iterator_t<urng_t>;
63 contrib::fixed_buffer_queue<std::ranges::range_value_t<urng_t>> buffer;
79 async_input_buffer_view() =
default;
80 async_input_buffer_view(async_input_buffer_view
const &) =
default;
81 async_input_buffer_view(async_input_buffer_view &&) =
default;
82 async_input_buffer_view & operator=(async_input_buffer_view
const &) =
default;
83 async_input_buffer_view & operator=(async_input_buffer_view &&) =
default;
84 ~async_input_buffer_view() =
default;
87 async_input_buffer_view(urng_t _urng,
size_t const buffer_size)
89 auto deleter = [](state * p)
100 new state{std::move(_urng),
101 contrib::fixed_buffer_queue<std::ranges::range_value_t<urng_t>>{buffer_size},
105 auto runner = [&state = *state_ptr]()
107 for (
auto && val : state.urange)
108 if (state.buffer.wait_push(
std::
move(val)) == contrib::queue_op_status::closed)
111 state.buffer.close();
118 template <
typename other_urng_t>
119 requires (!std::same_as<std::remove_cvref_t<other_urng_t>, async_input_buffer_view>)
121 std::ranges::viewable_range<other_urng_t>
123 async_input_buffer_view(other_urng_t && _urng,
size_t const buffer_size) :
124 async_input_buffer_view{std::views::all(_urng), buffer_size}
144 assert(state_ptr !=
nullptr);
145 return {state_ptr->buffer};
149 iterator
begin()
const =
delete;
152 std::default_sentinel_t
end()
154 return std::default_sentinel;
158 std::default_sentinel_t
end()
const =
delete;
163template <std::ranges::range urng_t>
164class async_input_buffer_view<urng_t>::iterator
167 using sentinel_type = std::default_sentinel_t;
170 contrib::fixed_buffer_queue<std::ranges::range_value_t<urng_t>> * buffer_ptr =
nullptr;
173 mutable std::ranges::range_value_t<urng_t> cached_value;
187 using pointer = detail::iter_pointer_t<urng_iterator_type>;
193 using iterator_concept = iterator_category;
200 iterator() =
default;
202 iterator(iterator
const & rhs) =
default;
203 iterator(iterator && rhs) =
default;
205 iterator & operator=(iterator
const & rhs) =
default;
206 iterator & operator=(iterator && rhs) =
default;
207 ~iterator() noexcept = default;
210 iterator(contrib::fixed_buffer_queue<
std::ranges::range_value_t<urng_t>> & buffer) noexcept : buffer_ptr{&buffer}
220 reference operator*() const noexcept
226 pointer operator->() const noexcept
236 iterator & operator++() noexcept
241 assert(buffer_ptr !=
nullptr);
243 if (buffer_ptr->wait_pop(cached_value) == contrib::queue_op_status::closed)
250 void operator++(
int)
noexcept
260 friend constexpr bool operator==(iterator
const & lhs, std::default_sentinel_t
const &)
noexcept
266 friend constexpr bool operator==(std::default_sentinel_t
const &, iterator
const & rhs)
noexcept
268 return rhs == std::default_sentinel_t{};
272 friend constexpr bool operator!=(iterator
const & lhs, std::default_sentinel_t
const &)
noexcept
274 return !(lhs == std::default_sentinel_t{});
278 friend constexpr bool operator!=(std::default_sentinel_t
const &, iterator
const & rhs)
noexcept
280 return rhs != std::default_sentinel_t{};
291template <std::ranges::viewable_range urng_t>
292async_input_buffer_view(urng_t &&,
size_t const buffer_size) -> async_input_buffer_view<std::views::all_t<urng_t>>;
300struct async_input_buffer_fn
303 constexpr auto operator()(
size_t const buffer_size)
const
305 return detail::adaptor_from_functor{*
this, buffer_size};
313 template <std::ranges::range urng_t>
314 constexpr auto operator()(urng_t && urange,
size_t const buffer_size)
const
316 static_assert(std::ranges::input_range<urng_t>,
317 "The range parameter to views::async_input_buffer must be at least a std::ranges::input_range.");
318 static_assert(std::ranges::viewable_range<urng_t>,
319 "The range parameter to views::async_input_buffer cannot be a temporary of a non-view range.");
320 static_assert(std::movable<std::ranges::range_value_t<urng_t>>,
321 "The range parameter to views::async_input_buffer must have a value_type that is std::movable.");
323 std::constructible_from<std::ranges::range_value_t<urng_t>,
325 "The range parameter to views::async_input_buffer must have a value_type that is constructible by a moved "
326 "value of its reference type.");
328 if (buffer_size == 0)
331 return detail::async_input_buffer_view{std::forward<urng_t>(urange), buffer_size};
Provides seqan3::detail::adaptor_from_functor.
constexpr auto async_input_buffer
A view adapter that returns a concurrent-queue-like view over the underlying range.
Definition async_input_buffer.hpp:478
The SeqAn namespace for views.
Definition char_strictly_to.hpp:19
SeqAn specific customisations in the standard namespace.