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-02-25 01:27:20 Functions: 100.0 % 9 9

           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<class Socket = tcp_socket, class Acceptor = tcp_acceptor>
      41                 : std::pair<Socket, Socket>
      42 HIT           4 : make_socket_pair(io_context& ctx)
      43                 : {
      44               4 :     auto ex = ctx.get_executor();
      45                 : 
      46               4 :     std::error_code accept_ec;
      47               4 :     std::error_code connect_ec;
      48               4 :     bool accept_done  = false;
      49               4 :     bool connect_done = false;
      50                 : 
      51               4 :     Acceptor acc(ctx);
      52               4 :     acc.open();
      53               4 :     acc.set_option(socket_option::reuse_address(true));
      54               4 :     if (auto ec = acc.bind(endpoint(ipv4_address::loopback(), 0)))
      55 MIS           0 :         throw std::runtime_error("socket_pair bind failed: " + ec.message());
      56 HIT           4 :     if (auto ec = acc.listen())
      57 MIS           0 :         throw std::runtime_error("socket_pair listen failed: " + ec.message());
      58 HIT           4 :     auto port = acc.local_endpoint().port();
      59                 : 
      60               4 :     Socket s1(ctx);
      61               4 :     Socket s2(ctx);
      62               4 :     s2.open();
      63                 : 
      64               4 :     capy::run_async(ex)(
      65               8 :         [](Acceptor& a, Socket& s, std::error_code& ec_out,
      66                 :            bool& done_out) -> capy::task<> {
      67                 :             auto [ec] = co_await a.accept(s);
      68                 :             ec_out    = ec;
      69                 :             done_out  = true;
      70                 :         }(acc, s1, accept_ec, accept_done));
      71                 : 
      72               4 :     capy::run_async(ex)(
      73               8 :         [](Socket& s, endpoint ep, std::error_code& ec_out,
      74                 :            bool& done_out) -> capy::task<> {
      75                 :             auto [ec] = co_await s.connect(ep);
      76                 :             ec_out    = ec;
      77                 :             done_out  = true;
      78                 :         }(s2, endpoint(ipv4_address::loopback(), port), connect_ec,
      79                 :                            connect_done));
      80                 : 
      81               4 :     ctx.run();
      82               4 :     ctx.restart();
      83                 : 
      84               4 :     if (!accept_done || accept_ec)
      85                 :     {
      86 MIS           0 :         std::fprintf(
      87                 :             stderr, "socket_pair: accept failed (done=%d, ec=%s)\n",
      88                 :             accept_done, accept_ec.message().c_str());
      89               0 :         acc.close();
      90               0 :         throw std::runtime_error("socket_pair accept failed");
      91                 :     }
      92                 : 
      93 HIT           4 :     if (!connect_done || connect_ec)
      94                 :     {
      95 MIS           0 :         std::fprintf(
      96                 :             stderr, "socket_pair: connect failed (done=%d, ec=%s)\n",
      97                 :             connect_done, connect_ec.message().c_str());
      98               0 :         acc.close();
      99               0 :         s1.close();
     100               0 :         throw std::runtime_error("socket_pair connect failed");
     101                 :     }
     102                 : 
     103 HIT           4 :     acc.close();
     104                 : 
     105               4 :     s1.set_option(socket_option::linger(true, 0));
     106               4 :     s2.set_option(socket_option::linger(true, 0));
     107                 : 
     108               8 :     return {std::move(s1), std::move(s2)};
     109               4 : }
     110                 : 
     111                 : } // namespace boost::corosio::test
     112                 : 
     113                 : #endif
        

Generated by: LCOV version 2.3