cueOS  2.4
cueOS - Universal Show Control OS for ARM
DMX512_rest_api.c
1/***============================================================================================================================
2 * Depedencies inclusion
3 * Necessary dependencies should be declared here. Header file should contain as little dependecies declarations as possible
4 *=============================================================================================================================*/
5
6#include "DMX512_engine.h"
7#include "http_server.h"
8#include "http_defs.h"
9#include "http_response.h"
10#include "http_request.h"
11#include "json_parser.h"
12#include "DMX512_rest_api.h"
13
14//TODO: Maybe migrate to third party JSON parsing tool JSMN (https://zerge.com/jsmn)
15//For ease of use and efficiency ?
16//MEH.. TBD JSMN doesn't allow for json string generation which wa the point of migrating...
17
18/***============================================================================================================================
19 * Private functions definitions
20 * These functions are only accessible from within the file's scope
21 *=============================================================================================================================*/
22
23//TODO: comment this !!
24static void _DMX512_rest_api_404(http_request_s *req){
29}
30
31//TODO: comment this !!
32static void _DMX512_rest_api_405(http_request_s *req){
37}
38
46static void _DMX512_rest_api_get_fixtures(http_request_s *req){
47
49
50 uint8_t endpoint_str_len = strlen(DMX512_rest_api_endpoints_str[DMX512_API_ENDPOINT_PATCH]);
51
52 char *id_str = strnstr(req->uri + endpoint_str_len, "/", strlen(req->uri) - endpoint_str_len);
53
55
56 if(id_str != NULL){
57
58 char *garbage = NULL;
59
60 uint16_t id = strtoul(id_str + 1, &garbage, 10);
61
62 DMX512_fixture_s *fixture;
63
64 err = garbage[0] == 0 ? DMX512_engine_patch_get(id, &fixture) : DMX512_ENGINE_INSTANCE_INVALID;
65
66 if(err == DMX512_ENGINE_OK){
71 json_parser_json_string_nest(res_body, "fixture", json_fixture, 1, 0);
72 json_parser_json_string_put_int_pair(res_body, "errcode", err, 0, 1);
74 json_parser_free_json_string(json_fixture);
76 }else{
78 json_parser_json_string_put_int_pair(res_body, "errcode", -err, 0, 1);
81 }
82
83
84 }else{
85
86 uint16_t start = 0;
87 uint16_t stop = 0;
88
89 for(uint8_t i=0; i<req->param_count; i++){
91 start = atoi(req->params[i].val);
92 }else if(!strcmp(req->params[i].arg, DMX512_rest_api_args_str[DMX512_API_PARAM_STOP])){
93 stop = atoi(req->params[i].val);
94 }
95 }
96
98
99 if(fixture_pool->fixture_count > 0 && start < stop){
100
102
103 for(uint i=start; i<stop; i++){
104 if(i < fixture_pool->fixture_count){
110 json_fixture_array, json_fixture,
111 i == start ? 1 : 0, i == stop-1 ||
112 i == fixture_pool->fixture_count-1 ?
113 1 : 0
114 );
115 json_parser_free_json_string(json_fixture);
116 }
117 }
118 json_parser_json_string_nest(res_body, "fixtures", json_fixture_array, 1, 0);
119 json_parser_free_json_string(json_fixture_array);
120 json_parser_json_string_put_int_pair(res_body, "errcode", -DMX512_ENGINE_OK, 0, 1);
123
124 }else{
129
130 }
131
132 }
133
134}
135
144static void _DMX512_rest_api_add_fixture(http_request_s *req){
145
147
148 uint16_t fixture_id = DMX512_CHANNEL_ADDRESS_MAX + 1;
149 uint16_t fixture_addr = DMX512_CHANNEL_ADDRESS_MAX + 1;
150 uint16_t fixture_ch = DMX512_CHANNEL_ADDRESS_MAX + 1;
151
154
155 for(uint8_t i=0; i<json_data->param_count;i++){
156 if(strcmp(json_data->params[i].key, DMX512_rest_api_args_str[DMX512_API_PARAM_ID]) == 0){
157 fixture_id = atoi(json_data->params[i].val);
158 }else if(strcmp(json_data->params[i].key, DMX512_rest_api_args_str[DMX512_API_PARAM_ADDRESS]) == 0){
159 fixture_addr = atoi(json_data->params[i].val);
160 }else if(strcmp(json_data->params[i].key, DMX512_rest_api_args_str[DMX512_API_PARAM_CHANNEL_COUNT]) == 0){
161 fixture_ch = atoi(json_data->params[i].val);
162 }
163 }
164
165 err = DMX512_engine_patch_add(fixture_id, fixture_addr, fixture_ch);
166
167 if(err == DMX512_ENGINE_OK){
172 json_parser_json_string_nest(res_body, "fixture", json_fixture, 1, 0);
173 json_parser_json_string_put_int_pair(res_body, "errcode", err, 0, 1);
175 json_parser_free_json_string(json_fixture);
177 }else{
179 json_parser_json_string_put_int_pair(res_body, "errcode", -err, 0, 1);
182 }
183
185
186}
187
195static void _DMX512_rest_api_get_scenes(http_request_s *req){
196
198
199 uint8_t endpoint_str_len = strlen(DMX512_rest_api_endpoints_str[DMX512_API_ENDPOINT_SCENES]);
200
201 char *id_str = strnstr(req->uri + endpoint_str_len, "/", strlen(req->uri) - endpoint_str_len);
202
204
205 if(id_str != NULL){
206
207 char *garbage = NULL;
208
209 uint16_t id = strtoul(id_str + 1, &garbage, 10);
210
211 DMX512_scene_s *scene;
212
213 err = garbage[0] == 0 ? DMX512_engine_scene_get(id, &scene) : DMX512_ENGINE_INSTANCE_INVALID;
214
215 if(err == DMX512_ENGINE_OK){
216
218
222
223 json_parser_json_string_s *json_fixture_preset_array = json_parser_json_string_new();
224
225
226 if(scene->fixture_preset_count>0){
227 for(uint16_t i=0; i<scene->fixture_preset_count; i++){
229 json_fixture_preset_array,
230 scene->fixture_presets[i].fixture->id,
231 i==0 ? 1:0,
232 i== scene->fixture_preset_count - 1 ? 1 : 0
233 );
234 }
235 }else{
236 json_parser_json_string_put_array_object(json_fixture_preset_array,NULL,1,1);
237 }
238
239 json_parser_json_string_nest(json_scene, "fixture_presets", json_fixture_preset_array, 0, 1);
240 json_parser_free_json_string(json_fixture_preset_array);
241 json_parser_json_string_nest(res_body, "scene", json_scene, 1, 0);
243 json_parser_json_string_put_int_pair(res_body, "errcode", err, 0, 1);
246 }else{
248 json_parser_json_string_put_int_pair(res_body, "errcode", -err, 0, 1);
251 }
252
253
254 }else{
255
256 uint16_t start = 0;
257 uint16_t stop = 0;
258
259 for(uint8_t i=0; i<req->param_count; i++){
262 start = atoi(req->params[i].val);
263 }else if(!strncmp(req->params[i].arg, DMX512_rest_api_args_str[DMX512_API_PARAM_STOP],
265 stop = atoi(req->params[i].val);
266 }
267 }
268
270
271 if(scene_pool->scene_count > 0 && start < stop){
272
274
275 for(uint i=start; i<stop; i++){
276 if(i < scene_pool->scene_count){
282 json_scenes_array, json_scene,
283 i == start ? 1 : 0, i == stop-1 ||
284 i == scene_pool->scene_count-1 ?
285 1 : 0
286 );
288 }
289 }
290
291 json_parser_json_string_nest(res_body, "scenes", json_scenes_array, 1, 0);
292 json_parser_free_json_string(json_scenes_array);
293 json_parser_json_string_put_int_pair(res_body, "errcode", -DMX512_ENGINE_OK, 0, 1);
296
297 }else{
298
303
304 }
305
306 }
307
308}
309
317static void _DMX512_rest_api_add_scene(http_request_s *req){
318
320
321 uint16_t scene_id = DMX512_CHANNEL_ADDRESS_MAX + 1;
322 uint16_t scene_fadein = DMX512_CHANNEL_ADDRESS_MAX + 1;
323 uint16_t scene_fadeout = DMX512_CHANNEL_ADDRESS_MAX + 1;
324
327
328 for(uint8_t i=0; i<json_data->param_count;i++){
329 if(strcmp(json_data->params[i].key, DMX512_rest_api_args_str[DMX512_API_PARAM_ID]) == 0){
330 scene_id = atoi(json_data->params[i].val);
331 }else if(strcmp(json_data->params[i].key, DMX512_rest_api_args_str[DMX512_API_PARAM_FADEIN_TIME]) == 0){
332 scene_fadein = atoi(json_data->params[i].val);
333 }else if(strcmp(json_data->params[i].key, DMX512_rest_api_args_str[DMX512_API_PARAM_FADEOUT_TIME]) == 0){
334 scene_fadeout = atoi(json_data->params[i].val);
335 }
336 }
337
338 err = DMX512_engine_scene_add(scene_id, scene_fadein, scene_fadeout);
339
340 if(err == DMX512_ENGINE_OK){
345 json_parser_json_string_nest(res_body, "fixture", json_fixture, 1, 0);
346 json_parser_json_string_put_int_pair(res_body, "errcode", err, 0, 1);
348 json_parser_free_json_string(json_fixture);
350 }else{
352 json_parser_json_string_put_int_pair(res_body, "errcode", -err, 0, 1);
355 }
356
358
359}
360
361static void _DMX512_rest_api_configure(http_request_s *req){
362 char *content = req->content;
363}
364
365
366/***============================================================================================================================
367 * Public functions definitions
368 * These functions can be accessed outside of the file's scope
369 * @see DMX512_rest_api.h for declarations
370 *=============================================================================================================================*/
371
381
383
384 for(endpoint=DMX512_API_ENDPOINT_PATCH; endpoint<DMX512_API_ENDPOINT_DEFAULT; endpoint++){
385 if(!strncmp(req->uri, DMX512_rest_api_endpoints_str[endpoint], strlen(DMX512_rest_api_endpoints_str[endpoint]))){
386 break;
387 }
388 }
389
390 switch(endpoint){
392 switch(req->method){
393 case HTTP_GET: _DMX512_rest_api_get_fixtures(req); break;
394 case HTTP_POST: _DMX512_rest_api_add_fixture(req); break;
395 default: _DMX512_rest_api_405(req); break;
396 }
397 break;
399 switch(req->method){
400 case HTTP_GET: _DMX512_rest_api_get_scenes(req); break;
401 case HTTP_POST: _DMX512_rest_api_add_scene(req); break;
402 default: _DMX512_rest_api_405(req); break;
403 }
404 break;
406 switch(req->method){
407 case HTTP_POST: _DMX512_rest_api_configure(req); break;
408 default: _DMX512_rest_api_405(req); break;
409 }
410 break;
411 default:
412 _DMX512_rest_api_404(req);
413 break;
414 }
415}
#define DMX512_CHANNEL_ADDRESS_MAX
DMX512 max channel address.
Definition: DMX512_defs.h:13
DMX512_engine_err_e
DMX512 engine error index constants.
Definition: DMX512_defs.h:24
static char *const DMX512_engine_errs_str[]
DMX512 engine error strings constants TODO: this should be put into DMX512 API TODO: largely extend s...
Definition: DMX512_defs.h:41
@ DMX512_ENGINE_OK
Definition: DMX512_defs.h:25
@ DMX512_ENGINE_INSTANCE_INVALID
Definition: DMX512_defs.h:27
DMX512_engine_err_e DMX512_engine_scene_get(uint16_t scene_id, DMX512_scene_s **scene)
Wrapper for "DMX512_scene_pool_get" function. Provides context to the specified function using DMX512...
DMX512_engine_err_e DMX512_engine_patch_add(uint16_t fixture_id, uint16_t address, uint16_t ch_count)
Wrapper for "DMX512_fixture_pool_add" function. Provides context to the specified function using DMX5...
DMX512_engine_err_e DMX512_engine_patch_get(uint16_t fixture_id, DMX512_fixture_s **fixture)
Wrapper for "DMX512_fixture_pool_get" function. Provides context to the specified function using DMX5...
DMX512_engine_err_e DMX512_engine_scene_add(uint16_t scene_id, uint16_t fadein_time, uint16_t fadeout_time)
Wrapper for "DMX512_scene_pool_add" function. Provides context to the specified function using DMX512...
DMX512_fixture_pool_s * DMX512_engine_patch_get_all(void)
Returns the current engine fixture patch.
DMX512_scene_pool_s * DMX512_engine_scene_get_all(void)
Returns the current engine scenes.
static char *const http_status_codes_str[]
list of HTTP status codes strings.
Definition: http_defs.h:83
@ HTTP_STATUS_CODE_400
Definition: http_defs.h:28
@ HTTP_STATUS_CODE_404
Definition: http_defs.h:31
@ HTTP_STATUS_CODE_200
Definition: http_defs.h:22
@ HTTP_STATUS_CODE_405
Definition: http_defs.h:32
@ HTTP_GET
Definition: http_defs.h:63
@ HTTP_POST
Definition: http_defs.h:64
@ HTTP_CONTENT_TYPE_JSON
Definition: http_defs.h:56
void http_response_prepare_dynamic(http_response_s *res, http_status_code_e status_code, http_content_types_e content_type, char *content)
Pre-formats response to HTTP response using provided response header parameters.
void json_parser_free_json_string(json_parser_json_string_s *json_string)
Frees a json string instance.
Definition: json_parser.c:294
void json_parser_json_string_put_str_pair(json_parser_json_string_s *json_string, char *key, char *val, uint8_t is_final, uint8_t is_initial)
Puts a JSON key/(string)value pair into a provided json string object;.
Definition: json_parser.c:89
void json_parser_json_string_nest(json_parser_json_string_s *json_string, char *key, json_parser_json_string_s *nested_json_string, uint8_t is_initial, uint8_t is_final)
Nests a child JSON string into a parent JSON string.
Definition: json_parser.c:121
void json_parser_json_string_put_array_int(json_parser_json_string_s *json_string, uint16_t val, uint8_t is_initial, uint8_t is_final)
Nests a child JSON string into a parent JSON array.
Definition: json_parser.c:177
void json_parser_json_string_put_array_object(json_parser_json_string_s *json_string, json_parser_json_string_s *json_object, uint8_t is_initial, uint8_t is_final)
Nests a child JSON string into a parent JSON array.
Definition: json_parser.c:150
void json_parser_json_string_put_int_pair(json_parser_json_string_s *json_string, char *key, uint16_t val, uint8_t is_initial, uint8_t is_final)
Puts a JSON key/(integer)value pair into a provided json string object;.
Definition: json_parser.c:56
void json_parser_free_json_object(json_parser_json_object_s *json_object)
Frees a parsed json object.
Definition: json_parser.c:263
json_parser_json_string_s * json_parser_json_string_new(void)
Creates and initialises a new json string instance.
Definition: json_parser.c:282
json_parser_json_object_s * json_parser_parse(char *json_string, uint16_t json_string_len)
Retrieves key/value pairs from a JSON encoded string.
Definition: json_parser.c:207
static const char *const DMX512_rest_api_endpoints_str[]
DMX512 API endpoint string.
DMX512_rest_api_endpoint_e
DMX512 API endpoint string indexes.
void DMX512_rest_api_router(http_request_s *req)
Parses the current HTTP request method and URI in order to route requests to a specific api function.
static char *const DMX512_rest_api_args_str[]
DMX512 API parameter strings.
@ DMX512_API_ENDPOINT_CONFIGURE
@ DMX512_API_ENDPOINT_DEFAULT
@ DMX512_API_ENDPOINT_SCENES
@ DMX512_API_ENDPOINT_PATCH
@ DMX512_API_PARAM_FADEOUT_TIME
@ DMX512_API_PARAM_CHANNEL_COUNT
@ DMX512_API_PARAM_STOP
@ DMX512_API_PARAM_ID
@ DMX512_API_PARAM_FADEIN_TIME
@ DMX512_API_PARAM_ADDRESS
@ DMX512_API_PARAM_START
DMX512 fixture pool structure object.
DMX512_fixture_s * fixtures
DMX512 fixture structure object.
uint16_t ch_count
DMX512 scene pool structure object.
DMX512_scene_s * scenes
DMX512 scene structure object.
Definition: DMX512_scene.h:54
DMX512_fixture_preset_s * fixture_presets
Definition: DMX512_scene.h:59
uint16_t fadein_time
Definition: DMX512_scene.h:56
uint16_t fixture_preset_count
Definition: DMX512_scene.h:58
uint16_t fadeout_time
Definition: DMX512_scene.h:57
HTTP request object.
Definition: http_request.h:44
uint8_t param_count
Definition: http_request.h:51
http_method_e method
Definition: http_request.h:49
http_param_s * params
Definition: http_request.h:55
uint16_t content_length
Definition: http_request.h:53
char * content
Definition: http_request.h:48
http_response_s * res
Definition: http_request.h:56
JSON Objects structure object.
Definition: json_parser.h:30
json_parser_json_param_s * params
Definition: json_parser.h:31
JSON String structure object.
Definition: json_parser.h:38