CI: added sync test

This commit is contained in:
SChernykh 2022-08-23 14:13:09 +02:00
parent cc60ab3d63
commit bde5b19c77
7 changed files with 228 additions and 47 deletions

103
.github/workflows/test-sync.yml vendored Normal file
View file

@ -0,0 +1,103 @@
name: Sync test
on: [push, pull_request]
jobs:
sync-test-ubuntu:
runs-on: ubuntu-22.04
steps:
- name: Install dependencies
run: |
sudo add-apt-repository -y ppa:ubuntu-toolchain-r/test
sudo apt update
sudo apt install -y git build-essential cmake libuv1-dev libzmq3-dev libsodium-dev libpgm-dev libnorm-dev libgss-dev libcurl4-openssl-dev libidn2-0-dev gcc-12 g++-12
- name: Checkout repository
uses: actions/checkout@v2
with:
submodules: true
- name: Build p2pool
run: |
mkdir build
cd build
cmake .. -DDEV_TEST_SYNC=ON -DCMAKE_C_COMPILER=gcc-12 -DCMAKE_CXX_COMPILER=g++-12
make -j$(nproc)
- name: Run p2pool
timeout-minutes: 15
run: |
cd build
./p2pool --host p2pmd.xmrvsbeast.com --zmq-port 18084 --wallet 44MnN1f3Eto8DZYUWuE5XZNUtE3vcRzt2j6PzqWpPau34e6Cf4fAxt6X2MBmrm6F9YMEiMNjN6W4Shn4pLcfNAja621jwyg --no-cache --loglevel 4
- name: Archive p2pool.log
uses: actions/upload-artifact@v2
with:
name: p2pool_ubuntu.log
path: build/p2pool.log
sync-test-macos:
runs-on: macos-12
steps:
- name: Checkout repository
uses: actions/checkout@v2
with:
submodules: recursive
- name: Install dependencies
run: HOMEBREW_NO_AUTO_UPDATE=1 brew install cmake autoconf libtool automake libuv zmq libpgm curl
- name: Build p2pool
run: |
mkdir build
cd build
cmake .. -DDEV_TEST_SYNC=ON
make -j3
- name: Run p2pool
timeout-minutes: 15
run: |
cd build
./p2pool --host p2pmd.xmrvsbeast.com --zmq-port 18084 --wallet 44MnN1f3Eto8DZYUWuE5XZNUtE3vcRzt2j6PzqWpPau34e6Cf4fAxt6X2MBmrm6F9YMEiMNjN6W4Shn4pLcfNAja621jwyg --no-cache --loglevel 4
- name: Archive p2pool.log
uses: actions/upload-artifact@v2
with:
name: p2pool_macos.log
path: build/p2pool.log
sync-test-windows:
runs-on: windows-2022
steps:
- name: Checkout repository
uses: actions/checkout@v2
with:
submodules: recursive
- name: Setup cmake
uses: lukka/get-cmake@latest
- name: Build p2pool
run: |
mkdir build
cd build
cmake .. -G "Visual Studio 17 2022" -DDEV_TEST_SYNC=ON
& "C:\\Program Files\\Microsoft Visual Studio\\2022\\Enterprise\\Msbuild\\Current\\Bin\\amd64\\msbuild" /m /p:Configuration=Release p2pool.vcxproj
- name: Run p2pool
timeout-minutes: 15
run: |
cd build/Release
./p2pool.exe --host p2pmd.xmrvsbeast.com --zmq-port 18084 --wallet 44MnN1f3Eto8DZYUWuE5XZNUtE3vcRzt2j6PzqWpPau34e6Cf4fAxt6X2MBmrm6F9YMEiMNjN6W4Shn4pLcfNAja621jwyg --no-cache --loglevel 4
- name: Archive p2pool.log
uses: actions/upload-artifact@v2
with:
name: p2pool_windows.log
path: build/Release/p2pool.log

View file

