LCOV - code coverage report
Current view: top level - capy - read.hpp (source / functions) Coverage Total Hit
Test: coverage_remapped.info Lines: 100.0 % 6 6
Test Date: 2026-05-29 21:45:20 Functions: 100.0 % 7 7

           TLA  Line data    Source code
       1                 : //
       2                 : // Copyright (c) 2025 Vinnie Falco (vinnie.falco@gmail.com)
       3                 : //
       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)
       6                 : //
       7                 : // Official repository: https://github.com/cppalliance/capy
       8                 : //
       9                 : 
      10                 : #ifndef BOOST_CAPY_READ_HPP
      11                 : #define BOOST_CAPY_READ_HPP
      12                 : 
      13                 : #include <boost/capy/detail/config.hpp>
      14                 : #include <boost/capy/cond.hpp>
      15                 : #include <boost/capy/io_task.hpp>
      16                 : #include <boost/capy/buffers.hpp>
      17                 : #include <boost/capy/buffers/buffer_slice.hpp>
      18                 : #include <boost/capy/concept/dynamic_buffer.hpp>
      19                 : #include <boost/capy/concept/read_source.hpp>
      20                 : #include <boost/capy/concept/read_stream.hpp>
      21                 : #include <system_error>
      22                 : 
      23                 : #include <cstddef>
      24                 : 
      25                 : namespace boost {
      26                 : namespace capy {
      27                 : 
      28                 : /** Read data from a stream until the buffer sequence is full.
      29                 : 
      30                 :     @par Await-effects
      31                 : 
      32                 :     Reads data from `stream` via awaiting `stream.read_some` repeatedly
      33                 :     until:
      34                 : 
      35                 :     @li either the entire buffer sequence  @c buffers is filled,
      36                 :     @li or a contingency occurs.
      37                 : 
      38                 :     If `buffer_size(buffers) == 0` then no awaiting `stream.read_some`
      39                 :     is performed. This is not a contingency.
      40                 : 
      41                 :     @par Await-returns
      42                 :     An object of type `io_result<std::size_t>` destructuring as `[ec, n]`.
      43                 : 
      44                 :     Upon a contingency, `n` represents the number of bytes read so far,
      45                 :     inclusive of the last partial read.
      46                 : 
      47                 :     Contingencies:
      48                 : 
      49                 :     @li The first contingency reported from awaiting @c stream.read_some .
      50                 : 
      51                 :     Notable conditions:
      52                 : 
      53                 :     @li @c cond::canceled — Operation was cancelled,
      54                 :     @li @c cond::eof — Stream reached end before `buffers` was filled.
      55                 : 
      56                 :     @par Await-postcondition
      57                 :     `ec || n == buffer_size(buffers)`.
      58                 : 
      59                 :     @param stream The stream to read from. If the lifetime of `stream` ends
      60                 :     before the coroutine finishes, the behavior is undefined.
      61                 : 
      62                 :     @param buffers The buffer sequence to fill. If the lifetime of the buffer
      63                 :     sequence represented by `buffers` ends before the coroutine finishes, the behavior is undefined.
      64                 : 
      65                 : 
      66                 :     @par Remarks
      67                 :     Supports _IoAwaitable cancellation_.
      68                 : 
      69                 : 
      70                 :     @par Example
      71                 : 
      72                 :     @code
      73                 :     capy::task<> process_message(capy::ReadStream auto& stream)
      74                 :     {
      75                 :         std::vector<char> header(16);  // known header size for some protocol
      76                 :         auto [ec, n] = co_await capy::read(stream, capy::mutable_buffer(header));
      77                 :         if (ec == capy::cond::eof)
      78                 :             co_return;  // Connection closed
      79                 :         if (ec)
      80                 :             throw std::system_error(ec);
      81                 : 
      82                 :         // at this point `header` contains exactly 16 bytes
      83                 :     }
      84                 :     @endcode
      85                 : 
      86                 :     @see ReadStream, MutableBufferSequence
      87                 : */
      88                 : template <typename S, typename MB>
      89                 :   requires ReadStream<S> && MutableBufferSequence<MB>
      90                 : auto
      91 HIT          94 : read(S& stream, MB buffers) ->
      92                 :         io_task<std::size_t>
      93                 : {
      94                 :     auto consuming = buffer_slice(buffers);
      95                 :     std::size_t const total_size = buffer_size(buffers);
      96                 :     std::size_t total_read = 0;
      97                 : 
      98                 :     while(total_read < total_size)
      99                 :     {
     100                 :         auto [ec, n] = co_await stream.read_some(consuming.data());
     101                 :         consuming.remove_prefix(n);
     102                 :         total_read += n;
     103                 :         if(ec)
     104                 :             co_return {ec, total_read};
     105                 :     }
     106                 : 
     107                 :     co_return {{}, total_read};
     108             188 : }
     109                 : 
     110                 : /** Read all data from a stream into a dynamic buffer.
     111                 : 
     112                 :     @par Await-effects
     113                 : 
     114                 :     Reads data from `stream` via awaiting `stream.read_some` repeatedly
     115                 :     and appending the results to `dynbuf`,
     116                 :     until a contingency occurs.
     117                 : 
     118                 :     Data is appended using prepare/commit semantics.
     119                 :     The buffer grows with 1.5x factor when filled.
     120                 : 
     121                 :     @par Await-returns
     122                 : 
     123                 :     An object of type `io_result<std::size_t>` destructuring as `[ec, n]`.
     124                 : 
     125                 :     `n` represents the total number of bytes read,
     126                 :     inclusive of the last partial read.
     127                 : 
     128                 :     Contingencies:
     129                 : 
     130                 :     @li The first contingency, other than one matching to @c cond::eof, reported from awaiting @c stream.read_some .
     131                 : 
     132                 :     @par Await-throws
     133                 :     `std::bad_alloc` when append to `dynbuf` fails.
     134                 : 
     135                 :     @param stream The stream to read from. If the lifetime of `stream` ends
     136                 :     before the coroutine finishes, the behavior is undefined.
     137                 : 
     138                 :     @param dynbuf The dynamic buffer to append data to. If the lifetime of the buffer
     139                 :     sequence represented by `dynbuf` ends before the coroutine finishes, the behavior is undefined.
     140                 : 
     141                 :     @param initial_amount Initial bytes to prepare (default 2048).
     142                 : 
     143                 :     
     144                 :     @par Remarks
     145                 :     Supports _IoAwaitable cancellation_.
     146                 : 
     147                 :     @par Example
     148                 : 
     149                 :     @code
     150                 :     capy::task<std::string> read_body(capy::ReadStream auto& stream)
     151                 :     {
     152                 :         std::string body;
     153                 :         auto [ec, n] = co_await capy::read(stream, capy::dynamic_buffer(body));
     154                 :         if (ec)
     155                 :             throw std::system_error(ec);
     156                 :         return body;
     157                 :     }
     158                 :     @endcode
     159                 : 
     160                 :     @see read_some, ReadStream, DynamicBufferParam
     161                 : */
     162                 : template <typename S, typename DB>
     163                 :   requires ReadStream<S> && DynamicBufferParam<DB>
     164                 : auto
     165              80 : read(
     166                 :     S& stream,
     167                 :     DB&& dynbuf,
     168                 :     std::size_t initial_amount = 2048) ->
     169                 :         io_task<std::size_t>
     170                 : {
     171                 :     std::size_t amount = initial_amount;
     172                 :     std::size_t total_read = 0;
     173                 :     for(;;)
     174                 :     {
     175                 :         auto mb = dynbuf.prepare(amount);
     176                 :         auto const mb_size = buffer_size(mb);
     177                 :         auto [ec, n] = co_await stream.read_some(mb);
     178                 :         dynbuf.commit(n);
     179                 :         total_read += n;
     180                 :         if(ec == cond::eof)
     181                 :             co_return {{}, total_read};
     182                 :         if(ec)
     183                 :             co_return {ec, total_read};
     184                 :         if(n == mb_size)
     185                 :             amount = amount / 2 + amount;
     186                 :     }
     187             160 : }
     188                 : 
     189                 : /** Read all data from a source into a dynamic buffer.
     190                 : 
     191                 :     @par Await-effects
     192                 : 
     193                 :     Reads data from `stream` by calling `source.read` repeatedly 
     194                 :     and appending it to `dynbuf` until a contingency occurs.
     195                 :     The last, potenitally partial, read is also appended.
     196                 :     
     197                 :     Data is appended using prepare/commit semantics.
     198                 :     The buffer grows with 1.5x factor when filled.
     199                 : 
     200                 :     @par Await-returns
     201                 : 
     202                 :     An object of type `io_result<std::size_t>` destructuring as `[ec, n]`.
     203                 : 
     204                 :     `n` represents the total number of bytes read,
     205                 :     inclusive of the last partial read.
     206                 : 
     207                 : 
     208                 :     Contingencies:
     209                 : 
     210                 :     @li The first contingency, other than one matching to @c cond::eof, reported from awaiting @c stream.read_some .
     211                 : 
     212                 :     @par Await-throws
     213                 : 
     214                 :     `std::bad_alloc` when append to `dynbuf` fails.
     215                 : 
     216                 :     @param source The source to read from. If the lifetime of `source` ends
     217                 :     before the coroutine finishes, the behavior is undefined.
     218                 : 
     219                 :     @param dynbuf The dynamic buffer to append data to. If the lifetime of the 
     220                 :     buffer sequence represented by `dynbuf` ends before the coroutine finishes, 
     221                 :     the behavior is undefined.
     222                 : 
     223                 :     @param initial_amount Initial bytes to prepare (default 2048).
     224                 : 
     225                 :     @par Remarks
     226                 :     Supports _IoAwaitable cancellation_.
     227                 : 
     228                 :     @par Example
     229                 : 
     230                 :     @code
     231                 :     capy::task<std::string> read_body(capy::ReadSource auto& source)
     232                 :     {
     233                 :         std::string body;
     234                 :         auto [ec, n] = co_await capy::read(source, capy::dynamic_buffer(body));
     235                 :         if (ec)
     236                 :             throw std::system_error(ec);
     237                 :         return body;
     238                 :     }
     239                 :     @endcode
     240                 : 
     241                 :     @see ReadSource, DynamicBufferParam
     242                 : */
     243                 : template <typename S, typename DB>
     244                 :   requires ReadSource<S> && DynamicBufferParam<DB>
     245                 : auto
     246              54 : read(
     247                 :     S& source,
     248                 :     DB&& dynbuf,
     249                 :     std::size_t initial_amount = 2048) ->
     250                 :         io_task<std::size_t>
     251                 : {
     252                 :     std::size_t amount = initial_amount;
     253                 :     std::size_t total_read = 0;
     254                 :     for(;;)
     255                 :     {
     256                 :         auto mb = dynbuf.prepare(amount);
     257                 :         auto const mb_size = buffer_size(mb);
     258                 :         auto [ec, n] = co_await source.read(mb);
     259                 :         dynbuf.commit(n);
     260                 :         total_read += n;
     261                 :         if(ec == cond::eof)
     262                 :             co_return {{}, total_read};
     263                 :         if(ec)
     264                 :             co_return {ec, total_read};
     265                 :         if(n == mb_size)
     266                 :             amount = amount / 2 + amount; // 1.5x growth
     267                 :     }
     268             108 : }
     269                 : 
     270                 : } // namespace capy
     271                 : } // namespace boost
     272                 : 
     273                 : #endif
        

Generated by: LCOV version 2.3