2 // Copyright (c) 2016-2019 Vinnie Falco (vinnie dot falco at gmail dot com)
4 // Distributed under the Boost Software License, Version 1.0. (See accompanying
5 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
7 // Official repository: https://github.com/boostorg/beast
10 #ifndef BOOST_BEAST_CORE_IMPL_ICY_STREAM_HPP
11 #define BOOST_BEAST_CORE_IMPL_ICY_STREAM_HPP
13 #include <boost/beast/core/async_base.hpp>
14 #include <boost/beast/core/buffer_traits.hpp>
15 #include <boost/beast/core/error.hpp>
16 #include <boost/beast/core/stream_traits.hpp>
17 #include <boost/beast/core/detail/is_invocable.hpp>
18 #include <boost/asio/coroutine.hpp>
19 #include <boost/assert.hpp>
20 #include <boost/throw_exception.hpp>
31 template<class ConstBufferSequence>
33 is_icy(ConstBufferSequence const& buffers)
36 auto const n = net::buffer_copy(
37 net::mutable_buffer(buf, 3),
39 if(n >= 1 && buf[0] != 'I')
41 if(n >= 2 && buf[1] != 'C')
43 if(n >= 3 && buf[2] != 'Y')
46 return boost::indeterminate;
52 template<class NextLayer>
53 struct icy_stream<NextLayer>::ops
56 template<class Buffers, class Handler>
58 : public beast::async_base<Handler,
59 beast::executor_type<icy_stream>>
60 , public asio::coroutine
69 template<class Handler_>
75 beast::executor_type<icy_stream>>(
76 std::forward<Handler_>(h), s.get_executor())
80 (*this)({}, 0, false);
86 std::size_t bytes_transferred,
89 BOOST_ASIO_CORO_REENTER(*this)
93 BOOST_ASSERT(s_.n_ == 0);
96 // Try to read the first three characters
99 BOOST_ASIO_HANDLER_LOCATION((
101 "http::icy_stream::async_read_some"));
103 s_.next_layer().async_read_some(
105 s_.buf_ + s_.n_, 3 - s_.n_),
108 s_.n_ += static_cast<char>(bytes_transferred);
111 auto result = detail::is_icy(
112 net::const_buffer(s_.buf_, s_.n_));
113 if(boost::indeterminate(result))
116 s_.n_ = static_cast<char>(net::buffer_copy(
117 net::buffer(s_.buf_, sizeof(s_.buf_)),
118 icy_stream::version()));
125 bytes_transferred = net::buffer_copy(
126 b_, net::const_buffer(s_.buf_, s_.n_));
127 s_.n_ -= static_cast<char>(bytes_transferred);
130 s_.buf_ + bytes_transferred,
131 sizeof(s_.buf_) - bytes_transferred);
135 BOOST_ASIO_CORO_YIELD
137 BOOST_ASIO_HANDLER_LOCATION((
139 "http::icy_stream::async_read_some"));
141 s_.next_layer().async_read_some(
142 b_, std::move(*this));
149 n_ = bytes_transferred;
150 BOOST_ASIO_CORO_YIELD
152 BOOST_ASIO_HANDLER_LOCATION((
154 "http::icy_stream::async_read_some"));
156 s_.next_layer().async_read_some(
157 net::mutable_buffer{},
161 bytes_transferred = n_;
163 this->complete_now(ec, bytes_transferred);
170 template<class ReadHandler, class Buffers>
177 // If you get an error on the following line it means
178 // that your handler does not meet the documented type
179 // requirements for the handler.
182 beast::detail::is_invocable<ReadHandler,
183 void(error_code, std::size_t)>::value,
184 "ReadHandler type requirements not met");
188 typename std::decay<ReadHandler>::type>(
189 std::forward<ReadHandler>(h), *s, b);
195 //------------------------------------------------------------------------------
197 template<class NextLayer>
198 template<class... Args>
199 icy_stream<NextLayer>::
200 icy_stream(Args&&... args)
201 : stream_(std::forward<Args>(args)...)
203 std::memset(buf_, 0, sizeof(buf_));
206 template<class NextLayer>
207 template<class MutableBufferSequence>
209 icy_stream<NextLayer>::
210 read_some(MutableBufferSequence const& buffers)
212 static_assert(is_sync_read_stream<next_layer_type>::value,
213 "SyncReadStream type requirements not met");
214 static_assert(net::is_mutable_buffer_sequence<
215 MutableBufferSequence>::value,
216 "MutableBufferSequence type requirements not met");
218 auto n = read_some(buffers, ec);
220 BOOST_THROW_EXCEPTION(system_error{ec});
224 template<class NextLayer>
225 template<class MutableBufferSequence>
227 icy_stream<NextLayer>::
228 read_some(MutableBufferSequence const& buffers, error_code& ec)
230 static_assert(is_sync_read_stream<next_layer_type>::value,
231 "SyncReadStream type requirements not met");
232 static_assert(net::is_mutable_buffer_sequence<
233 MutableBufferSequence>::value,
234 "MutableBufferSequence type requirements not met");
235 std::size_t bytes_transferred;
238 BOOST_ASSERT(n_ == 0);
241 // Try to read the first three characters
242 bytes_transferred = next_layer().read_some(
243 net::mutable_buffer(buf_ + n_, 3 - n_), ec);
244 n_ += static_cast<char>(bytes_transferred);
247 auto result = detail::is_icy(
248 net::const_buffer(buf_, n_));
249 if(boost::indeterminate(result))
252 n_ = static_cast<char>(net::buffer_copy(
253 net::buffer(buf_, sizeof(buf_)),
254 icy_stream::version()));
261 bytes_transferred = net::buffer_copy(
262 buffers, net::const_buffer(buf_, n_));
263 n_ -= static_cast<char>(bytes_transferred);
266 buf_ + bytes_transferred,
267 sizeof(buf_) - bytes_transferred);
272 next_layer().read_some(buffers, ec);
274 return bytes_transferred;
277 template<class NextLayer>
279 class MutableBufferSequence,
280 BOOST_BEAST_ASYNC_TPARAM2 ReadHandler>
281 BOOST_BEAST_ASYNC_RESULT2(ReadHandler)
282 icy_stream<NextLayer>::
284 MutableBufferSequence const& buffers,
285 ReadHandler&& handler)
287 static_assert(is_async_read_stream<next_layer_type>::value,
288 "AsyncReadStream type requirements not met");
289 static_assert(net::is_mutable_buffer_sequence<
290 MutableBufferSequence >::value,
291 "MutableBufferSequence type requirements not met");
292 return net::async_initiate<
294 void(error_code, std::size_t)>(
295 typename ops::run_read_op{},
301 template<class NextLayer>
302 template<class MutableBufferSequence>
304 icy_stream<NextLayer>::
305 write_some(MutableBufferSequence const& buffers)
307 static_assert(is_sync_write_stream<next_layer_type>::value,
308 "SyncWriteStream type requirements not met");
309 static_assert(net::is_const_buffer_sequence<
310 MutableBufferSequence>::value,
311 "MutableBufferSequence type requirements not met");
312 return stream_.write_some(buffers);
315 template<class NextLayer>
316 template<class MutableBufferSequence>
318 icy_stream<NextLayer>::
319 write_some(MutableBufferSequence const& buffers, error_code& ec)
321 static_assert(is_sync_write_stream<next_layer_type>::value,
322 "SyncWriteStream type requirements not met");
323 static_assert(net::is_const_buffer_sequence<
324 MutableBufferSequence>::value,
325 "MutableBufferSequence type requirements not met");
326 return stream_.write_some(buffers, ec);
329 template<class NextLayer>
331 class MutableBufferSequence,
332 BOOST_BEAST_ASYNC_TPARAM2 WriteHandler>
333 BOOST_BEAST_ASYNC_RESULT2(WriteHandler)
334 icy_stream<NextLayer>::
336 MutableBufferSequence const& buffers,
337 WriteHandler&& handler)
339 static_assert(is_async_write_stream<next_layer_type>::value,
340 "AsyncWriteStream type requirements not met");
341 static_assert(net::is_const_buffer_sequence<
342 MutableBufferSequence>::value,
343 "MutableBufferSequence type requirements not met");
344 return stream_.async_write_some(
345 buffers, std::forward<WriteHandler>(handler));