LCOV - code coverage report
Current view: top level - src/test - system_tests.cpp (source / functions) Coverage Total Hit
Test: test_bitcoin_coverage.info Lines: 97.5 % 40 39
Test Date: 2026-03-22 04:28:44 Functions: 100.0 % 5 5
Branches: 48.3 % 234 113

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2019-present The Bitcoin Core developers
       2                 :             : // Distributed under the MIT software license, see the accompanying
       3                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       4                 :             : //
       5                 :             : 
       6                 :             : #include <bitcoin-build-config.h> // IWYU pragma: keep
       7                 :             : 
       8                 :             : #include <common/run_command.h>
       9                 :             : #include <test/util/common.h>
      10                 :             : #include <test/util/setup_common.h>
      11                 :             : #include <univalue.h>
      12                 :             : #include <util/string.h>
      13                 :             : 
      14                 :             : #ifdef ENABLE_EXTERNAL_SIGNER
      15                 :             : #include <util/subprocess.h>
      16                 :             : #endif // ENABLE_EXTERNAL_SIGNER
      17                 :             : 
      18                 :             : #include <boost/cstdlib.hpp>
      19                 :             : #include <boost/test/unit_test.hpp>
      20                 :             : 
      21                 :             : #include <string>
      22                 :             : 
      23                 :             : BOOST_FIXTURE_TEST_SUITE(system_tests, BasicTestingSetup)
      24                 :             : 
      25                 :             : #ifdef ENABLE_EXTERNAL_SIGNER
      26                 :             : 
      27                 :           5 : static std::vector<std::string> mock_executable(std::string name)
      28                 :             : {
      29                 :             :     // Invoke the mock_process/* test case with all unsolicited output suppressed.
      30                 :           5 :     return {
      31         [ +  - ]:           5 :         boost::unit_test::framework::master_test_suite().argv[0],
      32                 :             :         // Disable false-positive memory leak dumps to stderr
      33                 :             :         // in debug builds when using the Windows UCRT.
      34                 :             :         "--detect_memory_leaks=0",
      35                 :             :         // Disable logging to stdout.
      36                 :             :         "--log_level=nothing",
      37                 :             :         // Disable the test report to stderr.
      38                 :             :         "--report_level=no",
      39                 :             :         "--run_test=mock_process/" + name,
      40   [ +  -  +  +  :          35 :     };
                   -  - ]
      41   [ +  -  +  -  :           5 : }
          +  -  +  -  +  
                -  -  - ]
      42                 :             : 
      43   [ +  -  +  -  :           7 : BOOST_AUTO_TEST_CASE(run_command)
          +  -  +  -  -  
          +  +  -  +  -  
          +  -  +  -  +  
          -  +  -  -  +  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  -  +  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  -  +  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  -  +  +  
                      - ]
      44                 :             : {
      45                 :           1 :     {
      46         [ +  - ]:           1 :         const UniValue result = RunCommandParseJSON({});
      47   [ +  -  +  - ]:           2 :         BOOST_CHECK(result.isNull());
      48                 :           1 :     }
      49                 :           1 :     {
      50   [ +  -  +  -  :           1 :         const UniValue result = RunCommandParseJSON(mock_executable("valid_json"));
                   +  - ]
      51   [ +  -  +  -  :           2 :         BOOST_CHECK(result.isObject());
                   +  - ]
      52         [ +  - ]:           1 :         const UniValue& success = result.find_value("success");
      53   [ +  -  +  -  :           2 :         BOOST_CHECK(!success.isNull());
                   +  - ]
      54   [ +  -  +  -  :           1 :         BOOST_CHECK_EQUAL(success.get_bool(), true);
                   +  - ]
      55                 :           1 :     }
      56                 :           1 :     {
      57                 :             :         // An invalid command is handled by cpp-subprocess
      58                 :             : #ifdef WIN32
      59                 :             :         const std::string expected{"CreateProcess failed: "};
      60                 :             : #else
      61                 :           1 :         const std::string expected{"execve failed: "};
      62                 :             : #endif
      63   [ +  -  +  -  :           3 :         BOOST_CHECK_EXCEPTION(RunCommandParseJSON({"invalid_command"}), subprocess::CalledProcessError, HasReason(expected));
          +  -  -  +  -  
          -  -  -  -  +  
          +  -  -  +  +  
                -  +  - ]
      64                 :           0 :     }
      65                 :           1 :     {
      66                 :             :         // Return non-zero exit code, no output to stderr
      67         [ +  - ]:           1 :         const std::vector<std::string> command = mock_executable("nonzeroexit_nooutput");
      68   [ +  -  +  -  :           6 :         BOOST_CHECK_EXCEPTION(RunCommandParseJSON(command), std::runtime_error, [&](const std::runtime_error& e) {
          +  +  +  -  -  
          -  -  +  +  -  
             +  -  +  - ]
      69                 :             :             const std::string what{e.what()};
      70                 :             :             BOOST_CHECK(what.find(strprintf("RunCommandParseJSON error: process(%s) returned %d: \n", util::Join(command, " "), boost::exit_test_failure)) != std::string::npos);
      71                 :             :             return true;
      72                 :             :         });
      73                 :           1 :     }
      74                 :           1 :     {
      75                 :             :         // Return non-zero exit code, with error message for stderr
      76         [ +  - ]:           1 :         const std::vector<std::string> command = mock_executable("nonzeroexit_stderroutput");
      77         [ +  - ]:           1 :         const std::string expected{"err"};
      78   [ +  -  +  -  :           8 :         BOOST_CHECK_EXCEPTION(RunCommandParseJSON(command), std::runtime_error, [&](const std::runtime_error& e) {
          +  +  +  -  +  
          -  -  +  +  -  
             +  -  +  - ]
      79                 :             :             const std::string what(e.what());
      80                 :             :             BOOST_CHECK(what.find(strprintf("RunCommandParseJSON error: process(%s) returned %s: %s", util::Join(command, " "), boost::exit_test_failure, "err")) != std::string::npos);
      81                 :             :             BOOST_CHECK(what.find(expected) != std::string::npos);
      82                 :             :             return true;
      83                 :             :         });
      84                 :           1 :     }
      85                 :           1 :     {
      86                 :             :         // Unable to parse JSON
      87   [ +  -  +  -  :           4 :         BOOST_CHECK_EXCEPTION(RunCommandParseJSON(mock_executable("invalid_json")), std::runtime_error, HasReason("Unable to parse JSON: {"));
          +  -  +  -  -  
          +  -  -  -  -  
          -  +  +  -  +  
                -  +  - ]
      88                 :             :     }
      89                 :           1 :     {
      90                 :             :         // Test stdin
      91   [ +  -  +  -  :           1 :         const UniValue result = RunCommandParseJSON(mock_executable("pass_stdin_to_stdout"), "{\"success\": true}");
                   +  - ]
      92   [ +  -  +  -  :           2 :         BOOST_CHECK(result.isObject());
                   +  - ]
      93         [ +  - ]:           1 :         const UniValue& success = result.find_value("success");
      94   [ +  -  +  -  :           2 :         BOOST_CHECK(!success.isNull());
                   +  - ]
      95   [ +  -  +  -  :           1 :         BOOST_CHECK_EQUAL(success.get_bool(), true);
                   +  - ]
      96                 :           1 :     }
      97                 :           1 : }
      98                 :             : #endif // ENABLE_EXTERNAL_SIGNER
      99                 :             : 
     100                 :             : BOOST_AUTO_TEST_SUITE_END()
        

Generated by: LCOV version 2.0-1