mirror of
https://github.com/falk-werner/webfuse-provider
synced 2024-10-27 20:44:10 +00:00
353 lines
11 KiB
C++
353 lines
11 KiB
C++
#include <gtest/gtest.h>
|
|
#include "webfuse_provider/impl/jsonrpc/proxy.h"
|
|
#include "webfuse_provider/impl/jsonrpc/error.h"
|
|
#include "webfuse_provider/impl/json/parser.h"
|
|
#include "webfuse_provider/status.h"
|
|
#include "webfuse_provider/impl/timer/manager.h"
|
|
|
|
#include "webfuse_provider/jsonrpc/mock_timer.hpp"
|
|
|
|
#include <libwebsockets.h>
|
|
|
|
#include <thread>
|
|
#include <chrono>
|
|
|
|
using namespace std::chrono_literals;
|
|
using wfp_jsonrpc_test::MockTimer;
|
|
using testing::Return;
|
|
using testing::_;
|
|
using testing::DoAll;
|
|
using testing::SaveArg;
|
|
|
|
#define WFP_DEFAULT_TIMEOUT (10 * 1000)
|
|
|
|
namespace
|
|
{
|
|
struct SendContext
|
|
{
|
|
std::string response;
|
|
bool is_called;
|
|
|
|
explicit SendContext()
|
|
: is_called(false)
|
|
{
|
|
}
|
|
|
|
~SendContext()
|
|
{
|
|
}
|
|
};
|
|
|
|
void jsonrpc_send(
|
|
char * request,
|
|
size_t length,
|
|
void * user_data)
|
|
{
|
|
SendContext * context = reinterpret_cast<SendContext*>(user_data);
|
|
context->is_called = true;
|
|
context->response = request;
|
|
|
|
char * raw_data = request - LWS_PRE;
|
|
free(raw_data);
|
|
}
|
|
|
|
struct FinishedContext
|
|
{
|
|
bool is_called;
|
|
wfp_json const * result;
|
|
wfp_jsonrpc_error * error;
|
|
|
|
FinishedContext()
|
|
: is_called(false)
|
|
, result(nullptr)
|
|
, error(nullptr)
|
|
{
|
|
|
|
}
|
|
|
|
~FinishedContext()
|
|
{
|
|
if (nullptr != error)
|
|
{
|
|
wfp_jsonrpc_error_dispose(error);
|
|
}
|
|
}
|
|
};
|
|
|
|
void jsonrpc_finished(
|
|
void * user_data,
|
|
wfp_json const * result,
|
|
wfp_jsonrpc_error const * error)
|
|
{
|
|
FinishedContext * context = reinterpret_cast<FinishedContext*>(user_data);
|
|
context->is_called = true;
|
|
context->result = result;
|
|
if (nullptr != error)
|
|
{
|
|
context->error = wfp_jsonrpc_error_create(error->code, error->message);
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, init)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext context;
|
|
void * user_data = reinterpret_cast<void*>(&context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, WFP_DEFAULT_TIMEOUT, &jsonrpc_send, user_data);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
|
|
ASSERT_FALSE(context.is_called);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, invoke)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, WFP_DEFAULT_TIMEOUT, &jsonrpc_send, send_data);
|
|
|
|
FinishedContext finished_context;
|
|
void * finished_data = reinterpret_cast<void*>(&finished_context);
|
|
wfp_jsonrpc_proxy_invoke(proxy, &jsonrpc_finished, finished_data, "foo", "si", "bar", 42);
|
|
|
|
ASSERT_TRUE(send_context.is_called);
|
|
ASSERT_STREQ("{\"method\":\"foo\",\"params\":[\"bar\",42],\"id\":42}", send_context.response.c_str());
|
|
|
|
ASSERT_FALSE(finished_context.is_called);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
|
|
ASSERT_TRUE(finished_context.is_called);
|
|
ASSERT_FALSE(nullptr == finished_context.error);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, invoke_fails_if_another_request_is_pending)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, WFP_DEFAULT_TIMEOUT, &jsonrpc_send, send_data);
|
|
|
|
FinishedContext finished_context;
|
|
void * finished_data = reinterpret_cast<void*>(&finished_context);
|
|
wfp_jsonrpc_proxy_invoke(proxy, &jsonrpc_finished, finished_data, "foo", "si", "bar", 42);
|
|
|
|
FinishedContext finished_context2;
|
|
void * finished_data2 = reinterpret_cast<void*>(&finished_context2);
|
|
wfp_jsonrpc_proxy_invoke(proxy, &jsonrpc_finished, finished_data2, "foo", "");
|
|
|
|
ASSERT_TRUE(send_context.is_called);
|
|
|
|
ASSERT_FALSE(finished_context.is_called);
|
|
|
|
ASSERT_TRUE(finished_context2.is_called);
|
|
ASSERT_EQ(WFP_BAD_BUSY, finished_context2.error->code);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, invoke_invalid_request)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, WFP_DEFAULT_TIMEOUT, &jsonrpc_send, send_data);
|
|
|
|
FinishedContext finished_context;
|
|
void * finished_data = reinterpret_cast<void*>(&finished_context);
|
|
wfp_jsonrpc_proxy_invoke(proxy, &jsonrpc_finished, finished_data, "foo", "?", "error");
|
|
|
|
ASSERT_TRUE(send_context.is_called);
|
|
|
|
ASSERT_FALSE(finished_context.is_called);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, on_result)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, WFP_DEFAULT_TIMEOUT, &jsonrpc_send, send_data);
|
|
|
|
FinishedContext finished_context;
|
|
void * finished_data = reinterpret_cast<void*>(&finished_context);
|
|
wfp_jsonrpc_proxy_invoke(proxy, &jsonrpc_finished, finished_data, "foo", "si", "bar", 42);
|
|
|
|
ASSERT_TRUE(send_context.is_called);
|
|
|
|
char response_text[] = "{\"result\": \"okay\", \"id\": 42}";
|
|
wfp_json_doc * doc = wfp_impl_json_parse(response_text);
|
|
|
|
wfp_jsonrpc_proxy_onresult(proxy, wfp_impl_json_root(doc));
|
|
wfp_impl_json_dispose(doc);
|
|
|
|
ASSERT_TRUE(finished_context.is_called);
|
|
ASSERT_EQ(nullptr, finished_context.error);
|
|
ASSERT_NE(nullptr, finished_context.result);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, on_result_reject_response_with_unknown_id)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, WFP_DEFAULT_TIMEOUT, &jsonrpc_send, send_data);
|
|
|
|
FinishedContext finished_context;
|
|
void * finished_data = reinterpret_cast<void*>(&finished_context);
|
|
wfp_jsonrpc_proxy_invoke(proxy, &jsonrpc_finished, finished_data, "foo", "si", "bar", 42);
|
|
|
|
ASSERT_TRUE(send_context.is_called);
|
|
|
|
char response_text[] = "{\"result\": \"okay\", \"id\": 1234}";
|
|
wfp_json_doc * doc = wfp_impl_json_parse(response_text);
|
|
|
|
wfp_jsonrpc_proxy_onresult(proxy, wfp_impl_json_root(doc));
|
|
wfp_impl_json_dispose(doc);
|
|
|
|
ASSERT_FALSE(finished_context.is_called);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, timeout)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, 0, &jsonrpc_send, send_data);
|
|
|
|
FinishedContext finished_context;
|
|
void * finished_data = reinterpret_cast<void*>(&finished_context);
|
|
wfp_jsonrpc_proxy_invoke(proxy, &jsonrpc_finished, finished_data, "foo", "si", "bar", 42);
|
|
|
|
ASSERT_TRUE(send_context.is_called);
|
|
|
|
std::this_thread::sleep_for(10ms);
|
|
wfp_timer_manager_check(timer_manager);
|
|
|
|
ASSERT_TRUE(finished_context.is_called);
|
|
ASSERT_EQ(WFP_BAD_TIMEOUT, finished_context.error->code);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, cleanup_pending_request)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, 10, &jsonrpc_send, send_data);
|
|
|
|
FinishedContext finished_context;
|
|
void * finished_data = reinterpret_cast<void*>(&finished_context);
|
|
wfp_jsonrpc_proxy_invoke(proxy, &jsonrpc_finished, finished_data, "foo", "si", "bar", 42);
|
|
|
|
ASSERT_TRUE(send_context.is_called);
|
|
|
|
ASSERT_FALSE(finished_context.is_called);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
|
|
ASSERT_TRUE(finished_context.is_called);
|
|
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
}
|
|
|
|
|
|
|
|
TEST(wfp_jsonrpc_proxy, notify)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, WFP_DEFAULT_TIMEOUT, &jsonrpc_send, send_data);
|
|
|
|
wfp_jsonrpc_proxy_notify(proxy, "foo", "si", "bar", 42);
|
|
|
|
ASSERT_TRUE(send_context.is_called);
|
|
ASSERT_STREQ("{\"method\":\"foo\",\"params\":[\"bar\",42]}", send_context.response.c_str());
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, notify_send_invalid_request)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, WFP_DEFAULT_TIMEOUT, &jsonrpc_send, send_data);
|
|
|
|
wfp_jsonrpc_proxy_notify(proxy, "foo", "?");
|
|
|
|
ASSERT_TRUE(send_context.is_called);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, swallow_timeout_if_no_request_pending)
|
|
{
|
|
MockTimer timer_api;
|
|
|
|
wfp_timer_on_timer_fn * on_timer = nullptr;
|
|
void * timer_context = nullptr;
|
|
EXPECT_CALL(timer_api, wfp_timer_create(_, _, _))
|
|
.Times(1)
|
|
.WillOnce(DoAll(SaveArg<1>(&on_timer), SaveArg<2>(&timer_context), Return(nullptr)));
|
|
EXPECT_CALL(timer_api, wfp_timer_dispose(_)).Times(1);
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(nullptr, 1, &jsonrpc_send, send_data);
|
|
|
|
on_timer(nullptr, timer_context);
|
|
ASSERT_FALSE(send_context.is_called);
|
|
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
}
|
|
|
|
TEST(wfp_jsonrpc_proxy, on_result_swallow_if_no_request_pending)
|
|
{
|
|
struct wfp_timer_manager * timer_manager = wfp_timer_manager_create();
|
|
|
|
SendContext send_context;
|
|
void * send_data = reinterpret_cast<void*>(&send_context);
|
|
struct wfp_jsonrpc_proxy * proxy = wfp_jsonrpc_proxy_create(timer_manager, WFP_DEFAULT_TIMEOUT, &jsonrpc_send, send_data);
|
|
|
|
char response_text[] = "{\"result\": \"okay\", \"id\": 42}";
|
|
wfp_json_doc * doc = wfp_impl_json_parse(response_text);
|
|
|
|
wfp_jsonrpc_proxy_onresult(proxy, wfp_impl_json_root(doc));
|
|
wfp_impl_json_dispose(doc);
|
|
|
|
wfp_jsonrpc_proxy_dispose(proxy);
|
|
wfp_timer_manager_dispose(timer_manager);
|
|
}
|
|
|