#include #include #include #include #include #include #include #include void api_v1_schedules_POST(struct http_message *hm, endpoint_args_t *args, endpoint_response_t *response) { (void)args; cJSON *json = cJSON_ParseWithLength(hm->body.p, hm->body.len); if(json == NULL) { static const char content[] = "no valid json was supplied"; response->status_code = 400; response->content_type = "text/plain"; response->content_length = STRLEN(content);; response->content = content; response->alloced_content = false; return; } cJSON *json_name = cJSON_GetObjectItemCaseSensitive(json, "name"); if(!cJSON_IsString(json_name) || (json_name->valuestring == NULL)) { LOG_DEBUG("no name for schedule provided\n"); cJSON_Delete(json); static const char content[] = "no name for schedule provided"; response->status_code = 400; response->content_type = "text/plain"; response->content_length = STRLEN(content);; response->content = content; response->alloced_content = false; return; } cJSON *json_period; cJSON *json_periods = cJSON_GetObjectItemCaseSensitive(json, "periods"); schedule_t *new_schedule = malloc(sizeof(schedule_t)); new_schedule->id = 0; uuid_generate(new_schedule->uid); strncpy(new_schedule->name, json_name->valuestring, MAX_NAME_LENGTH); new_schedule->name[MAX_NAME_LENGTH] = '\0'; int periods_count = cJSON_GetArraySize(json_periods); new_schedule->periods = malloc(sizeof(period_t) * periods_count); int periods_valid = 0; cJSON_ArrayForEach(json_period, json_periods) { cJSON *json_period_start = cJSON_GetObjectItemCaseSensitive(json_period, "start"); cJSON *json_period_end = cJSON_GetObjectItemCaseSensitive(json_period, "end"); if(!cJSON_IsString(json_period_start) || (json_period_start->valuestring == NULL)) { LOG_DEBUG("period is missing start\n"); cJSON_Delete(json); schedule_free(new_schedule); static const char content[] = "one period is missing a start"; response->status_code = 400; response->content_type = "text/plain"; response->content_length = STRLEN(content);; response->content = content; response->alloced_content = false; return; } if(!cJSON_IsString(json_period_end) || (json_period_end->valuestring == NULL)) { LOG_DEBUG("period is missing end\n"); cJSON_Delete(json); schedule_free(new_schedule); static const char content[] = "one period is missing an end"; response->status_code = 400; response->content_type = "text/plain"; response->content_length = STRLEN(content);; response->content = content; response->alloced_content = false; return; } uint16_t start; uint16_t end; if(period_helper_parse_hhmm(json_period_start->valuestring, &start)) { LOG_DEBUG("couldn't parse start '%s'\n", json_period_start->valuestring); cJSON_Delete(json); schedule_free(new_schedule); static const char content[] = "the start for one period is invalid"; response->status_code = 400; response->content_type = "text/plain"; response->content_length = STRLEN(content);; response->content = content; response->alloced_content = false; return; } if(period_helper_parse_hhmm(json_period_end->valuestring, &end)) { LOG_DEBUG("couldn't parse end '%s'\n", json_period_end->valuestring); cJSON_Delete(json); schedule_free(new_schedule); static const char content[] = "the end for one period is invalid"; response->status_code = 400; response->content_type = "text/plain"; response->content_length = STRLEN(content);; response->content = content; response->alloced_content = false; return; } new_schedule->periods[periods_valid].start = start; new_schedule->periods[periods_valid].end = end; ++periods_valid; } new_schedule->periods_count = periods_valid; schedule_save(new_schedule); junction_tag_remove_for_schedule(new_schedule->id); cJSON *json_tag; cJSON *json_tags = cJSON_GetObjectItemCaseSensitive(json, "tags"); cJSON_ArrayForEach(json_tag, json_tags) { if(!cJSON_IsString(json_tag) || (json_tag->valuestring == NULL)) { LOG_DEBUG("invalid tag in tags\n"); continue; } const char *tag = json_tag->valuestring; int tag_id = tag_get_id(tag); if(tag_id == 0) { tag_save(tag_id, tag); tag_id = tag_get_id(tag); } junction_tag_insert(tag_id, 0, new_schedule->id); } cJSON_Delete(json); json = schedule_to_json(new_schedule); char *json_str = cJSON_Print(json); if (json_str == NULL) { LOG_ERROR("failed to print schedule json\n"); static const char content[] = "failed to print json for schedule"; response->status_code = 500; response->content_type = "text/plain"; response->content_length = STRLEN(content);; response->content = content; response->alloced_content = false; } else { response->status_code = 201; response->content_type = "application/json"; response->content_length = strlen(json_str); response->content = json_str; response->alloced_content = true; } cJSON_Delete(json); schedule_free(new_schedule); } void api_v1_schedules_GET(struct http_message *hm, endpoint_args_t *args, endpoint_response_t *response) { (void)args; (void)hm; schedule_t** all_schedules = schedule_get_all(); cJSON *json = cJSON_CreateArray(); for(int i = 0; all_schedules[i] != NULL; ++i) { cJSON *json_schedule = schedule_to_json(all_schedules[i]); cJSON_AddItemToArray(json, json_schedule); } char *json_str = cJSON_Print(json); if (json_str == NULL) { LOG_ERROR("failed to print schedules json\n"); static const char content[] = "failed to print json for schedules"; response->status_code = 500; response->content_type = "text/plain"; response->content_length = STRLEN(content);; response->content = content; response->alloced_content = false; } else { response->status_code = 200; response->content_type = "application/json"; response->content_length = strlen(json_str); response->content = json_str; response->alloced_content = true; } cJSON_Delete(json); schedule_free_list(all_schedules); }