2021-11-22 11:24:16 +00:00
|
|
|
|
|
|
|
#include <esp_http_server.h>
|
|
|
|
#include <svelte-portal.h>
|
|
|
|
#include <esp_log.h>
|
|
|
|
#include <cJSON.h>
|
2021-11-23 21:59:54 +00:00
|
|
|
#include <esp_wifi.h>
|
2021-11-30 05:27:56 +00:00
|
|
|
#include <esp_system.h>
|
2021-11-22 11:24:16 +00:00
|
|
|
#include "network.h"
|
|
|
|
#include "nvs.h"
|
2021-11-23 21:22:57 +00:00
|
|
|
#include "led.h"
|
2021-11-23 21:59:54 +00:00
|
|
|
#include "helpers.h"
|
2021-11-22 11:24:16 +00:00
|
|
|
|
|
|
|
#define TAG "network-http"
|
|
|
|
#define JSON_ERROR(error_text) "{\"error\": \"" error_text "\"}"
|
|
|
|
#define JSON_RESULT(result_text) "{\"result\": \"" result_text "\"}"
|
|
|
|
|
|
|
|
#define WIFI_SCAN_SIZE 20
|
|
|
|
|
|
|
|
static const char* get_auth_mode(int authmode) {
|
|
|
|
switch(authmode) {
|
|
|
|
case WIFI_AUTH_OPEN:
|
|
|
|
return "OPEN";
|
|
|
|
case WIFI_AUTH_WEP:
|
|
|
|
return "WEP";
|
|
|
|
case WIFI_AUTH_WPA_PSK:
|
2021-12-02 13:43:04 +00:00
|
|
|
return "WPA_PSK";
|
2021-11-22 11:24:16 +00:00
|
|
|
case WIFI_AUTH_WPA2_PSK:
|
2021-12-02 13:43:04 +00:00
|
|
|
return "WPA2_PSK";
|
2021-11-22 11:24:16 +00:00
|
|
|
case WIFI_AUTH_WPA_WPA2_PSK:
|
2021-12-02 13:43:04 +00:00
|
|
|
return "WPA_WPA2_PSK";
|
2021-11-22 11:24:16 +00:00
|
|
|
case WIFI_AUTH_WPA2_ENTERPRISE:
|
2021-12-02 13:43:04 +00:00
|
|
|
return "WPA2_ENTERPRISE";
|
2021-11-22 11:24:16 +00:00
|
|
|
case WIFI_AUTH_WPA3_PSK:
|
2021-12-02 13:43:04 +00:00
|
|
|
return "WPA3_PSK";
|
2021-11-22 11:24:16 +00:00
|
|
|
case WIFI_AUTH_WPA2_WPA3_PSK:
|
2021-12-02 13:43:04 +00:00
|
|
|
return "WPA2_WPA3_PSK";
|
2021-11-22 11:24:16 +00:00
|
|
|
default:
|
|
|
|
return "UNKNOWN";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char* get_pairwise_cipher(int pairwise_cipher) {
|
|
|
|
switch(pairwise_cipher) {
|
|
|
|
case WIFI_CIPHER_TYPE_NONE:
|
|
|
|
return "NONE";
|
|
|
|
case WIFI_CIPHER_TYPE_WEP40:
|
|
|
|
return "WEP40";
|
|
|
|
case WIFI_CIPHER_TYPE_WEP104:
|
|
|
|
return "WEP104";
|
|
|
|
case WIFI_CIPHER_TYPE_TKIP:
|
|
|
|
return "TKIP";
|
|
|
|
case WIFI_CIPHER_TYPE_CCMP:
|
|
|
|
return "CCMP";
|
|
|
|
case WIFI_CIPHER_TYPE_TKIP_CCMP:
|
|
|
|
return "TKIP_CCMP";
|
|
|
|
default:
|
|
|
|
return "UNKNOWN";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char* get_group_cipher(int group_cipher) {
|
|
|
|
switch(group_cipher) {
|
|
|
|
case WIFI_CIPHER_TYPE_NONE:
|
|
|
|
return "NONE";
|
|
|
|
case WIFI_CIPHER_TYPE_WEP40:
|
|
|
|
return "WEP40";
|
|
|
|
case WIFI_CIPHER_TYPE_WEP104:
|
|
|
|
return "WEP104";
|
|
|
|
case WIFI_CIPHER_TYPE_TKIP:
|
|
|
|
return "TKIP";
|
|
|
|
case WIFI_CIPHER_TYPE_CCMP:
|
|
|
|
return "CCMP";
|
|
|
|
case WIFI_CIPHER_TYPE_TKIP_CCMP:
|
|
|
|
return "TKIP_CCMP";
|
|
|
|
default:
|
|
|
|
return "UNKNOWN";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static esp_err_t http_common_get_handler(httpd_req_t* req) {
|
|
|
|
const uint8_t* file_start = NULL;
|
|
|
|
const uint8_t* file_end = NULL;
|
|
|
|
if(strcmp("/", req->uri) == 0 || strcmp("/index.html", req->uri) == 0) {
|
|
|
|
file_start = index_html_start;
|
|
|
|
file_end = index_html_end;
|
|
|
|
httpd_resp_set_type(req, "text/html");
|
|
|
|
} else if(strcmp("/build/bundle.css", req->uri) == 0) {
|
|
|
|
file_start = build_bundle_css_start;
|
|
|
|
file_end = build_bundle_css_end;
|
|
|
|
httpd_resp_set_type(req, "text/css");
|
|
|
|
} else if(strcmp("/build/bundle.js", req->uri) == 0) {
|
|
|
|
file_start = build_bundle_js_start;
|
|
|
|
file_end = build_bundle_js_end;
|
|
|
|
httpd_resp_set_type(req, "application/javascript");
|
|
|
|
} else if(strcmp("/assets/ega8.otf", req->uri) == 0) {
|
|
|
|
file_start = assets_ega8_otf_start;
|
|
|
|
file_end = assets_ega8_otf_end;
|
|
|
|
httpd_resp_set_type(req, "application/x-font-opentype");
|
|
|
|
} else if(strcmp("/assets/favicon.ico", req->uri) == 0 || strcmp("/favicon.ico", req->uri) == 0) {
|
|
|
|
file_start = assets_favicon_ico_start;
|
|
|
|
file_end = assets_favicon_ico_end;
|
|
|
|
httpd_resp_set_type(req, "image/x-icon");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(file_start == NULL) {
|
|
|
|
ESP_LOGE(TAG, "file not exist: %s", req->uri);
|
|
|
|
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, "file not exist");
|
|
|
|
return ESP_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t err =
|
|
|
|
httpd_resp_send_chunk(req, (const char*)file_start, (ssize_t)(file_end - file_start));
|
|
|
|
if(err != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "file sending failed, %d", err);
|
|
|
|
// Abort
|
|
|
|
httpd_resp_sendstr_chunk(req, NULL);
|
|
|
|
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "file sending failed");
|
|
|
|
return ESP_FAIL;
|
|
|
|
} else {
|
|
|
|
// HTTP response completion
|
|
|
|
httpd_resp_send_chunk(req, NULL, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ESP_LOGI(TAG, "file sending complete");
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static esp_err_t system_ping_handler(httpd_req_t* req) {
|
|
|
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
|
|
|
httpd_resp_set_type(req, "application/json");
|
|
|
|
httpd_resp_sendstr(req, JSON_RESULT("OK"));
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
2021-11-23 21:22:57 +00:00
|
|
|
static esp_err_t system_tasks_handler(httpd_req_t* req) {
|
|
|
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
|
|
|
httpd_resp_set_type(req, "application/json");
|
|
|
|
|
|
|
|
uint32_t task_count = uxTaskGetNumberOfTasks();
|
|
|
|
TaskStatus_t* tasks = malloc(task_count * sizeof(TaskStatus_t));
|
|
|
|
task_count = uxTaskGetSystemState(tasks, task_count, NULL);
|
|
|
|
|
|
|
|
cJSON* root = cJSON_CreateObject();
|
|
|
|
cJSON_AddNumberToObject(root, "count", task_count);
|
|
|
|
cJSON* array = cJSON_AddArrayToObject(root, "list");
|
|
|
|
|
|
|
|
for(uint32_t i = 0; i < task_count; i++) {
|
|
|
|
cJSON* object = cJSON_CreateObject();
|
|
|
|
cJSON_AddNumberToObject(object, "handle", (uint32_t)tasks[i].xHandle);
|
|
|
|
cJSON_AddStringToObject(object, "name", tasks[i].pcTaskName);
|
|
|
|
cJSON_AddNumberToObject(object, "number", (uint32_t)tasks[i].xTaskNumber);
|
|
|
|
|
|
|
|
switch(tasks[i].eCurrentState) {
|
|
|
|
case eRunning:
|
|
|
|
cJSON_AddStringToObject(object, "state", "Running");
|
|
|
|
break;
|
|
|
|
case eReady:
|
|
|
|
cJSON_AddStringToObject(object, "state", "Ready");
|
|
|
|
break;
|
|
|
|
case eBlocked:
|
|
|
|
cJSON_AddStringToObject(object, "state", "Blocked");
|
|
|
|
break;
|
|
|
|
case eSuspended:
|
|
|
|
cJSON_AddStringToObject(object, "state", "Suspended");
|
|
|
|
break;
|
|
|
|
case eDeleted:
|
|
|
|
cJSON_AddStringToObject(object, "state", "Deleted");
|
|
|
|
break;
|
|
|
|
case eInvalid:
|
|
|
|
cJSON_AddStringToObject(object, "state", "Invalid");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cJSON_AddNumberToObject(object, "current_priority", (uint32_t)tasks[i].uxCurrentPriority);
|
|
|
|
cJSON_AddNumberToObject(object, "base_priority", (uint32_t)tasks[i].uxBasePriority);
|
|
|
|
cJSON_AddNumberToObject(object, "runtime", (uint32_t)tasks[i].ulRunTimeCounter);
|
|
|
|
cJSON_AddNumberToObject(object, "stack_base", (uint32_t)tasks[i].pxStackBase);
|
|
|
|
cJSON_AddNumberToObject(object, "watermark", (uint32_t)tasks[i].usStackHighWaterMark);
|
|
|
|
|
|
|
|
cJSON_AddItemToArray(array, object);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* json_text = cJSON_Print(root);
|
|
|
|
httpd_resp_sendstr(req, json_text);
|
|
|
|
free((void*)json_text);
|
|
|
|
cJSON_Delete(root);
|
|
|
|
free((void*)tasks);
|
|
|
|
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
2021-11-22 11:24:16 +00:00
|
|
|
static esp_err_t system_info_get_handler(httpd_req_t* req) {
|
|
|
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
|
|
|
httpd_resp_set_type(req, "application/json");
|
|
|
|
cJSON* root = cJSON_CreateObject();
|
|
|
|
esp_chip_info_t chip_info;
|
|
|
|
esp_chip_info(&chip_info);
|
|
|
|
|
2021-11-23 21:22:57 +00:00
|
|
|
cJSON_AddStringToObject(root, "idf_version", IDF_VER);
|
2021-11-22 11:24:16 +00:00
|
|
|
switch(chip_info.model) {
|
|
|
|
case CHIP_ESP32:
|
|
|
|
cJSON_AddStringToObject(root, "model", "ESP32");
|
|
|
|
break;
|
|
|
|
case CHIP_ESP32S2:
|
|
|
|
cJSON_AddStringToObject(root, "model", "ESP32-S2");
|
|
|
|
break;
|
2021-11-22 18:19:40 +00:00
|
|
|
case CHIP_ESP32H2:
|
|
|
|
cJSON_AddStringToObject(root, "model", "ESP32-H2");
|
|
|
|
break;
|
2021-11-22 11:24:16 +00:00
|
|
|
case CHIP_ESP32S3:
|
|
|
|
cJSON_AddStringToObject(root, "model", "ESP32-S3");
|
|
|
|
break;
|
|
|
|
case CHIP_ESP32C3:
|
|
|
|
cJSON_AddStringToObject(root, "model", "ESP32-C3");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cJSON_AddNumberToObject(root, "revision", chip_info.revision);
|
|
|
|
cJSON_AddNumberToObject(root, "cores", chip_info.cores);
|
|
|
|
|
|
|
|
multi_heap_info_t info;
|
|
|
|
heap_caps_get_info(&info, MALLOC_CAP_8BIT);
|
|
|
|
cJSON* heap = cJSON_AddObjectToObject(root, "heap");
|
|
|
|
cJSON_AddNumberToObject(heap, "total_free_bytes", info.total_free_bytes);
|
|
|
|
cJSON_AddNumberToObject(heap, "total_allocated_bytes", info.total_allocated_bytes);
|
|
|
|
cJSON_AddNumberToObject(heap, "largest_free_block", info.largest_free_block);
|
|
|
|
cJSON_AddNumberToObject(heap, "minimum_free_bytes", info.minimum_free_bytes);
|
|
|
|
|
2021-11-30 05:27:56 +00:00
|
|
|
// ip addr
|
|
|
|
cJSON_AddNumberToObject(root, "ip", network_get_ip());
|
|
|
|
|
|
|
|
// mac
|
|
|
|
uint8_t mac_addr[6] = {0};
|
|
|
|
ESP_ERROR_CHECK(esp_read_mac(mac_addr, ESP_MAC_WIFI_STA));
|
|
|
|
|
|
|
|
cJSON* mac = cJSON_AddArrayToObject(root, "mac");
|
|
|
|
cJSON_AddItemToArray(mac, cJSON_CreateNumber(mac_addr[0]));
|
|
|
|
cJSON_AddItemToArray(mac, cJSON_CreateNumber(mac_addr[1]));
|
|
|
|
cJSON_AddItemToArray(mac, cJSON_CreateNumber(mac_addr[2]));
|
|
|
|
cJSON_AddItemToArray(mac, cJSON_CreateNumber(mac_addr[3]));
|
|
|
|
cJSON_AddItemToArray(mac, cJSON_CreateNumber(mac_addr[4]));
|
|
|
|
cJSON_AddItemToArray(mac, cJSON_CreateNumber(mac_addr[5]));
|
|
|
|
|
2021-11-22 11:24:16 +00:00
|
|
|
const char* json_text = cJSON_Print(root);
|
|
|
|
httpd_resp_sendstr(req, json_text);
|
|
|
|
free((void*)json_text);
|
|
|
|
cJSON_Delete(root);
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static esp_err_t wifi_mode_get_handler(httpd_req_t* req) {
|
|
|
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
|
|
|
httpd_resp_set_type(req, "application/json");
|
|
|
|
cJSON* root = cJSON_CreateObject();
|
|
|
|
|
|
|
|
switch(network_get_mode()) {
|
|
|
|
case WIFIModeSTA:
|
|
|
|
cJSON_AddStringToObject(root, "mode", "STA");
|
|
|
|
break;
|
|
|
|
case WIFIModeAP:
|
|
|
|
cJSON_AddStringToObject(root, "mode", "AP");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* json_text = cJSON_Print(root);
|
|
|
|
httpd_resp_sendstr(req, json_text);
|
|
|
|
free((void*)json_text);
|
|
|
|
cJSON_Delete(root);
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static esp_err_t wifi_get_credenitals_handler(httpd_req_t* req) {
|
|
|
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
|
|
|
httpd_resp_set_type(req, "application/json");
|
|
|
|
cJSON* root = cJSON_CreateObject();
|
|
|
|
|
|
|
|
mstring_t* ap_ssid = mstring_alloc();
|
|
|
|
mstring_t* ap_pass = mstring_alloc();
|
|
|
|
mstring_t* ap_mode = mstring_alloc();
|
|
|
|
|
|
|
|
if(nvs_load_string("ap_ssid", ap_ssid) != ESP_OK) {
|
|
|
|
mstring_set(ap_ssid, "");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(nvs_load_string("ap_pass", ap_pass) != ESP_OK) {
|
|
|
|
mstring_set(ap_pass, "");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(nvs_load_string("ap_mode", ap_mode) != ESP_OK) {
|
|
|
|
mstring_set(ap_mode, "");
|
|
|
|
}
|
|
|
|
|
|
|
|
cJSON_AddStringToObject(root, "ssid", mstring_get_cstr(ap_ssid));
|
|
|
|
cJSON_AddStringToObject(root, "pass", mstring_get_cstr(ap_pass));
|
|
|
|
cJSON_AddStringToObject(root, "mode", mstring_get_cstr(ap_mode));
|
|
|
|
|
|
|
|
const char* json_text = cJSON_Print(root);
|
|
|
|
httpd_resp_sendstr(req, json_text);
|
|
|
|
free((void*)json_text);
|
|
|
|
cJSON_Delete(root);
|
|
|
|
mstring_free(ap_ssid);
|
|
|
|
mstring_free(ap_pass);
|
|
|
|
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static esp_err_t wifi_set_credenitals_handler(httpd_req_t* req) {
|
|
|
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
|
|
|
int total_length = req->content_len;
|
|
|
|
int cur_len = 0;
|
|
|
|
char* buffer = malloc(256);
|
|
|
|
mstring_t* ap_ssid = mstring_alloc();
|
|
|
|
mstring_t* ap_pass = mstring_alloc();
|
|
|
|
mstring_t* ap_mode = mstring_alloc();
|
|
|
|
const char* error_text = JSON_ERROR("unknown error");
|
|
|
|
int received = 0;
|
|
|
|
httpd_resp_set_type(req, "application/json");
|
|
|
|
|
|
|
|
if(total_length >= 256) {
|
|
|
|
error_text = JSON_ERROR("request too long");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
while(cur_len < total_length) {
|
|
|
|
received = httpd_req_recv(req, buffer + cur_len, total_length);
|
|
|
|
if(received <= 0) {
|
|
|
|
error_text = JSON_ERROR("cannot receive request data");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
cur_len += received;
|
|
|
|
}
|
|
|
|
buffer[total_length] = '\0';
|
|
|
|
|
|
|
|
cJSON* root = cJSON_Parse(buffer);
|
|
|
|
if(cJSON_GetObjectItem(root, "ssid") != NULL) {
|
|
|
|
mstring_set(ap_ssid, cJSON_GetObjectItem(root, "ssid")->valuestring);
|
|
|
|
} else {
|
|
|
|
cJSON_Delete(root);
|
|
|
|
error_text = JSON_ERROR("request dont have [ssid] field");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(cJSON_GetObjectItem(root, "pass") != NULL) {
|
|
|
|
mstring_set(ap_pass, cJSON_GetObjectItem(root, "pass")->valuestring);
|
|
|
|
} else {
|
|
|
|
cJSON_Delete(root);
|
|
|
|
error_text = JSON_ERROR("request dont have [pass] field");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(cJSON_GetObjectItem(root, "mode") != NULL) {
|
|
|
|
mstring_set(ap_mode, cJSON_GetObjectItem(root, "mode")->valuestring);
|
|
|
|
} else {
|
|
|
|
cJSON_Delete(root);
|
|
|
|
error_text = JSON_ERROR("request dont have [mode] field");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
cJSON_Delete(root);
|
|
|
|
|
|
|
|
if(strcmp(mstring_get_cstr(ap_mode), ESP_WIFI_MODE_AP) != 0 &&
|
|
|
|
strcmp(mstring_get_cstr(ap_mode), ESP_WIFI_MODE_STA) != 0) {
|
|
|
|
error_text = JSON_ERROR("invalid value in [mode]");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
|
2021-12-01 04:15:45 +00:00
|
|
|
if(mstring_size(ap_pass) < 8) {
|
2021-11-22 11:24:16 +00:00
|
|
|
error_text = JSON_ERROR("too short value in [pass]");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
2021-12-01 04:15:45 +00:00
|
|
|
if(mstring_size(ap_ssid) < 1) {
|
2021-11-22 11:24:16 +00:00
|
|
|
error_text = JSON_ERROR("too short value in [ssid]");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(nvs_save_string("ap_ssid", ap_ssid) != ESP_OK) {
|
|
|
|
error_text = JSON_ERROR("cannot save [ap_ssid]");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
if(nvs_save_string("ap_pass", ap_pass) != ESP_OK) {
|
|
|
|
error_text = JSON_ERROR("cannot save [ap_pass]");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
if(nvs_save_string("ap_mode", ap_mode) != ESP_OK) {
|
|
|
|
error_text = JSON_ERROR("cannot save [ap_mode]");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
httpd_resp_sendstr(req, JSON_RESULT("WIFI settings saved"));
|
|
|
|
return ESP_OK;
|
|
|
|
|
|
|
|
err_fail:
|
|
|
|
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, error_text);
|
|
|
|
free(buffer);
|
|
|
|
mstring_free(ap_ssid);
|
|
|
|
mstring_free(ap_pass);
|
|
|
|
mstring_free(ap_mode);
|
|
|
|
return ESP_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static esp_err_t wifi_list_get_handler(httpd_req_t* req) {
|
|
|
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
|
|
|
uint16_t number = WIFI_SCAN_SIZE;
|
|
|
|
wifi_ap_record_t* ap_info = calloc(WIFI_SCAN_SIZE, sizeof(wifi_ap_record_t));
|
|
|
|
uint16_t ap_count = 0;
|
|
|
|
memset(ap_info, 0, sizeof(wifi_ap_record_t));
|
|
|
|
|
|
|
|
esp_wifi_scan_start(NULL, true);
|
|
|
|
ESP_ERROR_CHECK(esp_wifi_scan_get_ap_records(&number, ap_info));
|
|
|
|
ESP_ERROR_CHECK(esp_wifi_scan_get_ap_num(&ap_count));
|
|
|
|
|
|
|
|
httpd_resp_set_type(req, "application/json");
|
|
|
|
cJSON* root = cJSON_CreateObject();
|
|
|
|
cJSON* array = cJSON_AddArrayToObject(root, "net_list");
|
|
|
|
|
|
|
|
for(int i = 0; (i < WIFI_SCAN_SIZE) && (i < ap_count); i++) {
|
|
|
|
cJSON* object = cJSON_CreateObject();
|
|
|
|
cJSON_AddStringToObject(object, "ssid", (const char*)&ap_info[i].ssid);
|
|
|
|
cJSON_AddNumberToObject(object, "channel", ap_info[i].primary);
|
|
|
|
cJSON_AddNumberToObject(object, "rssi", ap_info[i].rssi);
|
|
|
|
cJSON_AddStringToObject(object, "auth", get_auth_mode(ap_info[i].authmode));
|
|
|
|
if(ap_info[i].authmode != WIFI_AUTH_WEP) {
|
|
|
|
cJSON_AddStringToObject(
|
|
|
|
object, "pairwise_cipher", get_pairwise_cipher(ap_info[i].pairwise_cipher));
|
|
|
|
cJSON_AddStringToObject(
|
|
|
|
object, "group_cipher", get_group_cipher(ap_info[i].group_cipher));
|
|
|
|
}
|
|
|
|
cJSON_AddItemToArray(array, object);
|
|
|
|
}
|
|
|
|
cJSON_AddNumberToObject(root, "total", ap_count);
|
|
|
|
const char* json_text = cJSON_Print(root);
|
|
|
|
httpd_resp_sendstr(req, json_text);
|
|
|
|
free((void*)json_text);
|
|
|
|
|
|
|
|
cJSON_Delete(root);
|
|
|
|
free(ap_info);
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
2021-11-23 21:22:57 +00:00
|
|
|
static esp_err_t gpio_led_set_handler(httpd_req_t* req) {
|
|
|
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
|
|
|
int total_length = req->content_len;
|
|
|
|
int current_length = 0;
|
|
|
|
char* buffer = malloc(256);
|
|
|
|
const char* error_text = JSON_ERROR("unknown error");
|
|
|
|
int received = 0;
|
|
|
|
int32_t led_red = -1;
|
|
|
|
int32_t led_green = -1;
|
|
|
|
int32_t led_blue = -1;
|
|
|
|
httpd_resp_set_type(req, "application/json");
|
|
|
|
|
|
|
|
if(total_length >= 256) {
|
|
|
|
error_text = JSON_ERROR("request too long");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
while(current_length < total_length) {
|
|
|
|
received = httpd_req_recv(req, buffer + current_length, total_length);
|
|
|
|
if(received <= 0) {
|
|
|
|
error_text = JSON_ERROR("cannot receive request data");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
current_length += received;
|
|
|
|
}
|
|
|
|
buffer[total_length] = '\0';
|
|
|
|
|
|
|
|
cJSON* root = cJSON_Parse(buffer);
|
|
|
|
cJSON* element;
|
|
|
|
element = cJSON_GetObjectItem(root, "red");
|
|
|
|
if(element != NULL && element->type == cJSON_Number) {
|
|
|
|
led_red = element->valuedouble;
|
|
|
|
}
|
|
|
|
|
|
|
|
element = cJSON_GetObjectItem(root, "green");
|
|
|
|
if(element != NULL && element->type == cJSON_Number) {
|
|
|
|
led_green = element->valuedouble;
|
|
|
|
}
|
|
|
|
|
|
|
|
element = cJSON_GetObjectItem(root, "blue");
|
|
|
|
if(element != NULL && element->type == cJSON_Number) {
|
|
|
|
led_blue = element->valuedouble;
|
|
|
|
}
|
|
|
|
cJSON_Delete(root);
|
|
|
|
|
|
|
|
if((led_red < 0 && led_green < 0 && led_blue < 0) ||
|
|
|
|
(led_red > 255 || led_green > 255 || led_blue > 255)) {
|
|
|
|
error_text = JSON_ERROR("[red], [green], [blue] must contain numbers in the range 0-255");
|
|
|
|
goto err_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(led_red >= 0) {
|
|
|
|
led_set_red(led_red);
|
|
|
|
}
|
|
|
|
if(led_green >= 0) {
|
|
|
|
led_set_green(led_green);
|
|
|
|
}
|
|
|
|
if(led_blue >= 0) {
|
|
|
|
led_set_blue(led_blue);
|
|
|
|
}
|
|
|
|
|
|
|
|
httpd_resp_sendstr(req, JSON_RESULT("OK"));
|
|
|
|
return ESP_OK;
|
|
|
|
|
|
|
|
err_fail:
|
|
|
|
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, error_text);
|
|
|
|
free(buffer);
|
|
|
|
return ESP_FAIL;
|
|
|
|
}
|
|
|
|
|
2021-11-23 21:59:54 +00:00
|
|
|
const httpd_uri_t uri_handlers[] = {
|
|
|
|
|
|
|
|
/*************** SYSTEM ***************/
|
|
|
|
|
|
|
|
{.uri = "/api/v1/system/ping",
|
|
|
|
.method = HTTP_GET,
|
|
|
|
.handler = system_ping_handler,
|
|
|
|
.user_ctx = NULL},
|
|
|
|
|
|
|
|
{.uri = "/api/v1/system/tasks",
|
|
|
|
.method = HTTP_GET,
|
|
|
|
.handler = system_tasks_handler,
|
|
|
|
.user_ctx = NULL},
|
|
|
|
|
|
|
|
{.uri = "/api/v1/system/info",
|
|
|
|
.method = HTTP_GET,
|
|
|
|
.handler = system_info_get_handler,
|
|
|
|
.user_ctx = NULL},
|
|
|
|
|
|
|
|
/*************** GPIO ***************/
|
|
|
|
|
|
|
|
{.uri = "/api/v1/gpio/led",
|
|
|
|
.method = HTTP_POST,
|
|
|
|
.handler = gpio_led_set_handler,
|
|
|
|
.user_ctx = NULL},
|
|
|
|
|
|
|
|
/*************** WIFI ***************/
|
|
|
|
|
|
|
|
{.uri = "/api/v1/wifi/list",
|
|
|
|
.method = HTTP_GET,
|
|
|
|
.handler = wifi_list_get_handler,
|
|
|
|
.user_ctx = NULL},
|
|
|
|
|
|
|
|
{.uri = "/api/v1/wifi/mode",
|
|
|
|
.method = HTTP_GET,
|
|
|
|
.handler = wifi_mode_get_handler,
|
|
|
|
.user_ctx = NULL},
|
|
|
|
|
|
|
|
{.uri = "/api/v1/wifi/set_credenitals",
|
|
|
|
.method = HTTP_POST,
|
|
|
|
.handler = wifi_set_credenitals_handler,
|
|
|
|
.user_ctx = NULL},
|
|
|
|
|
|
|
|
{.uri = "/api/v1/wifi/get_credenitals",
|
|
|
|
.method = HTTP_GET,
|
|
|
|
.handler = wifi_get_credenitals_handler,
|
|
|
|
.user_ctx = NULL},
|
|
|
|
|
|
|
|
/*************** HTTP ***************/
|
|
|
|
|
|
|
|
{.uri = "/*", .method = HTTP_GET, .handler = http_common_get_handler, .user_ctx = NULL},
|
|
|
|
};
|
|
|
|
|
2021-11-22 11:24:16 +00:00
|
|
|
void network_http_server_init(void) {
|
|
|
|
ESP_LOGI(TAG, "init rest server");
|
|
|
|
|
|
|
|
httpd_handle_t server = NULL;
|
|
|
|
httpd_config_t config = HTTPD_DEFAULT_CONFIG();
|
2021-11-23 21:59:54 +00:00
|
|
|
config.max_uri_handlers = COUNT_OF(uri_handlers);
|
2021-11-22 11:24:16 +00:00
|
|
|
config.uri_match_fn = httpd_uri_match_wildcard;
|
|
|
|
|
|
|
|
ESP_LOGI(TAG, "starting http server");
|
|
|
|
if(httpd_start(&server, &config) != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "start http server failed");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-11-23 21:59:54 +00:00
|
|
|
for(size_t i = 0; i < COUNT_OF(uri_handlers); i++) {
|
|
|
|
ESP_ERROR_CHECK(httpd_register_uri_handler(server, &uri_handlers[i]));
|
|
|
|
}
|
2021-11-22 11:24:16 +00:00
|
|
|
|
|
|
|
ESP_LOGI(TAG, "init rest server done");
|
|
|
|
}
|