mirror of
https://github.com/polhenarejos/pico-keys-sdk
synced 2026-05-28 17:11:23 +02:00
Add headers in response.
Signed-off-by: Pol Henarejos <pol.henarejos@cttc.es>
This commit is contained in:
@@ -27,8 +27,16 @@
|
|||||||
} \
|
} \
|
||||||
} while(0)
|
} while(0)
|
||||||
|
|
||||||
|
#define CJSON_ADD_GENERIC_ITEM(hdl, ptr, item, value) do { \
|
||||||
|
if (hdl(ptr, item, value) == false) { \
|
||||||
|
response->status_code = 500; \
|
||||||
|
return 1; \
|
||||||
|
} \
|
||||||
|
} while(0)
|
||||||
|
|
||||||
#define CJSON_ADD_STRING(ptr, item, value) CJSON_ADD_GENERIC(cJSON_AddStringToObject, (ptr)->json, item, value)
|
#define CJSON_ADD_STRING(ptr, item, value) CJSON_ADD_GENERIC(cJSON_AddStringToObject, (ptr)->json, item, value)
|
||||||
#define CJSON_ADD_NUMBER(ptr, item, value) CJSON_ADD_GENERIC(cJSON_AddNumberToObject, (ptr)->json, item, value)
|
#define CJSON_ADD_NUMBER(ptr, item, value) CJSON_ADD_GENERIC(cJSON_AddNumberToObject, (ptr)->json, item, value)
|
||||||
#define CJSON_ADD_BOOL(ptr, item, value) CJSON_ADD_GENERIC(cJSON_AddBoolToObject, (ptr)->json, item, value)
|
#define CJSON_ADD_BOOL(ptr, item, value) CJSON_ADD_GENERIC(cJSON_AddBoolToObject, (ptr)->json, item, value)
|
||||||
|
#define CJSON_ADD_ITEM(ptr, item, value) CJSON_ADD_GENERIC_ITEM(cJSON_AddItemToObject, (ptr)->json, item, value)
|
||||||
|
|
||||||
#endif // _JSON_H_
|
#endif // _JSON_H_
|
||||||
|
|||||||
@@ -198,13 +198,20 @@ int rest_execute_route_handler(const rest_request_t *request, rest_route_handler
|
|||||||
response->json = NULL;
|
response->json = NULL;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
if (response->status_code == 0 || response->status_code == 200) {
|
if (response->status_code == 0 || response->status_code == 200 || response->status_code == 201 || response->status_code == 204) {
|
||||||
char *body = cJSON_PrintUnformatted(response->json);
|
char *body = cJSON_PrintUnformatted(response->json);
|
||||||
cJSON_Delete(response->json);
|
cJSON_Delete(response->json);
|
||||||
response->json = NULL;
|
response->json = NULL;
|
||||||
if (body == NULL) {
|
if (body == NULL) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
if (memcmp(body, "{}", 2) == 0) {
|
||||||
|
free(body);
|
||||||
|
body = strdup("");
|
||||||
|
if (body == NULL) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
response->body = body;
|
response->body = body;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -58,6 +58,7 @@ typedef enum {
|
|||||||
REST_HEADER_CONTENT_TYPE,
|
REST_HEADER_CONTENT_TYPE,
|
||||||
REST_HEADER_CONTENT_LENGTH,
|
REST_HEADER_CONTENT_LENGTH,
|
||||||
REST_HEADER_HOST,
|
REST_HEADER_HOST,
|
||||||
|
REST_HEADER_LOCATION,
|
||||||
REST_HEADER_ACCEPT,
|
REST_HEADER_ACCEPT,
|
||||||
REST_HEADER_X_SESSION_ID,
|
REST_HEADER_X_SESSION_ID,
|
||||||
REST_HEADER_X_SEQ,
|
REST_HEADER_X_SEQ,
|
||||||
@@ -73,34 +74,12 @@ typedef enum {
|
|||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
union {
|
union {
|
||||||
uint32_t int_param;
|
int32_t int_param;
|
||||||
char *str_param;
|
char *str_param;
|
||||||
} param;
|
} param;
|
||||||
rest_param_type_t type;
|
rest_param_type_t type;
|
||||||
} rest_param_t;
|
} rest_param_t;
|
||||||
|
|
||||||
typedef struct
|
|
||||||
{
|
|
||||||
rest_http_method_t method;
|
|
||||||
char path[REST_MAX_PATH_SIZE];
|
|
||||||
const char *body;
|
|
||||||
size_t body_len;
|
|
||||||
const char *content_type;
|
|
||||||
char *headers[REST_HEADER_TOTAL_COUNT];
|
|
||||||
rest_param_t params[REST_MAX_REQUEST_PARAMS];
|
|
||||||
} rest_request_t;
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
uint16_t status_code;
|
|
||||||
const char *content_type;
|
|
||||||
char *body; // heap !
|
|
||||||
size_t body_len;
|
|
||||||
cJSON *json;
|
|
||||||
char *headers[REST_HEADER_TOTAL_COUNT];
|
|
||||||
} rest_response_t;
|
|
||||||
|
|
||||||
typedef int (*rest_route_handler_t)(const rest_request_t *request, rest_response_t *response);
|
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
REST_ROUTE_NONE = 0x0,
|
REST_ROUTE_NONE = 0x0,
|
||||||
REST_ROUTE_REQUIRE_AUTH = 0x1,
|
REST_ROUTE_REQUIRE_AUTH = 0x1,
|
||||||
@@ -115,15 +94,10 @@ typedef enum {
|
|||||||
REST_SESSION_ROLE_ADMIN = 0x2
|
REST_SESSION_ROLE_ADMIN = 0x2
|
||||||
} rest_session_role_t;
|
} rest_session_role_t;
|
||||||
|
|
||||||
typedef struct {
|
typedef enum {
|
||||||
rest_http_method_t method;
|
REST_REQUEST_CONN_TYPE_PLAIN = 0,
|
||||||
const char *path;
|
REST_REQUEST_CONN_TYPE_TLS = 1
|
||||||
rest_route_handler_t handler;
|
} rest_request_conn_type_t;
|
||||||
rest_route_flags_t flags;
|
|
||||||
rest_route_param_parser_t param_parser;
|
|
||||||
rest_session_role_t role; // Minimum required role to access this route (only relevant if REST_ROUTE_REQUIRE_AUTH flag is set)
|
|
||||||
} rest_route_t;
|
|
||||||
|
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
REST_SESSION_UNKNOWN = 0,
|
REST_SESSION_UNKNOWN = 0,
|
||||||
@@ -144,8 +118,41 @@ typedef struct {
|
|||||||
uint32_t last_seq;
|
uint32_t last_seq;
|
||||||
rest_session_role_t role;
|
rest_session_role_t role;
|
||||||
rest_session_status_t status;
|
rest_session_status_t status;
|
||||||
|
uint8_t token[32];
|
||||||
|
uint8_t user_id;
|
||||||
} rest_session_t;
|
} rest_session_t;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
rest_http_method_t method;
|
||||||
|
char path[REST_MAX_PATH_SIZE];
|
||||||
|
const char *body;
|
||||||
|
size_t body_len;
|
||||||
|
const char *content_type;
|
||||||
|
char *headers[REST_HEADER_TOTAL_COUNT];
|
||||||
|
rest_param_t params[REST_MAX_REQUEST_PARAMS];
|
||||||
|
rest_session_t *session;
|
||||||
|
rest_request_conn_type_t conn_type;
|
||||||
|
} rest_request_t;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint16_t status_code;
|
||||||
|
const char *content_type;
|
||||||
|
char *body; // heap !
|
||||||
|
size_t body_len;
|
||||||
|
cJSON *json;
|
||||||
|
char *headers[REST_HEADER_TOTAL_COUNT];
|
||||||
|
} rest_response_t;
|
||||||
|
|
||||||
|
typedef int (*rest_route_handler_t)(const rest_request_t *request, rest_response_t *response);
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
rest_http_method_t method;
|
||||||
|
const char *path;
|
||||||
|
rest_route_handler_t handler;
|
||||||
|
rest_route_flags_t flags;
|
||||||
|
rest_route_param_parser_t param_parser;
|
||||||
|
rest_session_role_t role; // Minimum required role to access this route (only relevant if REST_ROUTE_REQUIRE_AUTH flag is set)
|
||||||
|
} rest_route_t;
|
||||||
|
|
||||||
extern int rest_execute_route_handler(const rest_request_t *request, rest_route_handler_t handler, rest_response_t *response);
|
extern int rest_execute_route_handler(const rest_request_t *request, rest_route_handler_t handler, rest_response_t *response);
|
||||||
extern int rest_response_set_error(rest_response_t *response, int status_code, const char *message);
|
extern int rest_response_set_error(rest_response_t *response, int status_code, const char *message);
|
||||||
@@ -174,4 +181,12 @@ extern void rest_debug_dump_payload(const char *tag, const char *buffer, size_t
|
|||||||
#define REST_DEBUG_LOG(...) do {} while (0)
|
#define REST_DEBUG_LOG(...) do {} while (0)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef ENABLE_EMULATION
|
||||||
|
#define REST_ABSOLUTE_HTTP_URI "http://127.0.0.1"
|
||||||
|
#define REST_ABSOLUTE_HTTPS_URI "https://127.0.0.1"
|
||||||
|
#else
|
||||||
|
#define REST_ABSOLUTE_HTTP_URI "http://192.168.7.1"
|
||||||
|
#define REST_ABSOLUTE_HTTPS_URI "https://192.168.7.2"
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -70,8 +70,26 @@ typedef struct {
|
|||||||
static rest_core1_job_t rest_core1_job = {0};
|
static rest_core1_job_t rest_core1_job = {0};
|
||||||
static rest_core1_result_t rest_core1_result = {0};
|
static rest_core1_result_t rest_core1_result = {0};
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
rest_header_id_t id;
|
||||||
|
const char *name;
|
||||||
|
} rest_header_descriptor_t;
|
||||||
|
|
||||||
|
static const rest_header_descriptor_t rest_http_headers[REST_HEADER_TOTAL_COUNT] = {
|
||||||
|
{ REST_HEADER_USER_AGENT, "User-Agent" },
|
||||||
|
{ REST_HEADER_AUTHORIZATION, "Authorization" },
|
||||||
|
{ REST_HEADER_CONTENT_TYPE, "Content-Type" },
|
||||||
|
{ REST_HEADER_CONTENT_LENGTH, "Content-Length" },
|
||||||
|
{ REST_HEADER_HOST, "Host" },
|
||||||
|
{ REST_HEADER_LOCATION, "Location" },
|
||||||
|
{ REST_HEADER_ACCEPT, "Accept" },
|
||||||
|
{ REST_HEADER_X_SESSION_ID, "X-Session-ID" },
|
||||||
|
{ REST_HEADER_X_SEQ, "X-Seq" },
|
||||||
|
{ REST_HEADER_X_SIGNATURE, "X-Signature" }
|
||||||
|
};
|
||||||
|
|
||||||
static void *rest_core1_thread(void *arg);
|
static void *rest_core1_thread(void *arg);
|
||||||
static void send_response(rest_conn_t *conn, int status_code, const char *status_text, const char *content_type, const char *body, size_t body_len);
|
static void send_response(rest_conn_t *conn, int status_code, const char *status_text, const char *content_type, const char *body, size_t body_len, char *headers[REST_HEADER_TOTAL_COUNT]);
|
||||||
void rest_close_conn(rest_conn_t *conn);
|
void rest_close_conn(rest_conn_t *conn);
|
||||||
|
|
||||||
static int rest_start_core1_job(rest_conn_t *conn, const rest_request_t *request, rest_route_handler_t handler) {
|
static int rest_start_core1_job(rest_conn_t *conn, const rest_request_t *request, rest_route_handler_t handler) {
|
||||||
@@ -122,7 +140,7 @@ static void *rest_core1_thread(void *arg) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void send_json(rest_conn_t *conn, int status_code, const char *status_text, const char *json_body) {
|
static void send_json(rest_conn_t *conn, int status_code, const char *status_text, const char *json_body) {
|
||||||
send_response(conn, status_code, status_text, "application/json", json_body, strlen(json_body));
|
send_response(conn, status_code, status_text, "application/json", json_body, strlen(json_body), NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void send_json_error(rest_conn_t *conn, int status_code, const char *error_message) {
|
static void send_json_error(rest_conn_t *conn, int status_code, const char *error_message) {
|
||||||
@@ -157,7 +175,7 @@ void rest_task(void) {
|
|||||||
if (conn != NULL) {
|
if (conn != NULL) {
|
||||||
if (rest_core1_result.ready && response->body != NULL && response->content_type != NULL) {
|
if (rest_core1_result.ready && response->body != NULL && response->content_type != NULL) {
|
||||||
uint16_t code = response->status_code == 0 ? 200 : response->status_code;
|
uint16_t code = response->status_code == 0 ? 200 : response->status_code;
|
||||||
send_response(conn, code, rest_status_text_from_code(code), response->content_type, response->body, response->body_len);
|
send_response(conn, code, rest_status_text_from_code(code), response->content_type, response->body, response->body_len, response->headers);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
send_json_error(conn, 500, "internal_error");
|
send_json_error(conn, 500, "internal_error");
|
||||||
@@ -248,8 +266,8 @@ void rest_close_conn(rest_conn_t *conn) {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
static void send_response(rest_conn_t *conn, int status_code, const char *status_text, const char *content_type, const char *body, size_t body_len) {
|
static void send_response(rest_conn_t *conn, int status_code, const char *status_text, const char *content_type, const char *body, size_t body_len, char *headers[REST_HEADER_TOTAL_COUNT]) {
|
||||||
char headers[256];
|
char headers_buf[256];
|
||||||
int header_len;
|
int header_len;
|
||||||
#ifdef ENABLE_EMULATION
|
#ifdef ENABLE_EMULATION
|
||||||
size_t sent_total = 0;
|
size_t sent_total = 0;
|
||||||
@@ -277,14 +295,34 @@ static void send_response(rest_conn_t *conn, int status_code, const char *status
|
|||||||
);
|
);
|
||||||
rest_debug_dump_payload("response-body", body, body_len);
|
rest_debug_dump_payload("response-body", body, body_len);
|
||||||
|
|
||||||
header_len = snprintf(headers, sizeof(headers),
|
char *p = headers_buf;
|
||||||
|
header_len = snprintf(p, sizeof(headers_buf),
|
||||||
"HTTP/1.0 %d %s\r\n"
|
"HTTP/1.0 %d %s\r\n"
|
||||||
"Content-Type: %s\r\n"
|
"Content-Type: %s\r\n"
|
||||||
"Content-Length: %lu\r\n"
|
"Content-Length: %lu\r\n"
|
||||||
"Connection: close\r\n"
|
"Connection: close\r\n",
|
||||||
"\r\n",
|
|
||||||
status_code, status_text, content_type, (unsigned long)body_len);
|
status_code, status_text, content_type, (unsigned long)body_len);
|
||||||
if (header_len <= 0 || (size_t)header_len >= sizeof(headers)) {
|
if (headers) {
|
||||||
|
for (int i = 0; i < REST_HEADER_TOTAL_COUNT; i++) {
|
||||||
|
if (headers[i] != NULL) {
|
||||||
|
int n = snprintf(p + header_len, sizeof(headers_buf) - (size_t)header_len, "%s: %s\r\n", rest_http_headers[i].name, headers[i]);
|
||||||
|
free(headers[i]);
|
||||||
|
headers[i] = NULL;
|
||||||
|
if (n <= 0 || header_len + n >= (int)sizeof(headers_buf)) {
|
||||||
|
rest_close_conn(conn);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
header_len += n;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (header_len + 2 >= (int)sizeof(headers_buf)) {
|
||||||
|
rest_close_conn(conn);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
memcpy(p + header_len, "\r\n", 2);
|
||||||
|
header_len += 2;
|
||||||
|
if (header_len <= 0 || (size_t)header_len >= sizeof(headers_buf)) {
|
||||||
rest_close_conn(conn);
|
rest_close_conn(conn);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -295,7 +333,7 @@ static void send_response(rest_conn_t *conn, int status_code, const char *status
|
|||||||
int want_retries = 0;
|
int want_retries = 0;
|
||||||
|
|
||||||
while (written < (size_t)header_len) {
|
while (written < (size_t)header_len) {
|
||||||
ret = mbedtls_ssl_write(&conn->ssl, (const unsigned char *)headers + written, (size_t)header_len - written);
|
ret = mbedtls_ssl_write(&conn->ssl, (const unsigned char *)headers_buf + written, (size_t)header_len - written);
|
||||||
if (ret == MBEDTLS_ERR_SSL_WANT_READ || ret == MBEDTLS_ERR_SSL_WANT_WRITE) {
|
if (ret == MBEDTLS_ERR_SSL_WANT_READ || ret == MBEDTLS_ERR_SSL_WANT_WRITE) {
|
||||||
if (++want_retries > 2048) {
|
if (++want_retries > 2048) {
|
||||||
rest_close_conn(conn);
|
rest_close_conn(conn);
|
||||||
@@ -336,7 +374,7 @@ static void send_response(rest_conn_t *conn, int status_code, const char *status
|
|||||||
#ifdef ENABLE_EMULATION
|
#ifdef ENABLE_EMULATION
|
||||||
while (sent_total < (size_t)header_len) {
|
while (sent_total < (size_t)header_len) {
|
||||||
#ifndef _MSC_VER
|
#ifndef _MSC_VER
|
||||||
ssize_t n = send(conn->sock, headers + sent_total, (size_t)header_len - sent_total, 0);
|
ssize_t n = send(conn->sock, headers_buf + sent_total, (size_t)header_len - sent_total, 0);
|
||||||
#else
|
#else
|
||||||
int n = -1;
|
int n = -1;
|
||||||
#endif
|
#endif
|
||||||
@@ -383,23 +421,6 @@ static void send_response(rest_conn_t *conn, int status_code, const char *status
|
|||||||
rest_close_conn(conn);
|
rest_close_conn(conn);
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
rest_header_id_t id;
|
|
||||||
const char *name;
|
|
||||||
} rest_header_descriptor_t;
|
|
||||||
|
|
||||||
static const rest_header_descriptor_t rest_http_headers[REST_HEADER_TOTAL_COUNT] = {
|
|
||||||
{ REST_HEADER_USER_AGENT, "User-Agent" },
|
|
||||||
{ REST_HEADER_AUTHORIZATION, "Authorization" },
|
|
||||||
{ REST_HEADER_CONTENT_TYPE, "Content-Type" },
|
|
||||||
{ REST_HEADER_CONTENT_LENGTH, "Content-Length" },
|
|
||||||
{ REST_HEADER_HOST, "Host" },
|
|
||||||
{ REST_HEADER_ACCEPT, "Accept" },
|
|
||||||
{ REST_HEADER_X_SESSION_ID, "X-Session-ID" },
|
|
||||||
{ REST_HEADER_X_SEQ, "X-Seq" },
|
|
||||||
{ REST_HEADER_X_SIGNATURE, "X-Signature" }
|
|
||||||
};
|
|
||||||
|
|
||||||
static void trim_ascii_ws_bounds(const char **start, const char **end) {
|
static void trim_ascii_ws_bounds(const char **start, const char **end) {
|
||||||
while (*start < *end && isspace((unsigned char)**start)) {
|
while (*start < *end && isspace((unsigned char)**start)) {
|
||||||
(*start)++;
|
(*start)++;
|
||||||
@@ -594,6 +615,7 @@ static int parse_request(rest_conn_t *conn, rest_request_t *request) {
|
|||||||
}
|
}
|
||||||
request->body = conn->request + headers_size;
|
request->body = conn->request + headers_size;
|
||||||
request->body_len = (size_t)content_length;
|
request->body_len = (size_t)content_length;
|
||||||
|
request->conn_type = (rest_request_conn_type_t)conn->conn_type;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -739,6 +761,7 @@ void rest_handle_request(rest_conn_t *conn) {
|
|||||||
send_json_error(conn, 401, "invalid_signature");
|
send_json_error(conn, 401, "invalid_signature");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
request->session = session;
|
||||||
}
|
}
|
||||||
if (rest_start_core1_job(conn, request, routes[i].handler) != 0) {
|
if (rest_start_core1_job(conn, request, routes[i].handler) != 0) {
|
||||||
send_json_error(conn, 500, "internal_error");
|
send_json_error(conn, 500, "internal_error");
|
||||||
|
|||||||
Reference in New Issue
Block a user