Files
cpp-project-template/third_party/prometheus/3rdparty/civetweb/examples/rest/rest.c
tqcq c68893bace
Some checks failed
sm-rpc / build (Debug, aarch64-linux-gnu) (push) Failing after 29s
sm-rpc / build (Debug, arm-linux-gnueabihf) (push) Failing after 16s
sm-rpc / build (Debug, host.gcc) (push) Failing after 11s
sm-rpc / build (Debug, mipsel-linux-gnu) (push) Failing after 12s
sm-rpc / build (Release, aarch64-linux-gnu) (push) Failing after 11s
sm-rpc / build (Release, arm-linux-gnueabihf) (push) Failing after 11s
sm-rpc / build (Release, host.gcc) (push) Failing after 12s
sm-rpc / build (Release, mipsel-linux-gnu) (push) Failing after 16s
feat add spdlog
2025-08-22 18:22:57 +08:00

329 lines
7.3 KiB
C

/*
* Copyright (c) 2018 the CivetWeb developers
* Revisited version: Copyright (c) 2022 the CivetWeb developers
* MIT License
*/
/* Simple demo of a REST callback. */
#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#endif
#include <stdarg.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include "cJSON.h"
#include "civetweb.h"
#define PORT "8089"
#define HOST_INFO "http://localhost:8089"
#define EXAMPLE_URI "/res/*/*"
#define EXIT_URI "/exit"
int exitNow = 0;
static int
SendJSON(struct mg_connection *conn, cJSON *json_obj)
{
char *json_str = cJSON_PrintUnformatted(json_obj);
size_t json_str_len = strlen(json_str);
/* Send HTTP message header (+1 for \n) */
mg_send_http_ok(conn, "application/json; charset=utf-8", json_str_len + 1);
/* Send HTTP message content */
mg_write(conn, json_str, json_str_len);
/* Add a newline. This is not required, but the result is more
* human-readable in a debugger. */
mg_write(conn, "\n", 1);
/* Free string allocated by cJSON_Print* */
cJSON_free(json_str);
return (int)json_str_len;
}
static unsigned request = 0; /* demo data: request counter */
static int
ExampleGET(struct mg_connection *conn, const char *p1, const char *p2)
{
cJSON *obj = cJSON_CreateObject();
if (!obj) {
/* insufficient memory? */
mg_send_http_error(conn, 500, "Server error");
return 500;
}
printf("GET %s/%s\n", p1, p2);
cJSON_AddStringToObject(obj, "version", CIVETWEB_VERSION);
cJSON_AddStringToObject(obj, "path1", p1);
cJSON_AddStringToObject(obj, "path2", p2);
cJSON_AddNumberToObject(obj, "request", ++request);
SendJSON(conn, obj);
cJSON_Delete(obj);
return 200;
}
static int
ExampleDELETE(struct mg_connection *conn, const char *p1, const char *p2)
{
printf("DELETE %s/%s\n", p1, p2);
mg_send_http_error(conn,
204,
"%s",
""); /* Return "deleted" = "204 No Content" */
return 204;
}
static int
ExamplePUT(struct mg_connection *conn, const char *p1, const char *p2)
{
char buffer[1024];
int dlen = mg_read(conn, buffer, sizeof(buffer) - 1);
cJSON *obj, *elem;
unsigned newvalue;
printf("PUT %s/%s\n", p1, p2);
if ((dlen < 1) || (dlen >= sizeof(buffer))) {
mg_send_http_error(conn, 400, "%s", "No request body data");
return 400;
}
buffer[dlen] = 0;
obj = cJSON_Parse(buffer);
if (obj == NULL) {
mg_send_http_error(conn, 400, "%s", "Invalid request body data");
return 400;
}
elem = cJSON_GetObjectItemCaseSensitive(obj, "request");
if (!cJSON_IsNumber(elem)) {
cJSON_Delete(obj);
mg_send_http_error(conn,
400,
"%s",
"No \"request\" number in body data");
return 400;
}
newvalue = (unsigned)elem->valuedouble;
if ((double)newvalue != elem->valuedouble) {
cJSON_Delete(obj);
mg_send_http_error(conn,
400,
"%s",
"Invalid \"request\" number in body data");
return 400;
}
request = newvalue;
cJSON_Delete(obj);
mg_send_http_error(conn, 201, "%s", ""); /* Return "201 Created" */
return 201;
}
#if 0 /* Old version: User code had to split the url. */
static int
mg_vsplit(const char *url, const char *pattern, va_list va)
{
int ret = 0;
while (*url && *pattern) {
if (*url == *pattern) {
url++;
pattern++;
} else if (*pattern == '*') {
char *p = va_arg(va, char *);
size_t l = va_arg(va, size_t);
if (p == NULL || l == 0) {
return 0;
}
while ((*url != '/') && (*url != 0)) {
if (l == 0) {
return 0;
}
l--;
*p = *url;
p++;
url++;
}
*p = 0;
pattern++;
ret++;
} else {
return 0;
}
}
return ret;
}
static int
mg_split(const char *url, const char *pattern, ...)
{
int ret;
va_list va;
va_start(va, pattern);
ret = mg_vsplit(url, pattern, va);
va_end(va);
return ret;
}
#endif
static int
ExampleHandler(struct mg_connection *conn, void *cbdata)
{
char path1[1024], path2[1024];
const struct mg_request_info *ri = mg_get_request_info(conn);
const char *url = ri->local_uri;
size_t url_len = strlen(url);
/* Pattern matching */
#if 0 /* Old version: User code had to split the url. */
if (2
!= mg_split(
url, EXAMPLE_URI, path1, sizeof(path1), path2, sizeof(path2))) {
mg_send_http_error(conn, 404, "Invalid path: %s\n", url);
return 404;
}
#else /* New version: User mg_match. */
struct mg_match_context mcx;
mcx.case_sensitive = 0;
ptrdiff_t ret = mg_match(EXAMPLE_URI, url, &mcx);
if ((ret != url_len) || (mcx.num_matches != 2)) {
/* Note: Could have done this with a $ at the end of the match
* pattern as well. Then we would have to check for a return value
* of -1 only. Here we use this version as minimum modification
* of the existing code. */
printf("Match ret: %i\n", (int)ret);
mg_send_http_error(conn, 404, "Invalid path: %s\n", url);
return 404;
}
memcpy(path1, mcx.match[0].str, mcx.match[0].len);
path1[mcx.match[0].len] = 0;
memcpy(path2, mcx.match[1].str, mcx.match[1].len);
path2[mcx.match[1].len] = 0;
#endif
(void)cbdata; /* currently unused */
/* According to method */
if (0 == strcmp(ri->request_method, "GET")) {
return ExampleGET(conn, path1, path2);
}
if ((0 == strcmp(ri->request_method, "PUT"))
|| (0 == strcmp(ri->request_method, "POST"))
|| (0 == strcmp(ri->request_method, "PATCH"))) {
/* In this example, do the same for PUT, POST and PATCH */
return ExamplePUT(conn, path1, path2);
}
if (0 == strcmp(ri->request_method, "DELETE")) {
return ExampleDELETE(conn, path1, path2);
}
/* this is not a GET request */
mg_send_http_error(
conn, 405, "Only GET, PUT, POST, DELETE and PATCH method supported");
return 405;
}
static int
ExitHandler(struct mg_connection *conn, void *cbdata)
{
mg_printf(conn,
"HTTP/1.1 200 OK\r\nContent-Type: "
"text/plain\r\nConnection: close\r\n\r\n");
mg_printf(conn, "Server will shut down.\n");
mg_printf(conn, "Bye!\n");
exitNow = 1;
return 1;
}
static int
log_message(const struct mg_connection *conn, const char *message)
{
puts(message);
return 1;
}
int
main(int argc, char *argv[])
{
const char *options[] = {"listening_ports",
PORT,
"request_timeout_ms",
"10000",
"error_log_file",
"error.log",
0};
struct mg_callbacks callbacks;
struct mg_context *ctx;
int err = 0;
/* Init libcivetweb. */
mg_init_library(0);
/* Callback will print error messages to console */
memset(&callbacks, 0, sizeof(callbacks));
callbacks.log_message = log_message;
/* Start CivetWeb web server */
ctx = mg_start(&callbacks, 0, options);
/* Check return value: */
if (ctx == NULL) {
fprintf(stderr, "Cannot start CivetWeb - mg_start failed.\n");
return EXIT_FAILURE;
}
/* Add handler EXAMPLE_URI, to explain the example */
mg_set_request_handler(ctx, EXAMPLE_URI, ExampleHandler, 0);
mg_set_request_handler(ctx, EXIT_URI, ExitHandler, 0);
/* Show some info */
printf("Start example: %s%s\n", HOST_INFO, EXAMPLE_URI);
printf("Exit example: %s%s\n", HOST_INFO, EXIT_URI);
/* Wait until the server should be closed */
while (!exitNow) {
#ifdef _WIN32
Sleep(1000);
#else
sleep(1);
#endif
}
/* Stop the server */
mg_stop(ctx);
printf("Server stopped.\n");
printf("Bye!\n");
return EXIT_SUCCESS;
}