add: OPTIONS method support
add: relay tag endpoint
This commit is contained in:
parent
dab106c1f6
commit
420bdeb2aa
6 changed files with 158 additions and 26 deletions
57
endpoints/api_v1_relays_tag_STR.c
Normal file
57
endpoints/api_v1_relays_tag_STR.c
Normal file
|
@ -0,0 +1,57 @@
|
|||
#include <cJSON.h>
|
||||
#include <constants.h>
|
||||
#include <endpoints/api_v1_relays.h>
|
||||
#include <logger.h>
|
||||
#include <models/junction_tag.h>
|
||||
#include <models/relay.h>
|
||||
#include <models/tag.h>
|
||||
|
||||
void
|
||||
api_v1_relays_tag_STR_GET(struct mg_connection *c, endpoint_args_t *args, struct http_message *hm)
|
||||
{
|
||||
(void)hm;
|
||||
|
||||
int tag_id = tag_get_id(args[0].value.v_str);
|
||||
int *relays_ids = junction_tag_get_relays_for_tag_id(tag_id);
|
||||
if(relays_ids == NULL)
|
||||
{
|
||||
LOG_ERROR("failed to print relays json\n");
|
||||
mg_send_head(c, 500, 2, "Content-Type: application/json\r\n" STANDARD_HEADERS);
|
||||
mg_printf(c, "[]");
|
||||
return;
|
||||
}
|
||||
|
||||
cJSON *json = cJSON_CreateArray();
|
||||
|
||||
for(int i = 0; relays_ids[i] != 0; ++i)
|
||||
{
|
||||
relay_t* relay = relay_get_by_id(relays_ids[i]);
|
||||
|
||||
if(!relay)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
cJSON *json_relay = relay_to_json(relay);
|
||||
|
||||
cJSON_AddItemToArray(json, json_relay);
|
||||
|
||||
relay_free(relay);
|
||||
}
|
||||
|
||||
char *json_str = cJSON_Print(json);
|
||||
if (json_str == NULL)
|
||||
{
|
||||
LOG_ERROR("failed to print relays json\n");
|
||||
mg_send_head(c, 500, 2, "Content-Type: application/json\r\n" STANDARD_HEADERS);
|
||||
mg_printf(c, "[]");
|
||||
}
|
||||
else
|
||||
{
|
||||
mg_send_head(c, 200, strlen(json_str), "Content-Type: application/json\r\n" STANDARD_HEADERS);
|
||||
mg_printf(c, "%s", json_str);
|
||||
free(json_str);
|
||||
}
|
||||
|
||||
cJSON_Delete(json);
|
||||
free(relays_ids);
|
||||
}
|
|
@ -1,3 +1,4 @@
|
|||
#include <constants.h>
|
||||
#include <mongoose.h>
|
||||
#include <logger.h>
|
||||
#include <router.h>
|
||||
|
@ -13,7 +14,7 @@ handler_connection(struct mg_connection *c, int ev, void *p)
|
|||
|
||||
endpoint_t *endpoint = router_find_endpoint(hm->uri.p, hm->uri.len, &hm->method);
|
||||
|
||||
if(endpoint && endpoint->func)
|
||||
if(endpoint)
|
||||
{
|
||||
if(endpoint->func)
|
||||
{
|
||||
|
@ -29,7 +30,21 @@ handler_connection(struct mg_connection *c, int ev, void *p)
|
|||
}
|
||||
else
|
||||
{
|
||||
mg_send_head(c, 501, 0, "Content-Type: text/plain");
|
||||
if(endpoint->method == HTTP_METHOD_OPTIONS)
|
||||
{
|
||||
char options_header[256]; // TODO make more generic
|
||||
sprintf(options_header, "Allow: OPTIONS%s%s%s%s\r\n" STANDARD_HEADERS,
|
||||
endpoint->options & HTTP_METHOD_GET ? ", GET" : "",
|
||||
endpoint->options & HTTP_METHOD_POST ? ", POST" : "",
|
||||
endpoint->options & HTTP_METHOD_PUT ? ", PUT" : "",
|
||||
endpoint->options & HTTP_METHOD_DELETE ? ", DELETE" : ""
|
||||
);
|
||||
mg_send_head(c, 204, 0, options_header);
|
||||
}
|
||||
else
|
||||
{
|
||||
mg_send_head(c, 501, 0, "Content-Type: text/plain");
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
|
|
|
@ -6,4 +6,7 @@
|
|||
void
|
||||
api_v1_relays_GET(struct mg_connection *c, endpoint_args_t *args, struct http_message *hm);
|
||||
|
||||
void
|
||||
api_v1_relays_tag_STR_GET(struct mg_connection *c, endpoint_args_t *args, struct http_message *hm);
|
||||
|
||||
#endif /* CORE_ENDPOINTS_API_V1_RELAYS_H */
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
|
||||
#include <mongoose.h>
|
||||
|
||||
#define ENDPOINTS_MAX_COUNT 16
|
||||
#define ENDPOINTS_MAX_COUNT 128
|
||||
|
||||
typedef enum
|
||||
{
|
||||
|
@ -11,6 +11,15 @@ typedef enum
|
|||
ENDPOINT_ARG_TYPE_STR
|
||||
} endpoint_arg_type_e;
|
||||
|
||||
typedef enum
|
||||
{
|
||||
HTTP_METHOD_GET = (1 << 0),
|
||||
HTTP_METHOD_POST = (1 << 1),
|
||||
HTTP_METHOD_PUT = (1 << 2),
|
||||
HTTP_METHOD_DELETE = (1 << 3),
|
||||
HTTP_METHOD_OPTIONS = (1 << 4)
|
||||
} http_method_e;
|
||||
|
||||
typedef struct
|
||||
{
|
||||
endpoint_arg_type_e type;
|
||||
|
@ -25,10 +34,13 @@ typedef void (*endpoint_func_f)(struct mg_connection *c, endpoint_args_t *args,
|
|||
|
||||
typedef struct
|
||||
{
|
||||
const char *full_route;
|
||||
char **route;
|
||||
char *route_keeper;
|
||||
int methods;
|
||||
int method;
|
||||
int options;
|
||||
endpoint_func_f func;
|
||||
int trailing_slash;
|
||||
|
||||
int args_count;
|
||||
endpoint_args_t *args;
|
||||
|
@ -40,11 +52,11 @@ typedef struct
|
|||
void
|
||||
router_init();
|
||||
|
||||
void
|
||||
router_register_endpoint(const char *route, int methods, endpoint_func_f func);
|
||||
endpoint_t*
|
||||
router_register_endpoint(const char *route, int method, endpoint_func_f func);
|
||||
|
||||
endpoint_t*
|
||||
router_find_endpoint(const char *uri_str, size_t uri_len, struct mg_str *method);
|
||||
router_find_endpoint(const char *uri_str, size_t uri_len, struct mg_str *method_str);
|
||||
|
||||
void
|
||||
router_free();
|
||||
|
|
83
router.c
83
router.c
|
@ -7,12 +7,6 @@
|
|||
#include <endpoints/api_v1_controllers.h>
|
||||
#include <endpoints/api_v1_relays.h>
|
||||
|
||||
static const int HTTP_METHOD_GET = (1 << 0);
|
||||
static const int HTTP_METHOD_POST = (1 << 1);
|
||||
static const int HTTP_METHOD_PUT = (1 << 2);
|
||||
static const int HTTP_METHOD_DELETE = (1 << 3);
|
||||
static const int HTTP_METHOD_OPTIONS = (1 << 4);
|
||||
|
||||
static endpoint_t endpoints[ENDPOINTS_MAX_COUNT];
|
||||
static endpoint_t endpoint_index;
|
||||
static endpoint_t endpoint_not_found;
|
||||
|
@ -37,29 +31,55 @@ endpoint_not_found_func(struct mg_connection *c, endpoint_args_t *args, struct h
|
|||
mg_printf(c, "not found");
|
||||
}
|
||||
|
||||
static struct mg_str
|
||||
get_method_str_for_int(int method_int)
|
||||
{
|
||||
if(method_int == HTTP_METHOD_GET)
|
||||
{
|
||||
return mg_mk_str("GET");
|
||||
}
|
||||
if(method_int == HTTP_METHOD_POST)
|
||||
{
|
||||
return mg_mk_str("POST");
|
||||
}
|
||||
if(method_int == HTTP_METHOD_PUT)
|
||||
{
|
||||
return mg_mk_str("PUT");
|
||||
}
|
||||
if(method_int == HTTP_METHOD_DELETE)
|
||||
{
|
||||
return mg_mk_str("DELETE");
|
||||
}
|
||||
if(method_int == HTTP_METHOD_OPTIONS)
|
||||
{
|
||||
return mg_mk_str("OPTIONS");
|
||||
}
|
||||
return mg_mk_str("GET");
|
||||
}
|
||||
|
||||
void
|
||||
router_init()
|
||||
{
|
||||
// add index endpoint
|
||||
endpoint_index.route = NULL;
|
||||
endpoint_index.func = endpoint_index_func;
|
||||
endpoint_index.methods = 0;
|
||||
endpoint_index.method = 0;
|
||||
endpoint_index.args_count = 0;
|
||||
endpoint_index.args = NULL;
|
||||
|
||||
// add 404 endpoint
|
||||
endpoint_not_found.route = NULL;
|
||||
endpoint_not_found.func = endpoint_not_found_func;
|
||||
endpoint_not_found.methods = 0;
|
||||
endpoint_not_found.method = 0;
|
||||
endpoint_not_found.args_count = 0;
|
||||
endpoint_not_found.args = NULL;
|
||||
|
||||
router_register_endpoint("/api/v1/schedules/", HTTP_METHOD_GET, api_v1_schedules_GET);
|
||||
router_register_endpoint("/api/v1/schedules/", HTTP_METHOD_POST, api_v1_schedules_POST);
|
||||
router_register_endpoint("/api/v1/schedules/{str}/", HTTP_METHOD_GET, api_v1_schedules_STR_GET);
|
||||
router_register_endpoint("/api/v1/schedules/{str}/", HTTP_METHOD_PUT, api_v1_schedules_STR_PUT);
|
||||
router_register_endpoint("/api/v1/schedules/{str}/", HTTP_METHOD_DELETE, api_v1_schedules_STR_DELETE);
|
||||
router_register_endpoint("/api/v1/schedules/tag/{str}/", HTTP_METHOD_GET, api_v1_schedules_tag_STR_GET);
|
||||
router_register_endpoint("/api/v1/schedules/{str}", HTTP_METHOD_GET, api_v1_schedules_STR_GET);
|
||||
router_register_endpoint("/api/v1/schedules/{str}", HTTP_METHOD_PUT, api_v1_schedules_STR_PUT);
|
||||
router_register_endpoint("/api/v1/schedules/{str}", HTTP_METHOD_DELETE, api_v1_schedules_STR_DELETE);
|
||||
router_register_endpoint("/api/v1/schedules/tag/{str}", HTTP_METHOD_GET, api_v1_schedules_tag_STR_GET);
|
||||
|
||||
router_register_endpoint("/api/v1/controllers/discover/", HTTP_METHOD_POST, api_v1_controllers_discover_POST);
|
||||
router_register_endpoint("/api/v1/controllers/", HTTP_METHOD_GET, api_v1_controllers_GET);
|
||||
|
@ -72,19 +92,38 @@ router_init()
|
|||
router_register_endpoint("/api/v1/controllers/{str}/relays/{int}", HTTP_METHOD_PUT, api_v1_controllers_STR_relays_INT_PUT);
|
||||
|
||||
router_register_endpoint("/api/v1/relays/", HTTP_METHOD_GET, api_v1_relays_GET);
|
||||
router_register_endpoint("/api/v1/relays/tag/{str}", HTTP_METHOD_GET, api_v1_relays_tag_STR_GET);
|
||||
}
|
||||
|
||||
void
|
||||
router_register_endpoint(const char *route, int methods, endpoint_func_f func)
|
||||
endpoint_t*
|
||||
router_register_endpoint(const char *route, int method, endpoint_func_f func)
|
||||
{
|
||||
endpoint_t *options_endpoint = NULL;
|
||||
if(method != HTTP_METHOD_OPTIONS)
|
||||
{
|
||||
struct mg_str method_str = get_method_str_for_int(HTTP_METHOD_OPTIONS);
|
||||
options_endpoint = router_find_endpoint(route, strlen(route), &method_str);
|
||||
if(options_endpoint == &endpoint_not_found)
|
||||
{
|
||||
options_endpoint = router_register_endpoint(route, HTTP_METHOD_OPTIONS, NULL);
|
||||
}
|
||||
}
|
||||
|
||||
if(endpoints_registered >= ENDPOINTS_MAX_COUNT)
|
||||
{
|
||||
LOG_ERROR("can't register more than %d endpoints\n", ENDPOINTS_MAX_COUNT);
|
||||
return;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
endpoint_t *endpoint = &endpoints[endpoints_registered];
|
||||
|
||||
endpoint->full_route = route;
|
||||
endpoint->trailing_slash = 0; // unused because trailing slashes are optional. TODO make option
|
||||
if(route[strlen(route)] == delimiter[0])
|
||||
{
|
||||
endpoint->trailing_slash = 1;
|
||||
}
|
||||
|
||||
int route_parts_count = 1;
|
||||
size_t route_len = strlen(route);
|
||||
|
||||
|
@ -130,8 +169,16 @@ router_register_endpoint(const char *route, int methods, endpoint_func_f func)
|
|||
}
|
||||
|
||||
endpoint->func = func;
|
||||
endpoint->methods = methods;
|
||||
endpoint->method = method;
|
||||
endpoint->options = 0;
|
||||
|
||||
if(options_endpoint)
|
||||
{
|
||||
options_endpoint->options |= method;
|
||||
}
|
||||
|
||||
++endpoints_registered;
|
||||
return endpoint;
|
||||
}
|
||||
|
||||
static int
|
||||
|
@ -163,8 +210,6 @@ get_method_int_for_str(struct mg_str *method_str)
|
|||
endpoint_t*
|
||||
router_find_endpoint(const char *uri_str, size_t uri_len, struct mg_str *method_str)
|
||||
{
|
||||
(void)uri_str;
|
||||
(void)uri_len;
|
||||
char *uri = malloc(sizeof(char) * (uri_len + 1));
|
||||
strncpy(uri, uri_str, uri_len);
|
||||
uri[uri_len] = '\0';
|
||||
|
@ -173,7 +218,7 @@ router_find_endpoint(const char *uri_str, size_t uri_len, struct mg_str *method_
|
|||
|
||||
for(int i = 0; i < endpoints_registered; ++i)
|
||||
{
|
||||
if(endpoints[i].methods & method)
|
||||
if(endpoints[i].method & method)
|
||||
{
|
||||
endpoints[i].possible_route = 1;
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue