blob: 684896f5f3b0f416c6611c11cd562c28f5a1c5cf [file]
/*****************************************************************************\
* partitions.c - Slurm REST API partitions http operations handlers
*****************************************************************************
* 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 "src/common/xassert.h"
#include "src/common/xmalloc.h"
#include "src/common/xstring.h"
#include "src/slurmrestd/operations.h"
#include "api.h"
typedef struct {
openapi_ctxt_t *ctxt;
partition_info_msg_t *part_info_ptr;
uint16_t index;
} foreach_create_update_part_t;
static void _warn_on_ignore_values(update_part_msg_t *update_part,
uint16_t index, openapi_ctxt_t *ctxt)
{
char *warn_str = NULL;
char *pos = NULL;
if (update_part->cluster_name)
xstrcatat(warn_str, &pos, "cluster");
if (update_part->cr_type)
xstrfmtcatat(warn_str, &pos, "%s%s", (warn_str ? ", " : ""),
"select_type");
if (update_part->resume_timeout)
xstrfmtcatat(warn_str, &pos, "%s%s", (warn_str ? ", " : ""),
"timeouts/resume");
if (update_part->suspend_time)
xstrfmtcatat(warn_str, &pos, "%s%s", (warn_str ? ", " : ""),
"suspend_time");
if (update_part->suspend_timeout)
xstrfmtcatat(warn_str, &pos, "%s%s", (warn_str ? ", " : ""),
"timeouts/suspend");
if (update_part->total_cpus)
xstrfmtcatat(warn_str, &pos, "%s%s", (warn_str ? ", " : ""),
"cpus/total");
if (update_part->total_nodes)
xstrfmtcatat(warn_str, &pos, "%s%s", (warn_str ? ", " : ""),
"nodes/total");
if (update_part->tres_fmt_str)
xstrfmtcatat(warn_str, &pos, "%s%s", (warn_str ? ", " : ""),
"tres/configured");
if (warn_str)
resp_warn(
ctxt, __func__,
"The partition description at index %u contains the following ignored fields: %s",
index, warn_str);
xfree(warn_str);
}
static int _foreach_create_update_partition(void *x, void *arg)
{
update_part_msg_t *update_part = x;
foreach_create_update_part_t *args = arg;
partition_info_msg_t *part_info_ptr = args->part_info_ptr;
bool part_exists = false;
int rc = SLURM_SUCCESS;
xassert(update_part);
if (!update_part->name) {
resp_error(
args->ctxt, ESLURM_INVALID_PARTITION_NAME, __func__,
"The partition description at index %u must specify a name. Skipping",
args->index);
args->index++;
return 0; /* Don't break list_for_each */
}
_warn_on_ignore_values(update_part, args->index, args->ctxt);
/*
* RPC does not use nodesets field, append it to nodes instead unless
* it is "ALL" only set by the controller when partition is configured
* with Nodes=ALL. Nodesets should not be set to ALL by the user.
*/
if (update_part->nodesets && xstrcmp(update_part->nodesets, "ALL")) {
xstrfmtcat(update_part->nodes, "%s%s",
(update_part->nodes ? "," : ""),
update_part->nodesets);
}
if (!xstrcasecmp(update_part->name, "default")) {
part_exists = true; /* updating the default partition config */
} else if (part_info_ptr->record_count) {
partition_info_t *part_ptr = part_info_ptr->partition_array;
for (int i = 0; i < part_info_ptr->record_count; i++) {
if (xstrcmp(update_part->name, part_ptr[i].name))
continue;
part_exists = true;
break;
}
}
errno = SLURM_SUCCESS;
if (part_exists)
rc = slurm_update_partition(update_part);
else
rc = slurm_create_partition(update_part);
if (rc) {
char *func_name = part_exists ? "slurm_update_partition" :
"slurm_create_partition";
char *action_str = part_exists ? "update" : "create";
/* slurm_[update|create]_partition both set error in errno */
if ((rc == SLURM_ERROR) && errno)
rc = errno;
resp_error(
args->ctxt, rc, func_name,
"Failed to %s partition %s with description at index %u",
action_str, update_part->name, args->index);
}
args->index++;
return 0; /* Don't break list_for_each */
}
static int _partitions_mod(openapi_ctxt_t *ctxt)
{
int rc = SLURM_SUCCESS;
openapi_part_mod_req_t req = { 0 };
openapi_part_mod_req_t *req_ptr = &req; /* required for free macro */
partition_info_msg_t *part_info_ptr = NULL;
foreach_create_update_part_t args = {
.ctxt = ctxt,
};
if (DATA_PARSE(ctxt->parser, OPENAPI_PARTITIONS_MOD_REQ, req,
ctxt->query, ctxt->parent_path)) {
resp_error(ctxt, ESLURM_REST_INVALID_QUERY, __func__,
"Rejecting request. Failure parsing query");
goto done;
}
if (!req.partition_list) {
resp_error(ctxt, ESLURM_REST_INVALID_QUERY, __func__,
"A partition description must be provided.");
goto done;
}
rc = slurm_load_partitions(0, &part_info_ptr, 0);
if (rc) {
if ((rc == SLURM_ERROR) && errno)
rc = errno;
resp_error(ctxt, rc, "slurm_load_partitions()",
"Unable to query partitions");
goto done;
}
args.part_info_ptr = part_info_ptr;
list_for_each(req.partition_list, _foreach_create_update_partition,
&args);
done:
slurm_free_partition_info_msg(part_info_ptr);
FREE_OPENAPI_RESP_COMMON_CONTENTS(req_ptr);
FREE_NULL_LIST(req.partition_list);
return rc ? rc : ctxt->rc;
}
static int _partitions_get(openapi_ctxt_t *ctxt)
{
int rc = SLURM_SUCCESS;
partition_info_msg_t *part_info_ptr = NULL;
openapi_partitions_query_t query = { 0 };
openapi_resp_partitions_info_msg_t resp = { 0 };
if (DATA_PARSE(ctxt->parser, OPENAPI_PARTITIONS_QUERY, query,
ctxt->query, ctxt->parent_path)) {
resp_error(ctxt, ESLURM_REST_INVALID_QUERY, __func__,
"Rejecting request. Failure parsing query");
goto done;
}
errno = 0;
if ((rc = slurm_load_partitions(query.update_time, &part_info_ptr,
query.show_flags))) {
if ((rc == SLURM_ERROR) && errno)
rc = errno;
goto done;
}
if (part_info_ptr) {
resp.last_update = part_info_ptr->last_update;
resp.partitions = part_info_ptr;
}
DATA_DUMP(ctxt->parser, OPENAPI_PARTITION_RESP, resp, ctxt->resp);
done:
slurm_free_partition_info_msg(part_info_ptr);
return rc;
}
extern int op_handler_partitions(openapi_ctxt_t *ctxt)
{
int rc = SLURM_SUCCESS;
switch (ctxt->method) {
case HTTP_REQUEST_GET:
rc = _partitions_get(ctxt);
break;
case HTTP_REQUEST_POST:
rc = _partitions_mod(ctxt);
break;
default:
resp_error(ctxt, ESLURM_REST_INVALID_QUERY, __func__,
"Unsupported HTTP method requested: %s",
get_http_method_string(ctxt->method));
}
return rc;
}
static int _partition_delete(openapi_ctxt_t *ctxt, char *partition_name)
{
delete_part_msg_t delete_msg = {
.name = partition_name,
};
int rc = EINVAL;
errno = SLURM_SUCCESS;
if ((rc = slurm_delete_partition(&delete_msg))) {
if ((rc == SLURM_ERROR) && errno)
rc = errno;
resp_error(ctxt, rc, "slurm_delete_partition()",
"Unable to delete partition");
}
return rc;
}
static int _partition_get(openapi_ctxt_t *ctxt, char *partition_name)
{
openapi_partitions_query_t query = { 0 };
partition_info_msg_t *part_info_ptr = NULL;
int rc = SLURM_SUCCESS;
if (DATA_PARSE(ctxt->parser, OPENAPI_PARTITIONS_QUERY, query,
ctxt->query, ctxt->parent_path)) {
resp_error(ctxt, ESLURM_REST_INVALID_QUERY, __func__,
"Rejecting request. Failure parsing query");
goto done;
}
if (!query.show_flags)
query.show_flags = SHOW_ALL;
errno = 0;
if ((rc = slurm_load_partitions(query.update_time, &part_info_ptr,
query.show_flags))) {
if ((rc == SLURM_ERROR) && errno)
rc = errno;
resp_error(ctxt, rc, __func__, "Unable to query partitions");
goto done;
}
if (part_info_ptr) {
partition_info_t *part = NULL;
for (int i = 0; !rc && i < part_info_ptr->record_count; i++) {
const char *n = part_info_ptr->partition_array[i].name;
if (!xstrcasecmp(partition_name, n)) {
part = &part_info_ptr->partition_array[i];
break;
}
}
if (!part) {
resp_warn(ctxt, __func__, "Unable to find partition %s",
partition_name);
} else {
partition_info_msg_t p = {
.last_update = part_info_ptr->last_update,
.record_count = 1,
.partition_array = part,
};
openapi_resp_partitions_info_msg_t resp = {
.partitions = &p,
.last_update = part_info_ptr->last_update,
};
DATA_DUMP(ctxt->parser, OPENAPI_PARTITION_RESP, resp,
ctxt->resp);
}
}
done:
slurm_free_partition_info_msg(part_info_ptr);
return rc;
}
extern int op_handler_partition(openapi_ctxt_t *ctxt)
{
openapi_partition_param_t params = { 0 };
int rc = SLURM_SUCCESS;
if ((ctxt->method != HTTP_REQUEST_GET) &&
(ctxt->method != HTTP_REQUEST_DELETE)) {
resp_error(ctxt, ESLURM_REST_INVALID_QUERY, __func__,
"Unsupported HTTP method requested: %s",
get_http_method_string(ctxt->method));
goto done;
}
if (DATA_PARSE(ctxt->parser, OPENAPI_PARTITION_PARAM, params,
ctxt->parameters, ctxt->parent_path)) {
resp_error(ctxt, ESLURM_REST_INVALID_QUERY, __func__,
"Rejecting request. Failure parsing parameters");
goto done;
}
if (ctxt->method == HTTP_REQUEST_GET)
rc = _partition_get(ctxt, params.partition_name);
else
rc = _partition_delete(ctxt, params.partition_name);
done:
xfree(params.partition_name);
return rc;
}