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-06-21 16:31:20 Functions: 100.0 % 8 8

           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 on `stream.read_some`.
      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                 :         while `buffers` is not yet filled. A contingency that accompanies
      51                 :         the read which fills `buffers` is not reported: a completed
      52                 :         transfer is a success.
      53                 : 
      54                 :     Notable conditions:
      55                 : 
      56                 :     @li @c cond::canceled — Operation was cancelled,
      57                 :     @li @c cond::eof — Stream reached end before @c buffers was filled.
      58                 : 
      59                 :     @par Await-postcondition
      60                 :     If `n == buffer_size(buffers)` the transfer completed and `ec` is
      61                 :     success; otherwise `ec` is set.
      62                 : 
      63                 :     @param stream The stream to read from. If the lifetime of `stream` ends
      64                 :     before the coroutine finishes, the behavior is undefined.
      65                 : 
      66                 :     @param buffers The buffer sequence to fill. If the lifetime of the buffer
      67                 :     sequence represented by `buffers` ends before the coroutine finishes, the behavior is undefined.
      68                 : 
      69                 : 
      70                 :     @par Remarks
      71                 :     Supports _IoAwaitable cancellation_.
      72                 : 
      73                 : 
      74                 :     @par Example
      75                 : 
      76                 :     @code
      77                 :     capy::task<> process_message(capy::ReadStream auto& stream)
      78                 :     {
      79                 :         std::vector<char> header(16);  // known header size for some protocol
      80                 :         auto [ec, n] = co_await capy::read(stream, capy::mutable_buffer(header));
      81                 :         if (ec == capy::cond::eof)
      82                 :             co_return;  // Connection closed
      83                 :         if (ec)
      84                 :             throw std::system_error(ec);
      85                 : 
      86                 :         // at this point `header` contains exactly 16 bytes
      87                 :     }
      88                 :     @endcode
      89                 : 
      90                 :     @see ReadStream, MutableBufferSequence
      91                 : */
      92                 : template <typename S, typename MB>
      93                 :   requires ReadStream<S> && MutableBufferSequence<MB>
      94                 : auto
      95 HIT          98 : read(S& stream, MB buffers) ->
      96                 :         io_task<std::size_t>
      97                 : {
      98                 :     auto consuming = buffer_slice(buffers);
      99                 :     std::size_t const total_size = buffer_size(buffers);
     100                 :     std::size_t total_read = 0;
     101                 : 
     102                 :     while(total_read < total_size)
     103                 :     {
     104                 :         auto [ec, n] = co_await stream.read_some(consuming.data());
     105                 :         consuming.remove_prefix(n);
     106                 :         total_read += n;
     107                 :         // A contingency that still completed the transfer is a success:
     108                 :         // report it only when the buffer was not filled.
     109                 :         if(ec && total_read < total_size)
     110                 :             co_return {ec, total_read};
     111                 :     }
     112                 : 
     113                 :     co_return {{}, total_read};
     114             196 : }
     115                 : 
     116                 : /** Read all data from a stream into a dynamic buffer.
     117                 : 
     118                 :     @par Await-effects
     119                 : 
     120                 :     Reads data from `stream` via awaiting `stream.read_some` repeatedly
     121                 :     and appending it to `dynbuf` using prepare/commit semantics
     122                 :     until:
     123                 :     
     124                 :         @li either @c dynbuf.size() == @c dynbuf.max_size() ,
     125                 :         @li or a contingency on @c stream.read_some occurs.
     126                 : 
     127                 :     The last, potenitally partial, read is also appended.
     128                 :     
     129                 :     The value passed in the first call to `dynbuf.prepare` is `initial_amount`.
     130                 :     The value is grown to 1.5 times the preceding value only after a read that
     131                 :     completely filled the prepared buffer; otherwise it is left unchanged for
     132                 :     the next call.
     133                 : 
     134                 : 
     135                 :     @par Await-returns
     136                 : 
     137                 :     An object of type `io_result<std::size_t>` destructuring as `[ec, n]`.
     138                 : 
     139                 :     `n` represents the total number of bytes read,
     140                 :     inclusive of the last partial read.
     141                 : 
     142                 :     Contingencies:
     143                 : 
     144                 :     @li The first contingency, other than one matching to @c cond::eof, reported from awaiting @c stream.read_some .
     145                 : 
     146                 : 
     147                 :     @par Await-throws
     148                 :     
     149                 :     Whatever operations on @c dunbuf throw.
     150                 : 
     151                 :     (Note: types modeling @c DynamicBufferParam provided by Capy throw 
     152                 :      @c std::bad_alloc  from member function 
     153                 :      @c prepare .)
     154                 : 
     155                 : 
     156                 :     @param stream The stream to read from. If the lifetime of `stream` ends
     157                 :     before the coroutine finishes, the behavior is undefined.
     158                 : 
     159                 :     @param dynbuf The dynamic buffer to append data to. If the lifetime of the buffer
     160                 :     sequence represented by `dynbuf` ends before the coroutine finishes, the behavior is undefined.
     161                 : 
     162                 :     @param initial_amount Hint for the value to be passed in the initial call to `dynbuf.prepare()`
     163                 :     (default 2048).
     164                 : 
     165                 :     
     166                 :     @par Remarks
     167                 :     Supports _IoAwaitable cancellation_.
     168                 : 
     169                 :     @par Example
     170                 : 
     171                 :     @code
     172                 :     capy::task<std::string> read_body(capy::ReadStream auto& stream)
     173                 :     {
     174                 :         std::string body;
     175                 :         auto [ec, n] = co_await capy::read(stream, capy::dynamic_buffer(body));
     176                 :         if (ec)
     177                 :             throw std::system_error(ec);
     178                 :         return body;
     179                 :     }
     180                 :     @endcode
     181                 : 
     182                 :     @see read_some, ReadStream, DynamicBufferParam
     183                 : */
     184                 : template <typename S, typename DB>
     185                 :   requires ReadStream<S> && DynamicBufferParam<DB>
     186                 : auto
     187              80 : read(
     188                 :     S& stream,
     189                 :     DB&& dynbuf,
     190                 :     std::size_t initial_amount = 2048) ->
     191                 :         io_task<std::size_t>
     192                 : {
     193                 :     std::size_t amount = initial_amount;
     194                 :     std::size_t total_read = 0;
     195                 :     for(;;)
     196                 :     {
     197                 :         auto mb = dynbuf.prepare(amount);
     198                 :         auto const mb_size = buffer_size(mb);
     199                 :         auto [ec, n] = co_await stream.read_some(mb);
     200                 :         dynbuf.commit(n);
     201                 :         total_read += n;
     202                 :         if(ec == cond::eof)
     203                 :             co_return {{}, total_read};
     204                 :         if(ec)
     205                 :             co_return {ec, total_read};
     206                 :         if(n == mb_size)
     207                 :             amount = amount / 2 + amount;
     208                 :     }
     209             160 : }
     210                 : 
     211                 : /** Read all data from a source into a dynamic buffer.
     212                 : 
     213                 :     @par Await-effects
     214                 : 
     215                 :     Reads data from `stream` by calling `source.read` repeatedly 
     216                 :     and appending it to `dynbuf` using prepare/commit semantics
     217                 :     until:
     218                 :     
     219                 :     @li either @c dynbuf.size() == @c dynbuf.max_size() ,
     220                 :     @li or a contingency on @c stream.read occurs.
     221                 : 
     222                 :     The last, potenitally partial, read is also appended.
     223                 :     
     224                 :     The value passed in the first call to `dynbuf.prepare` is `initial_amount`.
     225                 :     The value is grown to 1.5 times the preceding value only after a read that
     226                 :     completely filled the prepared buffer; otherwise it is left unchanged for
     227                 :     the next call.
     228                 : 
     229                 : 
     230                 :     @par Await-returns
     231                 : 
     232                 :     An object of type `io_result<std::size_t>` destructuring as `[ec, n]`.
     233                 : 
     234                 :     `n` represents the total number of bytes read,
     235                 :     inclusive of the last partial read.
     236                 : 
     237                 : 
     238                 :     Contingencies:
     239                 : 
     240                 :     @li The first contingency, other than one matching to @c cond::eof, reported from awaiting @c stream.read_some .
     241                 : 
     242                 : 
     243                 :     @par Await-throws
     244                 :     
     245                 :     Whatever operations on @c dunbuf throw.
     246                 : 
     247                 :     (Note: types modeling @c DynamicBufferParam provided by Capy throw 
     248                 :      @c std::bad_alloc from member function 
     249                 :      @c prepare .)
     250                 :      
     251                 : 
     252                 :     @param source The source to read from. If the lifetime of `source` ends
     253                 :     before the coroutine finishes, the behavior is undefined.
     254                 : 
     255                 :     @param dynbuf The dynamic buffer to append data to. If the lifetime of the 
     256                 :     buffer sequence represented by `dynbuf` ends before the coroutine finishes, 
     257                 :     the behavior is undefined.
     258                 : 
     259                 :     @param initial_amount Hint for the value to be passed in the initial call to `dynbuf.prepare()`
     260                 :     (default 2048).
     261                 : 
     262                 :     @par Remarks
     263                 :     Supports _IoAwaitable cancellation_.
     264                 : 
     265                 :     @par Example
     266                 : 
     267                 :     @code
     268                 :     capy::task<std::string> read_body(capy::ReadSource auto& source)
     269                 :     {
     270                 :         std::string body;
     271                 :         auto [ec, n] = co_await capy::read(source, capy::dynamic_buffer(body));
     272                 :         if (ec)
     273                 :             throw std::system_error(ec);
     274                 :         return body;
     275                 :     }
     276                 :     @endcode
     277                 : 
     278                 :     @see ReadSource, DynamicBufferParam
     279                 : */
     280                 : template <typename S, typename DB>
     281                 :   requires ReadSource<S> && DynamicBufferParam<DB>
     282                 : auto
     283              54 : read(
     284                 :     S& source,
     285                 :     DB&& dynbuf,
     286                 :     std::size_t initial_amount = 2048) ->
     287                 :         io_task<std::size_t>
     288                 : {
     289                 :     std::size_t amount = initial_amount;
     290                 :     std::size_t total_read = 0;
     291                 :     for(;;)
     292                 :     {
     293                 :         auto mb = dynbuf.prepare(amount);
     294                 :         auto const mb_size = buffer_size(mb);
     295                 :         auto [ec, n] = co_await source.read(mb);
     296                 :         dynbuf.commit(n);
     297                 :         total_read += n;
     298                 :         if(ec == cond::eof)
     299                 :             co_return {{}, total_read};
     300                 :         if(ec)
     301                 :             co_return {ec, total_read};
     302                 :         if(n == mb_size)
     303                 :             amount = amount / 2 + amount; // 1.5x growth
     304                 :     }
     305             108 : }
     306                 : 
     307                 : } // namespace capy
     308                 : } // namespace boost
     309                 : 
     310                 : #endif
        

Generated by: LCOV version 2.3