@ -5,6 +5,8 @@ option(STATIC_BINARY "Build static binary" OFF)
option(STATIC_LIBS "Link libuv and libzmq statically" OFF) option(STATIC_LIBS "Link libuv and libzmq statically" OFF)
option(WITH_RANDOMX "Include the RandomX library in the build. If this is turned off, p2pool will rely on monerod for verifying RandomX hashes" ON) option(WITH_RANDOMX "Include the RandomX library in the build. If this is turned off, p2pool will rely on monerod for verifying RandomX hashes" ON)
option(DEV_TEST_SYNC "[Developer only] Sync test, stop p2pool after sync is complete" OFF)
set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake") set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake")
set_property(DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR} PROPERTY VS_STARTUP_PROJECT p2pool) set_property(DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR} PROPERTY VS_STARTUP_PROJECT p2pool)
@ -15,6 +17,10 @@ if (WITH_RANDOMX)
set(LIBS randomx) set(LIBS randomx)
endif() endif()
if (DEV_TEST_SYNC)
add_definitions(-DDEV_TEST_SYNC)
endif()
include(cmake/flags.cmake) include(cmake/flags.cmake)
set(HEADERS set(HEADERS
@ -77,10 +83,19 @@ if (WITH_RANDOMX)
set(SOURCES ${SOURCES} src/miner.cpp) set(SOURCES ${SOURCES} src/miner.cpp)
endif() endif()
if (NOT STATIC_BINARY AND NOT STATIC_LIBS)
include(FindCURL)
endif()
if (CURL_INCLUDE_DIRS)
include_directories(CURL_INCLUDE_DIRS)
else()
include_directories(external/src/curl/include)
endif()
include_directories(src) include_directories(src)
include_directories(external/src) include_directories(external/src)
include_directories(external/src/cryptonote) include_directories(external/src/cryptonote)
include_directories(external/src/curl/include)
include_directories(external/src/libuv/include) include_directories(external/src/libuv/include)
include_directories(external/src/cppzmq) include_directories(external/src/cppzmq)
include_directories(external/src/libzmq/include) include_directories(external/src/libzmq/include)
@ -115,8 +130,13 @@ elseif (CMAKE_CXX_COMPILER_ID MATCHES GNU OR CMAKE_CXX_COMPILER_ID MATCHES Clang
find_library(CURL_LIBRARY_DEBUG NAMES libcurl.a PATHS "external/src/curl/lib/.libs" NO_DEFAULT_PATH) find_library(CURL_LIBRARY_DEBUG NAMES libcurl.a PATHS "external/src/curl/lib/.libs" NO_DEFAULT_PATH)
find_library(CURL_LIBRARY NAMES libcurl.a PATHS "external/src/curl/lib/.libs" NO_DEFAULT_PATH) find_library(CURL_LIBRARY NAMES libcurl.a PATHS "external/src/curl/lib/.libs" NO_DEFAULT_PATH)
else() else()
find_library(CURL_LIBRARY_DEBUG NAMES curl) if (CURL_LIBRARIES)
find_library(CURL_LIBRARY NAMES curl) set(CURL_LIBRARY_DEBUG ${CURL_LIBRARIES})
set(CURL_LIBRARY ${CURL_LIBRARIES})
else()
find_library(CURL_LIBRARY_DEBUG NAMES curl)
find_library(CURL_LIBRARY NAMES curl)
endif()
endif() endif()
find_library(SODIUM_LIBRARY sodium) find_library(SODIUM_LIBRARY sodium)
endif() endif()

View file

@ -253,7 +253,7 @@ Alternatively, you can select "Clone a repository" within the GUI, then select "
Run the following commands to install the necessary prerequisites, clone this repo, and build P2Pool locally on your Mac: Run the following commands to install the necessary prerequisites, clone this repo, and build P2Pool locally on your Mac:
``` ```
brew update && brew install git cmake libuv zmq libpgm brew update && brew install git cmake libuv zmq libpgm curl
git clone --recursive https://github.com/SChernykh/p2pool git clone --recursive https://github.com/SChernykh/p2pool
cd p2pool cd p2pool
mkdir build && cd build mkdir build && cd build
@ -265,7 +265,7 @@ make -j$(sysctl -n hw.logicalcpu)
Run the following commands to install the necessary prerequisites, clone this repo, and build P2Pool locally on FreeBSD: Run the following commands to install the necessary prerequisites, clone this repo, and build P2Pool locally on FreeBSD:
``` ```
pkg install git cmake libuv libzmq4 pkg install git cmake libuv libzmq4 curl
git clone --recursive https://github.com/SChernykh/p2pool git clone --recursive https://github.com/SChernykh/p2pool
cd p2pool cd p2pool
mkdir build && cd build mkdir build && cd build

View file

@ -60,10 +60,7 @@ struct CurlContext
void close_handles(); void close_handles();
bool m_closing; std::vector<std::pair<curl_socket_t, uv_poll_t*>> m_pollHandles;
uv_poll_t m_pollHandle;
curl_socket_t m_socket;
CallbackBase* m_callback; CallbackBase* m_callback;
CallbackBase* m_closeCallback; CallbackBase* m_closeCallback;
@ -85,10 +82,7 @@ struct CurlContext
}; };
CurlContext::CurlContext(const std::string& address, int port, const std::string& req, const std::string& auth, CallbackBase* cb, CallbackBase* close_cb, uv_loop_t* loop) CurlContext::CurlContext(const std::string& address, int port, const std::string& req, const std::string& auth, CallbackBase* cb, CallbackBase* close_cb, uv_loop_t* loop)
: m_closing(false) : m_callback(cb)
, m_pollHandle{}
, m_socket{}
, m_callback(cb)
, m_closeCallback(close_cb) , m_closeCallback(close_cb)
, m_loop(loop) , m_loop(loop)
, m_timer{} , m_timer{}
@ -99,6 +93,8 @@ CurlContext::CurlContext(const std::string& address, int port, const std::string
, m_auth(auth) , m_auth(auth)
, m_headers(nullptr) , m_headers(nullptr)
{ {
m_pollHandles.reserve(2);
{ {
char buf[log::Stream::BUF_SIZE + 1]; char buf[log::Stream::BUF_SIZE + 1];
buf[0] = '\0'; buf[0] = '\0';
@ -213,6 +209,15 @@ CurlContext::~CurlContext()
} }
delete m_callback; delete m_callback;
if (m_response.empty()) {
if (m_error.empty()) {
m_error = "Empty response";
}
else {
m_error += " (empty response)";
}
}
(*m_closeCallback)(m_error.c_str(), m_error.length()); (*m_closeCallback)(m_error.c_str(), m_error.length());
delete m_closeCallback; delete m_closeCallback;
@ -221,42 +226,72 @@ CurlContext::~CurlContext()
int CurlContext::on_socket(CURL* /*easy*/, curl_socket_t s, int action) int CurlContext::on_socket(CURL* /*easy*/, curl_socket_t s, int action)
{ {
auto it = std::find_if(m_pollHandles.begin(), m_pollHandles.end(), [s](const auto& value) { return value.first == s; });
switch (action) { switch (action) {
case CURL_POLL_IN: case CURL_POLL_IN:
case CURL_POLL_OUT: case CURL_POLL_OUT:
case CURL_POLL_INOUT: case CURL_POLL_INOUT:
if (!m_closing && !uv_is_closing(reinterpret_cast<uv_handle_t*>(&m_pollHandle))) { {
if (!m_socket) { uv_poll_t* h = nullptr;
m_socket = s;
curl_multi_assign(m_multiHandle, s, this); if (it != m_pollHandles.end()) {
h = it->second;
} }
else if (m_socket != s) { else {
LOGERR(1, "This code can't work with multiple parallel requests. Fix the code!"); h = new uv_poll_t{};
// cppcheck-suppress nullPointer
h->data = this;
const int result = uv_poll_init_socket(m_loop, h, s);
if (result < 0) {
LOGERR(1, "uv_poll_init_socket failed: " << uv_err_name(result));
delete h;
h = nullptr;
}
else {
m_pollHandles.emplace_back(s, h);
}
} }
int events = 0; if (h) {
if (action != CURL_POLL_IN) events |= UV_WRITABLE; const CURLMcode err = curl_multi_assign(m_multiHandle, s, this);
if (action != CURL_POLL_OUT) events |= UV_READABLE; if (err != CURLM_OK) {
LOGERR(1, "curl_multi_assign(action = " << action << ") failed: " << curl_multi_strerror(err));
}
if (!m_pollHandle.data) { int events = 0;
uv_poll_init_socket(m_loop, &m_pollHandle, s); if (action != CURL_POLL_IN) events |= UV_WRITABLE;
m_pollHandle.data = this; if (action != CURL_POLL_OUT) events |= UV_READABLE;
}
const int result = uv_poll_start(&m_pollHandle, events, curl_perform); const int result = uv_poll_start(h, events, curl_perform);
if (result < 0) { if (result < 0) {
LOGERR(1, "uv_poll_start failed with error " << uv_err_name(result)); LOGERR(1, "uv_poll_start failed with error " << uv_err_name(result));
}
}
else {
LOGERR(1, "failed to start polling on socket " << static_cast<int>(s));
} }
}
else {
LOGERR(1, "Poll handle is closing, can't process socket action " << action);
} }
break; break;
case CURL_POLL_REMOVE: case CURL_POLL_REMOVE:
default: default:
curl_multi_assign(m_multiHandle, s, nullptr); {
close_handles(); if (it != m_pollHandles.end()) {
uv_poll_t* h = it->second;
m_pollHandles.erase(it);
uv_poll_stop(h);
uv_close(reinterpret_cast<uv_handle_t*>(h), [](uv_handle_t* h) { delete reinterpret_cast<uv_poll_t*>(h); });
}
const CURLMcode err = curl_multi_assign(m_multiHandle, s, nullptr);
if (err != CURLM_OK) {
LOGERR(1, "curl_multi_assign(action = " << action << ") failed: " << curl_multi_strerror(err));
}
}
break; break;
} }
@ -285,7 +320,11 @@ void CurlContext::on_timeout(uv_handle_t* req)
CurlContext* ctx = reinterpret_cast<CurlContext*>(req->data); CurlContext* ctx = reinterpret_cast<CurlContext*>(req->data);
int running_handles = 0; int running_handles = 0;
curl_multi_socket_action(ctx->m_multiHandle, CURL_SOCKET_TIMEOUT, 0, &running_handles); CURLMcode err = curl_multi_socket_action(ctx->m_multiHandle, CURL_SOCKET_TIMEOUT, 0, &running_handles);
if (err != CURLM_OK) {
LOGERR(1, "curl_multi_socket_action failed, error " << curl_multi_strerror(err));
}
ctx->check_multi_info(); ctx->check_multi_info();
if (running_handles == 0) { if (running_handles == 0) {
@ -314,9 +353,17 @@ void CurlContext::curl_perform(uv_poll_t* req, int status, int events)
CurlContext* ctx = reinterpret_cast<CurlContext*>(req->data); CurlContext* ctx = reinterpret_cast<CurlContext*>(req->data);
int running_handles; int running_handles = 0;
curl_multi_socket_action(ctx->m_multiHandle, ctx->m_socket, flags, &running_handles); auto it = std::find_if(ctx->m_pollHandles.begin(), ctx->m_pollHandles.end(), [req](const auto& value) { return value.second == req; });
if (it != ctx->m_pollHandles.end()) {
curl_multi_socket_action(ctx->m_multiHandle, it->first, flags, &running_handles);
}
ctx->check_multi_info(); ctx->check_multi_info();
if (running_handles == 0) {
ctx->close_handles();
}
} }
void CurlContext::check_multi_info() void CurlContext::check_multi_info()
@ -355,7 +402,7 @@ void CurlContext::on_close(uv_handle_t* h)
CurlContext* ctx = reinterpret_cast<CurlContext*>(h->data); CurlContext* ctx = reinterpret_cast<CurlContext*>(h->data);
h->data = nullptr; h->data = nullptr;
if (ctx->m_timer.data || ctx->m_async.data || ctx->m_pollHandle.data) { if (ctx->m_timer.data || ctx->m_async.data) {
return; return;
} }
@ -364,12 +411,11 @@ void CurlContext::on_close(uv_handle_t* h)
void CurlContext::close_handles() void CurlContext::close_handles()
{ {
m_closing = true; for (const auto& p : m_pollHandles) {
uv_poll_stop(p.second);
if (m_pollHandle.data && !uv_is_closing(reinterpret_cast<uv_handle_t*>(&m_pollHandle))) { uv_close(reinterpret_cast<uv_handle_t*>(p.second), [](uv_handle_t* h) { delete reinterpret_cast<uv_poll_t*>(h); });
uv_poll_stop(&m_pollHandle);
uv_close(reinterpret_cast<uv_handle_t*>(&m_pollHandle), on_close);
} }
m_pollHandles.clear();
if (m_async.data && !uv_is_closing(reinterpret_cast<uv_handle_t*>(&m_async))) { if (m_async.data && !uv_is_closing(reinterpret_cast<uv_handle_t*>(&m_async))) {
uv_close(reinterpret_cast<uv_handle_t*>(&m_async), on_close); uv_close(reinterpret_cast<uv_handle_t*>(&m_async), on_close);

View file

@ -823,14 +823,15 @@ bool SideChain::get_outputs_blob(PoolBlock* block, uint64_t total_reward, std::v
return true; return true;
} }
void SideChain::print_status() const void SideChain::print_status(bool obtain_sidechain_lock) const
{ {
std::vector<hash> blocks_in_window; std::vector<hash> blocks_in_window;
blocks_in_window.reserve(m_chainWindowSize * 9 / 8); blocks_in_window.reserve(m_chainWindowSize * 9 / 8);
const difficulty_type diff = difficulty(); const difficulty_type diff = difficulty();
ReadLock lock(m_sidechainLock); if (obtain_sidechain_lock) uv_rwlock_rdlock(&m_sidechainLock);
ON_SCOPE_LEAVE([this, obtain_sidechain_lock]() { if (obtain_sidechain_lock) uv_rwlock_rdunlock(&m_sidechainLock); });
uint64_t rem; uint64_t rem;
uint64_t pool_hashrate = udiv128(diff.hi, diff.lo, m_targetBlockTime, &rem); uint64_t pool_hashrate = udiv128(diff.hi, diff.lo, m_targetBlockTime, &rem);
@ -2123,6 +2124,17 @@ void SideChain::finish_precalc()
{ {
LOGERR(1, "exception in finish_precalc(): " << e.what()); LOGERR(1, "exception in finish_precalc(): " << e.what());
} }
#ifdef DEV_TEST_SYNC
if (m_pool) {
LOGINFO(0, log::LightGreen() << "[DEV] Synchronization finished successfully, stopping P2Pool now");
print_status(false);
if (m_pool->p2p_server()) {
m_pool->p2p_server()->print_status();
}
m_pool->stop();
}
#endif
} }
} // namespace p2pool } // namespace p2pool

View file

@ -58,7 +58,7 @@ public:
bool get_block_blob(const hash& id, std::vector<uint8_t>& blob) const; bool get_block_blob(const hash& id, std::vector<uint8_t>& blob) const;
bool get_outputs_blob(PoolBlock* block, uint64_t total_reward, std::vector<uint8_t>& blob, uv_loop_t* loop) const; bool get_outputs_blob(PoolBlock* block, uint64_t total_reward, std::vector<uint8_t>& blob, uv_loop_t* loop) const;
void print_status() const; void print_status(bool obtain_sidechain_lock = true) const;
double get_reward_share(const Wallet& w) const; double get_reward_share(const Wallet& w) const;
// Consensus ID can be used to spawn independent P2Pools with their own sidechains // Consensus ID can be used to spawn independent P2Pools with their own sidechains

View file

@ -68,7 +68,7 @@ private:
template<typename T> FORCEINLINE ScopeGuard<T> on_scope_leave(T&& handler) { return ScopeGuard<T>(std::move(handler)); } template<typename T> FORCEINLINE ScopeGuard<T> on_scope_leave(T&& handler) { return ScopeGuard<T>(std::move(handler)); }
#define ON_SCOPE_LEAVE(x) auto CONCAT(scope_guard_, __LINE__) = on_scope_leave(x); #define ON_SCOPE_LEAVE(...) auto CONCAT(scope_guard_, __LINE__) = on_scope_leave(__VA_ARGS__);
struct MinerCallbackHandler struct MinerCallbackHandler
{ {