blob: 296926186b691a955e5ab3d58f7a6babfd042e99 [file] [edit]
/*****************************************************************************\
* http_request.c - HTTP on request handler
*****************************************************************************
* 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/http_request.h"
#include "src/common/http.h"
#include "src/common/http_con.h"
#include "src/common/http_mime.h"
#include "src/common/http_router.h"
#include "src/common/log.h"
#include "src/common/openapi.h"
#include "src/common/read_config.h"
#include "src/common/serdes.h"
#include "src/common/uid.h"
#include "src/common/xassert.h"
#include "src/common/xmalloc.h"
#include "src/common/xstring.h"
#include "src/interfaces/data_parser.h"
#include "src/interfaces/http_auth.h"
#define BOUND_REQUEST_MAGIC 0xab008800
typedef struct {
int magic; /* BOUND_REQUEST_MAGIC */
data_parser_t *parser;
char *path;
http_request_on_request_t on_request;
http_request_on_error_t on_error;
data_parser_type_t reply_type;
void *arg;
} bound_request_t;
#define HTTP_REQUEST_EVENT_MAGIC 0xbf0f8830
typedef struct http_request_event_s {
int magic; /* HTTP_REQUEST_EVENT_MAGIC */
bound_request_t *breq;
http_con_t *hcon;
const char *name;
const char *read_mime;
const char *write_mime;
const http_con_request_t *request;
} http_request_event_t;
static int _on_request(http_con_t *hcon, const char *name,
const http_con_request_t *request, void *arg,
void *path_arg)
{
bound_request_t *breq = path_arg;
http_request_event_t event = {
.magic = HTTP_REQUEST_EVENT_MAGIC,
.breq = breq,
.hcon = hcon,
.name = name,
.request = request,
};
int rc = EINVAL;
uid_t uid = SLURM_AUTH_NOBODY;
xassert(breq->magic == BOUND_REQUEST_MAGIC);
if ((rc = http_auth_g_authenticate(HTTP_AUTH_PLUGIN_ANY, &uid, hcon,
name, request))) {
error("%s: [%s] Rejecting HTTP authentication: %s",
__func__, name, slurm_strerror(rc));
return breq->on_error(&event, hcon, name, request, rc);
}
if ((rc = http_resolve_mime_types(name, request, &event.read_mime,
&event.write_mime))) {
error("%s: [%s] Rejecting HTTP mime headers: %s",
__func__, name, slurm_strerror(rc));
return breq->on_error(&event, hcon, name, request, rc);
}
if (slurm_conf.debug_flags & DEBUG_FLAG_NET) {
char *uid_str = uid_to_string_or_null(uid);
log_flag(NET, "%s: [%s] Accepted HTTP request: method=%s path=%s uid[%d]:%s read_mime:%s write_mime:%s",
__func__, name,
get_http_method_string(request->method),
request->url.path, uid, uid_str, event.read_mime,
event.write_mime);
xfree(uid_str);
}
return breq->on_request(&event, hcon, name, uid, request->method,
request, breq->arg);
}
static void _breq_free(http_router_on_request_event_t on_request,
void *path_arg)
{
bound_request_t *breq = path_arg;
xassert(breq->magic == BOUND_REQUEST_MAGIC);
xassert(on_request == _on_request);
breq->magic = ~BOUND_REQUEST_MAGIC;
xfree(breq->path);
xfree(breq);
}
static void _bind(data_parser_t *parser, http_request_method_t method,
const char *path, http_request_on_request_t on_request,
http_request_on_error_t on_error,
data_parser_type_t reply_type, void *arg)
{
bound_request_t *breq = xmalloc(sizeof(*breq));
*breq = (bound_request_t) {
.magic = BOUND_REQUEST_MAGIC,
.on_request = on_request,
.on_error = on_error,
.reply_type = reply_type,
.arg = arg,
.path = xstrdup(path),
.parser = parser,
};
if (parser)
xstrsubstitute(breq->path, OPENAPI_DATA_PARSER_PARAM,
data_parser_get_plugin_version(parser));
http_router_bind(method, breq->path, _on_request, _breq_free, breq);
}
extern void http_request_bind(data_parser_t **parsers,
http_request_method_t method, const char *path,
http_request_on_request_t on_request,
http_request_on_error_t on_error,
data_parser_type_t reply_type, void *arg)
{
if (reply_type == DATA_PARSER_TYPE_INVALID) {
_bind(NULL, method, path, on_request, on_error, reply_type,
arg);
return;
}
xassert(reply_type > DATA_PARSER_TYPE_INVALID);
xassert(reply_type < DATA_PARSER_TYPE_MAX);
for (int i = 0; parsers[i]; i++)
if (data_parser_g_resolve_type_string(parsers[i], reply_type))
_bind(parsers[i], method, path, on_request, on_error,
reply_type, arg);
}
extern int http_request_reply(http_request_event_t *event, int rc,
list_t *headers, bool close_header, void *reply,
size_t reply_bytes)
{
bound_request_t *breq = event->breq;
data_parser_t *parser = breq->parser;
http_con_t *hcon = event->hcon;
const char *name = event->name;
const http_con_request_t *request = event->request;
const char *write_mime = event->write_mime;
http_status_code_t status = http_status_from_error(rc);
xassert(event->magic == HTTP_REQUEST_EVENT_MAGIC);
xassert(event->breq->magic == BOUND_REQUEST_MAGIC);
if (reply && parser) {
buf_t *buffer = NULL;
xassert(breq->reply_type > DATA_PARSER_TYPE_INVALID);
xassert(breq->reply_type < DATA_PARSER_TYPE_MAX);
if (!(buffer = try_init_buf(BUF_SIZE)))
return breq->on_error(event, hcon, name, request,
ENOMEM);
if ((rc = serdes_dump_buf(parser, breq->reply_type, reply,
reply_bytes, buffer, write_mime,
SER_FLAGS_NONE)))
rc = breq->on_error(event, hcon, name, request, rc);
else
rc = http_con_send_response(hcon, status, headers,
close_header, buffer,
write_mime);
FREE_NULL_BUFFER(buffer);
} else if (reply && !parser) {
const buf_t body = SHADOW_BUF_INITIALIZER(reply, reply_bytes);
xassert(breq->reply_type == DATA_PARSER_TYPE_INVALID);
rc = http_con_send_response(hcon, status, headers, close_header,
&body, MIME_TYPE_TEXT);
} else {
rc = http_con_send_response(hcon, status, headers, close_header,
NULL, MIME_TYPE_TEXT);
}
return rc;
}