lokinet/test/test_libabyss.cpp

194 lines
3.9 KiB
C++

#include <libabyss.hpp>
#include <crypto/crypto.hpp>
#include <crypto/crypto_libsodium.hpp>
#include <ev/ev.h>
#include <net/net.hpp>
#include <util/threading.hpp>
#include <gtest/gtest.h>
struct AbyssTestBase : public ::testing::Test
{
llarp::sodium::CryptoLibSodium crypto;
llarp_threadpool* threadpool = nullptr;
llarp_ev_loop_ptr loop = nullptr;
std::unique_ptr< llarp::Logic > logic;
abyss::httpd::BaseReqHandler* server = nullptr;
abyss::http::JSONRPC* client = nullptr;
const std::string method = "test.method";
bool called = false;
AbyssTestBase()
{
llarp::SetLogLevel(llarp::eLogDebug);
}
void
AssertMethod(const std::string& meth) const
{
ASSERT_EQ(meth, method);
}
static void
CancelIt(void* u, ABSL_ATTRIBUTE_UNUSED uint64_t orig, uint64_t left)
{
if(left)
return;
static_cast< AbyssTestBase* >(u)->Stop();
}
static void
StopIt(void* u)
{
static_cast< AbyssTestBase* >(u)->Stop();
}
void
Start()
{
threadpool = llarp_init_same_process_threadpool();
loop = llarp_make_ev_loop();
logic = std::make_unique< llarp::Logic >(threadpool);
sockaddr_in addr;
addr.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
addr.sin_port = htons((llarp::randint() % 2000) + 2000);
addr.sin_family = AF_INET;
llarp::Addr a(addr);
while(true)
{
if(server->ServeAsync(loop, logic.get(), a))
{
client->RunAsync(loop, a.ToString());
logic->call_later({1000, this, &CancelIt});
return;
}
std::this_thread::sleep_for(std::chrono::seconds(1));
}
}
void
Stop()
{
llarp::LogDebug("test case Stop() called");
llarp_ev_loop_stop(loop.get());
}
void
AsyncStop()
{
logic->queue_job({this, &StopIt});
}
~AbyssTestBase()
{
logic.reset();
llarp_free_threadpool(&threadpool);
llarp::SetLogLevel(llarp::eLogInfo);
}
};
struct ClientHandler : public abyss::http::IRPCClientHandler
{
AbyssTestBase* test;
ClientHandler(abyss::http::ConnImpl* impl, AbyssTestBase* parent)
: abyss::http::IRPCClientHandler(impl), test(parent)
{
}
void
HandleError()
{
FAIL() << "unexpected error";
}
void
PopulateReqHeaders(ABSL_ATTRIBUTE_UNUSED abyss::http::Headers_t& hdr)
{
}
bool
HandleResponse(ABSL_ATTRIBUTE_UNUSED abyss::http::RPC_Response response)
{
test->AsyncStop();
return true;
}
};
struct ServerHandler : public abyss::httpd::IRPCHandler
{
AbyssTestBase* test;
ServerHandler(abyss::httpd::ConnImpl* impl, AbyssTestBase* parent)
: abyss::httpd::IRPCHandler(impl), test(parent)
{
}
absl::optional< Response >
HandleJSONRPC(Method_t method, ABSL_ATTRIBUTE_UNUSED const Params& params)
{
test->AssertMethod(method);
test->called = true;
return Response();
}
~ServerHandler()
{
}
};
struct AbyssTest : public AbyssTestBase,
public abyss::http::JSONRPC,
public abyss::httpd::BaseReqHandler
{
AbyssTest()
: AbyssTestBase()
, abyss::http::JSONRPC()
, abyss::httpd::BaseReqHandler(1000)
{
client = this;
server = this;
}
abyss::http::IRPCClientHandler*
NewConn(abyss::http::ConnImpl* impl)
{
return new ClientHandler(impl, this);
}
abyss::httpd::IRPCHandler*
CreateHandler(abyss::httpd::ConnImpl* impl)
{
return new ServerHandler(impl, this);
}
static void
FlushIt(void* u)
{
static_cast< AbyssTest* >(u)->Flush();
}
void
AsyncFlush()
{
logic->queue_job({this, &FlushIt});
}
void
RunLoop()
{
llarp_ev_loop_run_single_process(loop, threadpool, logic.get());
}
};
TEST_F(AbyssTest, TestClientAndServer)
{
Start();
QueueRPC(method, nlohmann::json::object(),
std::bind(&AbyssTest::NewConn, this, std::placeholders::_1));
AsyncFlush();
RunLoop();
ASSERT_TRUE(called);
}