LCOV - code coverage report
Current view: top level - corosio/test - socket_pair.hpp (source / functions) Coverage Total Hit Missed
Test: coverage_remapped.info Lines: 75.7 % 37 28 9
Test Date: 2026-03-17 15:52:49 Functions: 100.0 % 12 12

           TLA  Line data    Source code
       1                 : //
       2                 : // Copyright (c) 2025 Vinnie Falco (vinnie.falco@gmail.com)
       3                 : // Copyright (c) 2026 Steve Gerbino
       4                 : //
       5                 : // Distributed under the Boost Software License, Version 1.0. (See accompanying
       6                 : // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
       7                 : //
       8                 : // Official repository: https://github.com/cppalliance/corosio
       9                 : //
      10                 : 
      11                 : #ifndef BOOST_COROSIO_TEST_SOCKET_PAIR_HPP
      12                 : #define BOOST_COROSIO_TEST_SOCKET_PAIR_HPP
      13                 : 
      14                 : #include <boost/corosio/io_context.hpp>
      15                 : #include <boost/corosio/tcp_acceptor.hpp>
      16                 : #include <boost/corosio/tcp_socket.hpp>
      17                 : #include <boost/corosio/socket_option.hpp>
      18                 : #include <boost/capy/ex/run_async.hpp>
      19                 : #include <boost/capy/task.hpp>
      20                 : 
      21                 : #include <cstdio>
      22                 : #include <stdexcept>
      23                 : #include <system_error>
      24                 : #include <utility>
      25                 : 
      26                 : namespace boost::corosio::test {
      27                 : 
      28                 : /** Create a connected pair of sockets.
      29                 : 
      30                 :     Creates two sockets connected via loopback TCP sockets.
      31                 :     Data written to one socket can be read from the other.
      32                 : 
      33                 :     @tparam Socket The socket type (default `tcp_socket`).
      34                 :     @tparam Acceptor The acceptor type (default `tcp_acceptor`).
      35                 : 
      36                 :     @param ctx The I/O context for the sockets.
      37                 : 
      38                 :     @return A pair of connected sockets.
      39                 : */
      40                 : template<
      41                 :     class Socket   = tcp_socket,
      42                 :     class Acceptor = tcp_acceptor,
      43                 :     bool Linger    = true>
      44                 : std::pair<Socket, Socket>
      45 HIT          58 : make_socket_pair(io_context& ctx)
      46                 : {
      47              58 :     auto ex = ctx.get_executor();
      48                 : 
      49              58 :     std::error_code accept_ec;
      50              58 :     std::error_code connect_ec;
      51              58 :     bool accept_done  = false;
      52              58 :     bool connect_done = false;
      53                 : 
      54              58 :     Acceptor acc(ctx);
      55              58 :     acc.open();
      56              58 :     acc.set_option(socket_option::reuse_address(true));
      57              58 :     if (auto ec = acc.bind(endpoint(ipv4_address::loopback(), 0)))
      58 MIS           0 :         throw std::runtime_error("socket_pair bind failed: " + ec.message());
      59 HIT          58 :     if (auto ec = acc.listen())
      60 MIS           0 :         throw std::runtime_error("socket_pair listen failed: " + ec.message());
      61 HIT          58 :     auto port = acc.local_endpoint().port();
      62                 : 
      63              58 :     Socket s1(ctx);
      64              58 :     Socket s2(ctx);
      65              58 :     s2.open();
      66                 : 
      67              58 :     capy::run_async(ex)(
      68             116 :         [](Acceptor& a, Socket& s, std::error_code& ec_out,
      69                 :            bool& done_out) -> capy::task<> {
      70                 :             auto [ec] = co_await a.accept(s);
      71                 :             ec_out    = ec;
      72                 :             done_out  = true;
      73                 :         }(acc, s1, accept_ec, accept_done));
      74                 : 
      75              58 :     capy::run_async(ex)(
      76             116 :         [](Socket& s, endpoint ep, std::error_code& ec_out,
      77                 :            bool& done_out) -> capy::task<> {
      78                 :             auto [ec] = co_await s.connect(ep);
      79                 :             ec_out    = ec;
      80                 :             done_out  = true;
      81                 :         }(s2, endpoint(ipv4_address::loopback(), port), connect_ec,
      82                 :                            connect_done));
      83                 : 
      84              58 :     ctx.run();
      85              58 :     ctx.restart();
      86                 : 
      87              58 :     if (!accept_done || accept_ec)
      88                 :     {
      89 MIS           0 :         std::fprintf(
      90                 :             stderr, "socket_pair: accept failed (done=%d, ec=%s)\n",
      91                 :             accept_done, accept_ec.message().c_str());
      92               0 :         acc.close();
      93               0 :         throw std::runtime_error("socket_pair accept failed");
      94                 :     }
      95                 : 
      96 HIT          58 :     if (!connect_done || connect_ec)
      97                 :     {
      98 MIS           0 :         std::fprintf(
      99                 :             stderr, "socket_pair: connect failed (done=%d, ec=%s)\n",
     100                 :             connect_done, connect_ec.message().c_str());
     101               0 :         acc.close();
     102               0 :         s1.close();
     103               0 :         throw std::runtime_error("socket_pair connect failed");
     104                 :     }
     105                 : 
     106 HIT          58 :     acc.close();
     107                 : 
     108                 :     if constexpr (Linger)
     109                 :     {
     110               4 :         s1.set_option(socket_option::linger(true, 0));
     111               4 :         s2.set_option(socket_option::linger(true, 0));
     112                 :     }
     113                 : 
     114             116 :     return {std::move(s1), std::move(s2)};
     115              58 : }
     116                 : 
     117                 : } // namespace boost::corosio::test
     118                 : 
     119                 : #endif
        

Generated by: LCOV version 2.3