/** ****************************************************************************** * @file app_https.c * @author QQ DING * @version V1.0.0 * @date 1-September-2015 * @brief The main HTTPD server initialization and wsgi handle. ****************************************************************************** * * The MIT License * Copyright (c) 2016 MXCHIP Inc. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is furnished * to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR * IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * ****************************************************************************** */ #include #include #include #include #include "mico.h" #include "httpd_priv.h" #include "app_httpd.h" #include "user_gpio.h" #include "user_wifi.h" #include "user_power.h" #include "main.h" #include "web_data.c" #include "web_log.h" #include "timed_task/timed_task.h" #include "ota_server/user_ota.h" static bool is_http_init; static bool is_handlers_registered; struct httpd_wsgi_call g_app_handlers[]; char power_info_json[1536] = { 0 }; char up_time[16] = "00:00:00"; /* void GetPraFromUrl(char* url, char* pra, char* val) { char* sub = strstr(url, pra); if (sub == NULL) { val[0] = 0; return; } sub = strstr(sub, "="); if (sub == NULL) { val[0] = 0; return; } int len = strlen(sub); int n = 0; for (int i = 0; i < len; i++) { if (sub[i] == '&' || i == len - 1) { n = len; break; } } if (n > 0) { strncpy(val, sub + 1, n - 1); val[n - 1] = 0; return; } val[0] = 0; } */ static int HttpGetIndexPage(httpd_request_t *req) { OSStatus err = kNoErr; err = httpd_send_all_header(req, HTTP_RES_200, sizeof(index_html), HTTP_CONTENT_HTML_ZIP); require_noerr_action(err, exit, app_httpd_log("ERROR: Unable to send http wifisetting headers.")); err = httpd_send_body(req->sock, index_html, sizeof(index_html)); require_noerr_action(err, exit, app_httpd_log("ERROR: Unable to send http wifisetting body.")); exit: return err; } static int HttpGetTc1Status(httpd_request_t *req) { const unsigned char* sockets = GetSocketStatus(); char* tc1_status = malloc(412); sprintf(tc1_status, TC1_STATUS_JSON, sockets, ip_status.mode, sys_config->micoSystemConfig.ssid, sys_config->micoSystemConfig.user_key, ap_name, ap_key, "MQTT.ADDR", 1883, VERSION, ip_status.ip, ip_status.mask, ip_status.gateway, 0L); OSStatus err = kNoErr; send_http(tc1_status, strlen(tc1_status), exit, &err); exit: if (tc1_status) free(tc1_status); return err; } static int HttpSetSocketStatus(httpd_request_t *req) { OSStatus err = kNoErr; int buf_size = 512; char *buf = malloc(buf_size); err = httpd_get_data(req, buf, buf_size); require_noerr(err, exit); SetSocketStatus(buf); send_http("OK", 2, exit, &err); exit: if (buf) free(buf); return err; } static int HttpGetPowerInfo(httpd_request_t *req) { OSStatus err = kNoErr; char buf[16]; err = httpd_get_data(req, buf, 16); require_noerr(err, exit); int idx = 0; sscanf(buf, "%d", &idx); //计算系统运行时间 mico_time_t past_ms = 0; mico_time_get_time(&past_ms); int past = past_ms / 1000; int h = past / 3600; int m = past / 60 % 60; int s = past % 60; sprintf(up_time, "%d:%02d:%02d", h, m, s); char* powers = GetPowerRecord(idx); sprintf(power_info_json, POWER_INFO_JSON, power_record.idx, PW_NUM, p_count, powers, up_time); send_http(power_info_json, strlen(power_info_json), exit, &err); exit: return err; } static int HttpGetWifiConfig(httpd_request_t *req) { OSStatus err = kNoErr; const unsigned char* status = GetSocketStatus(); send_http(status, strlen((char*)status), exit, &err); exit: return err; } static int HttpSetWifiConfig(httpd_request_t *req) { OSStatus err = kNoErr; int buf_size = 97; char *buf = malloc(buf_size); int mode = -1; char *wifi_ssid = malloc(32); char *wifi_key = malloc(32); err = httpd_get_data(req, buf, buf_size); require_noerr(err, exit); sscanf(buf, "%d %s %s", &mode, wifi_ssid, wifi_key); if (mode == 1) { WifiConnect(wifi_ssid, wifi_key); } else { ApConfig(wifi_ssid, wifi_key); } send_http("OK", 2, exit, &err); exit: if (buf) free(buf); if (wifi_ssid) free(wifi_ssid); if (wifi_key) free(wifi_key); return err; } static int HttpGetWifiScan(httpd_request_t *req) { OSStatus err = kNoErr; if (scaned) { scaned = false; send_http(wifi_ret, strlen(wifi_ret), exit, &err); free(wifi_ret); } else { send_http("NO", 2, exit, &err); } exit: return err; } static int HttpSetWifiScan(httpd_request_t *req) { micoWlanStartScanAdv(); OSStatus err = kNoErr; send_http("OK", 2, exit, &err); exit: return err; } static int HttpGetLog(httpd_request_t *req) { OSStatus err = kNoErr; char* logs = GetLogRecord(); send_http(logs, strlen(logs), exit, &err); exit: return err; } static int HttpGetTasks(httpd_request_t *req) { OSStatus err = kNoErr; char* tasks_str = GetTaskStr(); send_http(tasks_str, strlen(tasks_str), exit, &err); exit: if (tasks_str) free(tasks_str); return err; } static int HttpAddTask(httpd_request_t *req) { OSStatus err = kNoErr; //1577369623 4 0 char buf[16] = { 0 }; err = httpd_get_data(req, buf, 16); require_noerr(err, exit); pTimedTask task = (pTimedTask)malloc(sizeof(struct TimedTask)); int re = sscanf(buf, "%ld %d %d", &task->prs_time, &task->socket_idx, &task->on); app_httpd_log("AddTask buf[%s] re[%d] (%ld %d %d)", buf, re, task->prs_time, task->socket_idx, task->on); if (task->prs_time < 1577428136 || task->prs_time > 9577428136 || task->socket_idx < 0 || task->socket_idx > 5 || (task->on != 0 && task->on != 1)) { app_httpd_log("AddTask Error!"); re = 0; } char* mess = re == 3 && AddTask(task) ? "OK" : "NO"; send_http(mess, strlen(mess), exit, &err); exit: return err; } static int HttpDelTask(httpd_request_t *req) { //TODO 从url获取时间 char* time_str = strstr(req->filename, "?time="); app_httpd_log("HttpDelTask url[%s] time_str[%s][%s]", req->filename, time_str, time_str + 6); int time1; sscanf(time_str + 6, "%d", &time1); char* mess = DelTask(time1) ? "OK" : "NO"; OSStatus err = kNoErr; send_http(mess, strlen(mess), exit, &err); exit: return err; } static int Otastatus(httpd_request_t *req) { OSStatus err = kNoErr; char buf[16] = { 0 }; sprintf(buf, "%.2f", ota_progress); send_http(buf, strlen(buf), exit, &err); exit: return err; } static int OtaStart(httpd_request_t *req) { OSStatus err = kNoErr; char buf[64] = { 0 }; err = httpd_get_data(req, buf, 64); require_noerr(err, exit); app_httpd_log("OtaStart ota_url[%s]", buf); user_ota_start(buf, NULL); send_http("OK", 2, exit, &err); exit: return err; } struct httpd_wsgi_call g_app_handlers[] = { { "/", HTTPD_HDR_DEFORT, 0, HttpGetIndexPage, NULL, NULL, NULL }, { "/socket", HTTPD_HDR_DEFORT, 0, NULL, HttpSetSocketStatus, NULL, NULL }, { "/status", HTTPD_HDR_DEFORT, 0, HttpGetTc1Status, NULL, NULL, NULL }, { "/power", HTTPD_HDR_DEFORT, 0, NULL, HttpGetPowerInfo, NULL, NULL }, { "/wifi/config", HTTPD_HDR_DEFORT, 0, HttpGetWifiConfig, HttpSetWifiConfig, NULL, NULL }, { "/wifi/scan", HTTPD_HDR_DEFORT, 0, HttpGetWifiScan, HttpSetWifiScan, NULL, NULL }, { "/log", HTTPD_HDR_DEFORT, 0, HttpGetLog, NULL, NULL, NULL }, { "/task", HTTPD_HDR_DEFORT, 0, HttpGetTasks, HttpAddTask, NULL, HttpDelTask }, { "/ota", HTTPD_HDR_DEFORT, 0, Otastatus, OtaStart, NULL, NULL }, }; static int g_app_handlers_no = sizeof(g_app_handlers)/sizeof(struct httpd_wsgi_call); static void AppHttpRegisterHandlers() { int rc; rc = httpd_register_wsgi_handlers(g_app_handlers, g_app_handlers_no); if (rc) { app_httpd_log("failed to register test web handler"); } } static int _AppHttpdStart() { OSStatus err = kNoErr; app_httpd_log("initializing web-services"); /*Initialize HTTPD*/ if(is_http_init == false) { err = httpd_init(); require_noerr_action(err, exit, app_httpd_log("failed to initialize httpd")); is_http_init = true; } /*Start http thread*/ err = httpd_start(); if(err != kNoErr) { app_httpd_log("failed to start httpd thread"); httpd_shutdown(); } exit: return err; } int AppHttpdStart(void) { OSStatus err = kNoErr; err = _AppHttpdStart(); require_noerr(err, exit); if (is_handlers_registered == false) { AppHttpRegisterHandlers(); is_handlers_registered = true; } exit: return err; } int AppHttpdStop() { OSStatus err = kNoErr; /* HTTPD and services */ app_httpd_log("stopping down httpd"); err = httpd_stop(); require_noerr_action(err, exit, app_httpd_log("failed to halt httpd")); exit: return err; }