per Intel policy to include file commit date using git cmd below. The policy does not apply to non-Intel (C) notices. git log --follow -C90% --format=%ad --date default <file> | tail -1 and then pull just the 4 digit year from the result. Intel copyrights were not added to files where Intel either had no contribution ot the contribution lacked substance (ie license header updates, formatting changes, etc). Contribution date used "--follow -C95%" to get the most accurate date. Note that several files in this patch didn't end the license/(c) block with a blank comment line so these were added as the vast majority of files do have this last blank line. Simply there for consistency. Signed-off-by: paul luse <paul.e.luse@intel.com> Change-Id: Id5b7ce4f658fe87132f14139ead58d6e285c04d4 Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/15192 Tested-by: SPDK CI Jenkins <sys_sgci@intel.com> Reviewed-by: Jim Harris <james.r.harris@intel.com> Reviewed-by: Ben Walker <benjamin.walker@intel.com> Community-CI: Mellanox Build Bot
268 lines
7.5 KiB
C
268 lines
7.5 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright (C) 2021 Intel Corporation.
|
|
* All rights reserved.
|
|
* Copyright (c) 2022 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
*/
|
|
|
|
#include "spdk/stdinc.h"
|
|
#include "spdk_cunit.h"
|
|
#include "spdk/jsonrpc.h"
|
|
#include "spdk_internal/mock.h"
|
|
#include "common/lib/test_env.c"
|
|
#include "spdk/log.h"
|
|
|
|
#include "rpc/rpc.c"
|
|
|
|
#include "unit/lib/json_mock.c"
|
|
|
|
static int g_rpc_err;
|
|
void fn_rpc_method_handler(struct spdk_jsonrpc_request *request,
|
|
const struct spdk_json_val *params);
|
|
|
|
DEFINE_STUB_V(spdk_jsonrpc_end_result, (struct spdk_jsonrpc_request *request,
|
|
struct spdk_json_write_ctx *w));
|
|
DEFINE_STUB(spdk_jsonrpc_begin_result, struct spdk_json_write_ctx *,
|
|
(struct spdk_jsonrpc_request *request), (void *)1);
|
|
DEFINE_STUB(spdk_json_decode_bool, int, (const struct spdk_json_val *val, void *out), 0);
|
|
DEFINE_STUB(spdk_jsonrpc_server_listen, struct spdk_jsonrpc_server *, (int domain, int protocol,
|
|
struct sockaddr *listen_addr, socklen_t addrlen, spdk_jsonrpc_handle_request_fn handle_request),
|
|
(struct spdk_jsonrpc_server *)0Xdeaddead);
|
|
DEFINE_STUB(spdk_jsonrpc_server_poll, int, (struct spdk_jsonrpc_server *server), 0);
|
|
DEFINE_STUB_V(spdk_jsonrpc_server_shutdown, (struct spdk_jsonrpc_server *server));
|
|
|
|
DECLARE_WRAPPER(open, int, (const char *pathname, int flags, mode_t mode));
|
|
DECLARE_WRAPPER(close, int, (int fd));
|
|
DECLARE_WRAPPER(flock, int, (int fd, int operation));
|
|
DEFINE_WRAPPER(open, int, (const char *pathname, int flags, mode_t mode), (pathname, flags, mode));
|
|
DEFINE_WRAPPER(close, int, (int fd), (fd));
|
|
DEFINE_WRAPPER(flock, int, (int fd, int operation), (fd, operation));
|
|
|
|
int
|
|
spdk_json_decode_object(const struct spdk_json_val *values,
|
|
const struct spdk_json_object_decoder *decoders, size_t num_decoders, void *out)
|
|
{
|
|
if (values ->type == SPDK_JSON_VAL_INVALID) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
bool
|
|
spdk_json_strequal(const struct spdk_json_val *val, const char *str)
|
|
{
|
|
size_t len;
|
|
|
|
if (val->type != SPDK_JSON_VAL_STRING && val->type != SPDK_JSON_VAL_NAME) {
|
|
return false;
|
|
}
|
|
|
|
len = strlen(str);
|
|
if (val->len != len) {
|
|
return false;
|
|
}
|
|
|
|
return memcmp(val->start, str, len) == 0;
|
|
}
|
|
|
|
void
|
|
spdk_jsonrpc_send_error_response(struct spdk_jsonrpc_request *request,
|
|
int error_code, const char *msg)
|
|
{
|
|
g_rpc_err = error_code;
|
|
}
|
|
|
|
void
|
|
spdk_jsonrpc_send_error_response_fmt(struct spdk_jsonrpc_request *request,
|
|
int error_code, const char *fmt, ...)
|
|
{
|
|
g_rpc_err = error_code;
|
|
}
|
|
|
|
void
|
|
fn_rpc_method_handler(struct spdk_jsonrpc_request *request,
|
|
const struct spdk_json_val *params)
|
|
{
|
|
g_rpc_err = 0;
|
|
}
|
|
|
|
static void
|
|
test_jsonrpc_handler(void)
|
|
{
|
|
struct spdk_jsonrpc_request *request = (struct spdk_jsonrpc_request *)0xdeadbeef;
|
|
struct spdk_json_val method = {};
|
|
struct spdk_json_val params = {};
|
|
char *str = "test";
|
|
struct spdk_rpc_method m = {
|
|
.name = "test",
|
|
};
|
|
|
|
struct spdk_rpc_method is_alias_of = {
|
|
.name = "aliastest",
|
|
.is_deprecated = false,
|
|
.deprecation_warning_printed = false,
|
|
.func = fn_rpc_method_handler,
|
|
.state_mask = SPDK_RPC_STARTUP,
|
|
};
|
|
|
|
/* Case 1: Method not found */
|
|
method.type = SPDK_JSON_VAL_INVALID;
|
|
jsonrpc_handler(request, &method, ¶ms);
|
|
CU_ASSERT(g_rpc_err == SPDK_JSONRPC_ERROR_METHOD_NOT_FOUND);
|
|
|
|
/* Case 2: Method is alias */
|
|
method.type = SPDK_JSON_VAL_STRING;
|
|
method.start = str;
|
|
method.len = 4;
|
|
m.is_alias_of = &is_alias_of;
|
|
m.is_deprecated = true;
|
|
m.deprecation_warning_printed = false;
|
|
m.state_mask = SPDK_RPC_STARTUP;
|
|
SLIST_INSERT_HEAD(&g_rpc_methods, &m, slist);
|
|
|
|
/* m->state_mask & g_rpc_state == g_rpc_state */
|
|
g_rpc_err = -1;
|
|
g_rpc_state = SPDK_RPC_STARTUP;
|
|
jsonrpc_handler(request, &method, ¶ms);
|
|
CU_ASSERT(g_rpc_err == 0);
|
|
|
|
/* g_rpc_state == SPDK_RPC_STARTUP */
|
|
is_alias_of.state_mask = SPDK_RPC_RUNTIME;
|
|
g_rpc_err = -1;
|
|
g_rpc_state = SPDK_RPC_STARTUP;
|
|
jsonrpc_handler(request, &method, ¶ms);
|
|
CU_ASSERT(g_rpc_err == SPDK_JSONRPC_ERROR_INVALID_STATE);
|
|
|
|
/* SPDK_RPC_RUNTIME is invalid for the aliastest RPC */
|
|
is_alias_of.state_mask = SPDK_RPC_STARTUP;
|
|
g_rpc_err = -1;
|
|
g_rpc_state = SPDK_RPC_RUNTIME;
|
|
jsonrpc_handler(request, &method, ¶ms);
|
|
CU_ASSERT(g_rpc_err == SPDK_JSONRPC_ERROR_INVALID_STATE);
|
|
|
|
SLIST_REMOVE_HEAD(&g_rpc_methods, slist);
|
|
}
|
|
|
|
static void
|
|
test_spdk_rpc_is_method_allowed(void)
|
|
{
|
|
const char method[] = "test";
|
|
uint32_t state_mask = SPDK_RPC_STARTUP, m_state_mask;
|
|
struct spdk_rpc_method m = {};
|
|
int rc = 0;
|
|
|
|
/* Case 1: Expect return -EPERM */
|
|
m.name = method;
|
|
m.state_mask = SPDK_RPC_RUNTIME;
|
|
SLIST_INSERT_HEAD(&g_rpc_methods, &m, slist);
|
|
rc = spdk_rpc_is_method_allowed(method, state_mask);
|
|
CU_ASSERT(rc == -EPERM);
|
|
rc = spdk_rpc_get_method_state_mask(method, &m_state_mask);
|
|
CU_ASSERT(rc == 0);
|
|
CU_ASSERT(m_state_mask == m.state_mask);
|
|
|
|
/* Case 2: Expect return 0 */
|
|
state_mask = SPDK_RPC_RUNTIME;
|
|
rc = spdk_rpc_is_method_allowed(method, state_mask);
|
|
CU_ASSERT(rc == 0);
|
|
|
|
/* Case 3: Expect return -ENOENT */
|
|
SLIST_REMOVE_HEAD(&g_rpc_methods, slist);
|
|
rc = spdk_rpc_is_method_allowed(method, state_mask);
|
|
CU_ASSERT(rc == -ENOENT);
|
|
rc = spdk_rpc_get_method_state_mask(method, &m_state_mask);
|
|
CU_ASSERT(rc == -ENOENT);
|
|
}
|
|
|
|
static void
|
|
test_rpc_get_methods(void)
|
|
{
|
|
struct spdk_jsonrpc_request *request = (struct spdk_jsonrpc_request *)0xbeefbeef;
|
|
struct spdk_json_val params = {};
|
|
struct spdk_rpc_method m = {};
|
|
|
|
/* Case 1: spdk_json_decode_object failed */
|
|
g_rpc_err = -1;
|
|
params.type = SPDK_JSON_VAL_INVALID;
|
|
rpc_get_methods(request, ¶ms);
|
|
CU_ASSERT(g_rpc_err == SPDK_JSONRPC_ERROR_INVALID_PARAMS);
|
|
|
|
/* Case 2: Expect pass */
|
|
params.type = SPDK_JSON_VAL_TRUE;
|
|
m.state_mask = SPDK_RPC_RUNTIME;
|
|
g_rpc_state = SPDK_RPC_STARTUP;
|
|
SLIST_INSERT_HEAD(&g_rpc_methods, &m, slist);
|
|
rpc_get_methods(request, ¶ms);
|
|
SLIST_REMOVE_HEAD(&g_rpc_methods, slist);
|
|
}
|
|
|
|
static void
|
|
test_rpc_spdk_get_version(void)
|
|
{
|
|
struct spdk_jsonrpc_request *request = (struct spdk_jsonrpc_request *)0xdeadbeef;
|
|
struct spdk_json_val params = {};
|
|
|
|
/* Case 1: spdk_get_version method requires no parameters */
|
|
g_rpc_err = -1;
|
|
params.type = SPDK_JSON_VAL_INVALID;
|
|
rpc_spdk_get_version(request, ¶ms);
|
|
CU_ASSERT(g_rpc_err == SPDK_JSONRPC_ERROR_INVALID_PARAMS);
|
|
|
|
/* Case 2: Expect pass */
|
|
rpc_spdk_get_version(request, NULL);
|
|
}
|
|
|
|
static void
|
|
test_spdk_rpc_listen_close(void)
|
|
{
|
|
const char listen_addr[128] = "/var/tmp/spdk-rpc-ut.sock";
|
|
char rpc_lock_path[128] = {};
|
|
|
|
MOCK_SET(open, 1);
|
|
MOCK_SET(close, 0);
|
|
MOCK_SET(flock, 0);
|
|
|
|
spdk_rpc_listen(listen_addr);
|
|
snprintf(rpc_lock_path, sizeof(g_rpc_lock_path), "%s.lock",
|
|
g_rpc_listen_addr_unix.sun_path);
|
|
|
|
CU_ASSERT(g_rpc_listen_addr_unix.sun_family == AF_UNIX);
|
|
CU_ASSERT(strcmp(g_rpc_listen_addr_unix.sun_path, listen_addr) == 0);
|
|
CU_ASSERT(strcmp(g_rpc_lock_path, rpc_lock_path) == 0);
|
|
CU_ASSERT(g_jsonrpc_server == (struct spdk_jsonrpc_server *)0Xdeaddead);
|
|
|
|
spdk_rpc_close();
|
|
|
|
CU_ASSERT(g_rpc_listen_addr_unix.sun_path[0] == '\0');
|
|
CU_ASSERT(g_jsonrpc_server == NULL);
|
|
CU_ASSERT(g_rpc_lock_fd == -1);
|
|
CU_ASSERT(g_rpc_lock_path[0] == '\0');
|
|
|
|
MOCK_CLEAR(open);
|
|
MOCK_CLEAR(close);
|
|
MOCK_CLEAR(flock);
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
CU_pSuite suite = NULL;
|
|
unsigned int num_failures;
|
|
|
|
CU_set_error_action(CUEA_ABORT);
|
|
CU_initialize_registry();
|
|
|
|
suite = CU_add_suite("rpc", NULL, NULL);
|
|
|
|
CU_ADD_TEST(suite, test_jsonrpc_handler);
|
|
CU_ADD_TEST(suite, test_spdk_rpc_is_method_allowed);
|
|
CU_ADD_TEST(suite, test_rpc_get_methods);
|
|
CU_ADD_TEST(suite, test_rpc_spdk_get_version);
|
|
CU_ADD_TEST(suite, test_spdk_rpc_listen_close);
|
|
|
|
CU_basic_set_mode(CU_BRM_VERBOSE);
|
|
CU_basic_run_tests();
|
|
num_failures = CU_get_number_of_failures();
|
|
CU_cleanup_registry();
|
|
return num_failures;
|
|
}
|