100.00% Lines (2/2) 100.00% Functions (1/1)
TLA Baseline Branch
Line Hits Code Line Hits Code
1   // 1   //
2   // Copyright (c) 2025 Vinnie Falco (vinnie.falco@gmail.com) 2   // Copyright (c) 2025 Vinnie Falco (vinnie.falco@gmail.com)
3   // 3   //
4   // Distributed under the Boost Software License, Version 1.0. (See accompanying 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) 5   // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6   // 6   //
7   // Official repository: https://github.com/cppalliance/capy 7   // Official repository: https://github.com/cppalliance/capy
8   // 8   //
9   9  
10   #ifndef BOOST_CAPY_WRITE_HPP 10   #ifndef BOOST_CAPY_WRITE_HPP
11   #define BOOST_CAPY_WRITE_HPP 11   #define BOOST_CAPY_WRITE_HPP
12   12  
13   #include <boost/capy/detail/config.hpp> 13   #include <boost/capy/detail/config.hpp>
14   #include <boost/capy/io_task.hpp> 14   #include <boost/capy/io_task.hpp>
15   #include <boost/capy/buffers.hpp> 15   #include <boost/capy/buffers.hpp>
16   #include <boost/capy/buffers/buffer_slice.hpp> 16   #include <boost/capy/buffers/buffer_slice.hpp>
17   #include <boost/capy/concept/write_stream.hpp> 17   #include <boost/capy/concept/write_stream.hpp>
18   #include <system_error> 18   #include <system_error>
19   19  
20   #include <cstddef> 20   #include <cstddef>
21   21  
22   namespace boost { 22   namespace boost {
23   namespace capy { 23   namespace capy {
24   24  
25   /** Write an entire buffer sequence to a stream. 25   /** Write an entire buffer sequence to a stream.
26   26  
27   @par Await-effects 27   @par Await-effects
28   28  
29 - Writes the contents of `buffers` to `stream` via awaiting on 29 + Writes the contents of `buffers` to `stream` via awaiting
30   `stream.write_some` with consecutive portions of data from `buffers` 30   `stream.write_some` with consecutive portions of data from `buffers`
31   until: 31   until:
32   32  
33   @li either the full content of @c buffers is processed, 33   @li either the full content of @c buffers is processed,
34   @li or a contingency occurs. 34   @li or a contingency occurs.
35   35  
36 - If `buffer_size(buffers) == 0` then no awaiting on `stream.write_some` 36 + If `buffer_size(buffers) == 0` then no awaiting `stream.write_some`
37   is performed. This is not a contingency. 37   is performed. This is not a contingency.
38   38  
39   39  
40   @par Await-returns 40   @par Await-returns
41   41  
42   An object of type `io_result<std::size_t>` destructuring as `[ec, n]`. 42   An object of type `io_result<std::size_t>` destructuring as `[ec, n]`.
43   43  
44   Upon a contingency, `n` represents the number of bytes written 44   Upon a contingency, `n` represents the number of bytes written
45   so far. 45   so far.
46   46  
47   Otherwise `n` represents the number of bytes written. 47   Otherwise `n` represents the number of bytes written.
48   48  
49   Contingencies: 49   Contingencies:
50   50  
51   @li The first contingency reported from 51   @li The first contingency reported from
52 - awaiting on @c stream.write_some . 52 + awaiting @c stream.write_some .
53   53  
54   Notable conditions: 54   Notable conditions:
55   55  
56   @li @c cond::canceled — Operation was cancelled, 56   @li @c cond::canceled — Operation was cancelled,
57   @li @c std::errc::broken_pipe — Peer closed connection. 57   @li @c std::errc::broken_pipe — Peer closed connection.
58   58  
59   59  
60   @par Await-postcondition 60   @par Await-postcondition
61   61  
62   `ec || n == buffer_size(buffers)`. 62   `ec || n == buffer_size(buffers)`.
63   63  
64 - @par Cancellation  
65 - Supports cancellation via `stop_token` propagated through the  
66 - `IoAwaitable` protocol. When cancelled, returns with `cond::canceled`.  
67 -  
68   64  
69   @param stream The stream to write to. If the lifetime of `stream` ends 65   @param stream The stream to write to. If the lifetime of `stream` ends
70   before the coroutine finishes, the behavior is undefined. 66   before the coroutine finishes, the behavior is undefined.
71   67  
72   @param buffers The buffer sequence to write. If the lifetime of the buffer 68   @param buffers The buffer sequence to write. If the lifetime of the buffer
73   sequence represented by `buffers` ends 69   sequence represented by `buffers` ends
74   before the coroutine finishes, the behavior is undefined. 70   before the coroutine finishes, the behavior is undefined.
75   71  
  72 + @par Remarks
  73 +
  74 + Supports _IoAwaitable cancellation_.
76   75  
77   @par Example 76   @par Example
78   77  
79   @code 78   @code
80   capy::task<> send_response(capy::WriteStream auto& stream, std::string_view body) 79   capy::task<> send_response(capy::WriteStream auto& stream, std::string_view body)
81   { 80   {
82   auto [ec, n] = co_await capy::write(stream, capy::make_buffer(body)); 81   auto [ec, n] = co_await capy::write(stream, capy::make_buffer(body));
83   if (ec) 82   if (ec)
84   throw std::system_error(ec); 83   throw std::system_error(ec);
85   84  
86   // All bytes written successfully 85   // All bytes written successfully
87   } 86   }
88 -  
89 - @return  
90   @endcode 87   @endcode
91   88  
92   @see WriteStream, ConstBufferSequence, IoAwaitable, io_result, cond. 89   @see WriteStream, ConstBufferSequence, IoAwaitable, io_result, cond.
93   */ 90   */
94   template <WriteStream S, ConstBufferSequence CB> 91   template <WriteStream S, ConstBufferSequence CB>
HITCBC 95   50 auto write(S& stream, CB buffers) -> io_task<std::size_t> 92   50 auto write(S& stream, CB buffers) -> io_task<std::size_t>
96   { 93   {
97   auto consuming = buffer_slice(buffers); 94   auto consuming = buffer_slice(buffers);
98   std::size_t const total_size = buffer_size(buffers); 95   std::size_t const total_size = buffer_size(buffers);
99   std::size_t total_written = 0; 96   std::size_t total_written = 0;
100   97  
101   while(total_written < total_size) 98   while(total_written < total_size)
102   { 99   {
103   auto [ec, n] = co_await stream.write_some(consuming.data()); 100   auto [ec, n] = co_await stream.write_some(consuming.data());
104   consuming.remove_prefix(n); 101   consuming.remove_prefix(n);
105   total_written += n; 102   total_written += n;
106   if(ec) 103   if(ec)
107   co_return {ec, total_written}; 104   co_return {ec, total_written};
108   } 105   }
109   106  
110   co_return {{}, total_written}; 107   co_return {{}, total_written};
HITCBC 111   100 } 108   100 }
112   109  
113   } // namespace capy 110   } // namespace capy
114   } // namespace boost 111   } // namespace boost
115   112  
116   #endif 113   #endif