blob: 5a5b6ab6de24b2784aafcb3b9499b400d2b2c3a3 [file] [log] [blame]
/*****************************************************************************\
* serializer plugin interface
******************************************************************************
* Copyright (C) SchedMD LLC.
*
* This file is part of Slurm, a resource management program.
* For details, see <https://slurm.schedmd.com/>.
* Please also read the included file: DISCLAIMER.
*
* Slurm is free software; you can redistribute it and/or modify it under
* the terms of the GNU General Public License as published by the Free
* Software Foundation; either version 2 of the License, or (at your option)
* any later version.
*
* In addition, as a special exception, the copyright holders give permission
* to link the code of portions of this program with the OpenSSL library under
* certain conditions as described in each individual source file, and
* distribute linked combinations including the two. You must obey the GNU
* General Public License in all respects for all of the code used other than
* OpenSSL. If you modify file(s) with this exception, you may extend this
* exception to your version of the file(s), but you are not obligated to do
* so. If you do not wish to do so, delete this exception statement from your
* version. If you delete this exception statement from all source files in
* the program, then also delete it here.
*
* Slurm is distributed in the hope that it will be useful, but WITHOUT ANY
* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License along
* with Slurm; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
\*****************************************************************************/
#include "slurm/slurm_errno.h"
#include "src/common/list.h"
#include "src/common/log.h"
#include "src/common/read_config.h"
#include "src/common/timers.h"
#include "src/common/xassert.h"
#include "src/common/xmalloc.h"
#include "src/common/xstring.h"
#include "src/interfaces/serializer.h"
/* Define slurm-specific aliases for use by plugins, see slurm_xlator.h. */
strong_alias(serializer_g_init, slurm_serializer_g_init);
strong_alias(serialize_g_data_to_string, slurm_serialize_g_data_to_string);
strong_alias(serialize_g_string_to_data, slurm_serialize_g_string_to_data);
strong_alias(serializer_g_fini, slurm_serializer_g_fini);
strong_alias(serializer_required, slurm_serializer_required);
#define SERIALIZER_MAJOR_TYPE "serializer"
#define SERIALIZER_MIME_TYPES_SYM "mime_types"
#define PMT_MAGIC 0xaaba8031
#define MIME_ARRAY_MAGIC 0xabb00031
typedef struct {
int (*init)(serializer_flags_t flags);
void (*fini)(void);
int (*data_to_string)(char **dest, size_t *length, const data_t *src,
serializer_flags_t flags);
int (*string_to_data)(data_t **dest, const char *src, size_t length);
} funcs_t;
typedef struct {
int magic; /* MIME_ARRAY_MAGIC */
int index;
} mime_type_array_args_t;
/* Must be synchronized with funcs_t above */
static const char *syms[] = {
"serialize_p_init",
"serialize_p_fini",
"serialize_p_data_to_string",
"serialize_p_string_to_data",
};
/* serializer plugin state */
static plugins_t *plugins = NULL;
typedef struct {
int magic;
const char *mime_type; /* never free - const data from plugins */
int index; /* plugin index in g_context[] */
} plugin_mime_type_t;
/* list of all of the known mime types */
static list_t *mime_types_list = NULL;
static const char **mime_array = NULL;
static pthread_mutex_t init_mutex = PTHREAD_MUTEX_INITIALIZER;
#ifndef NDEBUG
/* Track when the plugins should no longer be changed */
static bool should_not_change = false;
#endif /* !NDEBUG */
static const struct {
char *string;
serializer_flags_t flag;
} flags[] = {
{ "compact", SER_FLAGS_COMPACT },
{ "pretty", SER_FLAGS_PRETTY },
{ "complex", SER_FLAGS_COMPLEX },
{ "no_tag", SER_FLAGS_NO_TAG },
};
static serializer_flags_t _parse_flag(const char *flag)
{
for (int i = 0; i < ARRAY_SIZE(flags); i++)
if (!xstrcasecmp(flag, flags[i].string))
return flags[i].flag;
return SER_FLAGS_NONE;
}
static int _parse_config(const char *config, serializer_flags_t *flags)
{
int rc = SLURM_SUCCESS;
char *token = NULL, *save_ptr = NULL;
char *toklist = xstrdup(config);
token = strtok_r(toklist, ",", &save_ptr);
while (token) {
serializer_flags_t flag = SER_FLAGS_NONE;
if (!token[0])
continue;
if ((flag = _parse_flag(token)) == SER_FLAGS_NONE) {
debug("%s: Unknown flag \"%s\" in \"%s\"",
__func__, token, config);
rc = EINVAL;
}
*flags |= flag;
token = strtok_r(NULL, ",", &save_ptr);
}
xfree(toklist);
return rc;
}
static int _find_serializer_full_type(void *x, void *key)
{
plugin_mime_type_t *pmt = x;
const char *mime_type = key;
xassert(pmt->magic == PMT_MAGIC);
if (!xstrcasecmp(mime_type, pmt->mime_type))
return 1;
return 0;
}
static plugin_mime_type_t *_find_serializer(const char *mime_type)
{
if (!xstrcmp("*/*", mime_type)) {
/*
* default to JSON if client will accept anything to avoid
* finding url-encoded or any other less suitable plugin first.
*/
plugin_mime_type_t *pmt = _find_serializer(MIME_TYPE_JSON);
if (pmt) {
return pmt;
} else {
/* JSON must not be loaded - try first thing we find */
return list_peek(mime_types_list);
}
} else {
const int len = strlen(mime_type);
/* check if client gave {TYPE} / * */
if ((len > 3) && (mime_type[len] == '*') &&
(mime_type[len - 1] == '*')) {
debug("%s: Partial mime_type globbing not supported",
__func__);
return NULL;
}
}
return list_find_first(mime_types_list, _find_serializer_full_type,
(void *) mime_type);
}
extern int serialize_g_data_to_string(char **dest, size_t *length,
const data_t *src, const char *mime_type,
serializer_flags_t flags)
{
DEF_TIMERS;
int rc;
const funcs_t *func_ptr;
plugin_mime_type_t *pmt = NULL;
xassert(dest && ((*dest == NULL) || (*dest[0] == '\0')));
pmt = _find_serializer(mime_type);
if (!pmt)
return ESLURM_DATA_UNKNOWN_MIME_TYPE;
xassert(pmt->magic == PMT_MAGIC);
func_ptr = plugins->functions[pmt->index];
START_TIMER;
rc = (*func_ptr->data_to_string)(dest, length, src, flags);
END_TIMER2(__func__);
/* dest must never be changed on failure */
xassert(!rc || !*dest);
return rc;
}
extern int serialize_g_string_to_data(data_t **dest, const char *src,
size_t length, const char *mime_type)
{
DEF_TIMERS;
int rc;
plugin_mime_type_t *pmt = NULL;
const funcs_t *func_ptr;
xassert(dest && (*dest == NULL));
pmt = _find_serializer(mime_type);
if (!pmt)
return ESLURM_DATA_UNKNOWN_MIME_TYPE;
xassert(pmt->magic == PMT_MAGIC);
func_ptr = plugins->functions[pmt->index];
START_TIMER;
rc = (*func_ptr->string_to_data)(dest, src, length);
END_TIMER2(__func__);
return rc;
}
extern const char *resolve_mime_type(const char *mime_type,
const char **plugin_ptr)
{
plugin_mime_type_t *pmt = _find_serializer(mime_type);
if (!pmt)
return NULL;
*plugin_ptr = plugins->types[pmt->index];
return pmt->mime_type;
}
static int _register_mime_types(list_t *mime_types_list, size_t plugin_index,
const char **mime_type)
{
while (*mime_type) {
plugin_mime_type_t *pmt = xmalloc(sizeof(*pmt));
pmt->index = plugin_index;
pmt->mime_type = *mime_type;
pmt->magic = PMT_MAGIC;
list_append(mime_types_list, pmt);
log_flag(DATA, "registered serializer plugin %s for %s",
plugins->types[plugin_index], pmt->mime_type);
mime_type++;
}
return SLURM_SUCCESS;
}
extern const char **get_mime_type_array(void)
{
#ifndef NDEBUG
slurm_mutex_lock(&init_mutex);
should_not_change = true;
xassert(mime_array);
slurm_mutex_unlock(&init_mutex);
#endif /* !NDEBUG */
return mime_array;
}
extern int serializer_g_init(void)
{
int rc = SLURM_SUCCESS;
serializer_flags_t flags = SER_FLAGS_NONE;
slurm_mutex_lock(&init_mutex);
if (plugins) {
slurm_mutex_unlock(&init_mutex);
return rc;
}
xassert(!should_not_change);
xassert(sizeof(funcs_t) == sizeof(void *) * ARRAY_SIZE(syms));
rc = load_plugins(&plugins, SERIALIZER_MAJOR_TYPE, NULL, NULL, syms,
ARRAY_SIZE(syms));
if (rc)
fatal("%s: Unable to load serializer plugins: %s",
__func__, slurm_strerror(rc));
if (!mime_types_list)
mime_types_list = list_create(xfree_ptr);
xrecalloc(mime_array, (plugins->count + 1), sizeof(*mime_array));
for (size_t i = 0; plugins && (i < plugins->count) && !rc; i++) {
const char *config = NULL;
const char **mime_types;
const funcs_t *func_ptr = plugins->functions[i];
xassert(plugins->handles[i] != PLUGIN_INVALID_HANDLE);
mime_types = plugin_get_sym(plugins->handles[i],
SERIALIZER_MIME_TYPES_SYM);
if (!mime_types)
fatal_abort("%s: unable to load %s from plugin",
__func__, SERIALIZER_MIME_TYPES_SYM);
/* First mime_type is always considered primary */
mime_array[i] = mime_types[0];
_register_mime_types(mime_types_list, i, mime_types);
if (!xstrcmp(plugins->types[i], MIME_TYPE_JSON_PLUGIN)) {
if (running_in_slurmrestd())
config = getenv("SLURMRESTD_JSON");
if (!config)
config = getenv(ENV_CONFIG_JSON);
}
if (!xstrcmp(plugins->types[i], MIME_TYPE_YAML_PLUGIN)) {
if (running_in_slurmrestd())
config = getenv("SLURMRESTD_YAML");
if (!config)
config = getenv(ENV_CONFIG_YAML);
}
if (config && config[0] && (rc = _parse_config(config, &flags)))
fatal("Unable to parse serializer \"%s\" flags: %s",
config, slurm_strerror(rc));
rc = (*func_ptr->init)(flags);
}
slurm_mutex_unlock(&init_mutex);
return rc;
}
extern void serializer_required(const char *mime_type)
{
serializer_g_init();
slurm_mutex_lock(&init_mutex);
if (!_find_serializer(mime_type))
fatal("%s: could not find plugin for %s", __func__, mime_type);
slurm_mutex_unlock(&init_mutex);
}
extern void serializer_g_fini(void)
{
#ifndef NDEBUG
/* There should not be a init() and then fini() and then init() again */
slurm_mutex_lock(&init_mutex);
should_not_change = true;
slurm_mutex_unlock(&init_mutex);
#endif /* !NDEBUG */
#ifdef MEMORY_LEAK_DEBUG
debug3("%s: cleaning up", __func__);
slurm_mutex_lock(&init_mutex);
for (size_t i = 0; plugins && (i < plugins->count); i++) {
const funcs_t *func_ptr = plugins->functions[i];
xassert(plugins->handles[i] != PLUGIN_INVALID_HANDLE);
(*func_ptr->fini)();
}
xfree(mime_array);
FREE_NULL_LIST(mime_types_list);
FREE_NULL_PLUGINS(plugins);
slurm_mutex_unlock(&init_mutex);
#endif
}