| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 |
- /* Persistent Sockets Example
- This example code is in the Public Domain (or CC0 licensed, at your option.)
- Unless required by applicable law or agreed to in writing, this
- software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
- CONDITIONS OF ANY KIND, either express or implied.
- */
- #include <esp_wifi.h>
- #include <esp_event_loop.h>
- #include <esp_log.h>
- #include <esp_system.h>
- #include <nvs_flash.h>
- #include "tcpip_adapter.h"
- #include "esp_eth.h"
- #include "protocol_examples_common.h"
- #include <esp_http_server.h>
- /* An example to demonstrate persistent sockets, with context maintained across
- * multiple requests on that socket.
- */
- static const char *TAG = "example";
-
- /* Function to free context */
- static void adder_free_func(void *ctx)
- {
- ESP_LOGI(TAG, "/adder Free Context function called");
- free(ctx);
- }
- /* This handler keeps accumulating data that is posted to it into a per
- * socket/session context. And returns the result.
- */
- static esp_err_t adder_post_handler(httpd_req_t *req)
- {
- /* Log total visitors */
- unsigned *visitors = (unsigned *)req->user_ctx;
- ESP_LOGI(TAG, "/adder visitor count = %d", ++(*visitors));
- char buf[10];
- char outbuf[50];
- int ret;
- /* Read data received in the request */
- ret = httpd_req_recv(req, buf, sizeof(buf));
- if (ret <= 0) {
- if (ret == HTTPD_SOCK_ERR_TIMEOUT) {
- httpd_resp_send_408(req);
- }
- return ESP_FAIL;
- }
- buf[ret] = '\0';
- int val = atoi(buf);
- ESP_LOGI(TAG, "/adder handler read %d", val);
- /* Create session's context if not already available */
- if (! req->sess_ctx) {
- ESP_LOGI(TAG, "/adder allocating new session");
- req->sess_ctx = malloc(sizeof(int));
- req->free_ctx = adder_free_func;
- *(int *)req->sess_ctx = 0;
- }
- /* Add the received data to the context */
- int *adder = (int *)req->sess_ctx;
- *adder += val;
- /* Respond with the accumulated value */
- snprintf(outbuf, sizeof(outbuf),"%d", *adder);
- httpd_resp_send(req, outbuf, strlen(outbuf));
- return ESP_OK;
- }
- /* This handler gets the present value of the accumulator */
- static esp_err_t adder_get_handler(httpd_req_t *req)
- {
- /* Log total visitors */
- unsigned *visitors = (unsigned *)req->user_ctx;
- ESP_LOGI(TAG, "/adder visitor count = %d", ++(*visitors));
- char outbuf[50];
- /* Create session's context if not already available */
- if (! req->sess_ctx) {
- ESP_LOGI(TAG, "/adder GET allocating new session");
- req->sess_ctx = malloc(sizeof(int));
- req->free_ctx = adder_free_func;
- *(int *)req->sess_ctx = 0;
- }
- ESP_LOGI(TAG, "/adder GET handler send %d", *(int *)req->sess_ctx);
- /* Respond with the accumulated value */
- snprintf(outbuf, sizeof(outbuf),"%d", *((int *)req->sess_ctx));
- httpd_resp_send(req, outbuf, strlen(outbuf));
- return ESP_OK;
- }
- /* This handler resets the value of the accumulator */
- static esp_err_t adder_put_handler(httpd_req_t *req)
- {
- /* Log total visitors */
- unsigned *visitors = (unsigned *)req->user_ctx;
- ESP_LOGI(TAG, "/adder visitor count = %d", ++(*visitors));
- char buf[10];
- char outbuf[50];
- int ret;
- /* Read data received in the request */
- ret = httpd_req_recv(req, buf, sizeof(buf));
- if (ret <= 0) {
- if (ret == HTTPD_SOCK_ERR_TIMEOUT) {
- httpd_resp_send_408(req);
- }
- return ESP_FAIL;
- }
- buf[ret] = '\0';
- int val = atoi(buf);
- ESP_LOGI(TAG, "/adder PUT handler read %d", val);
- /* Create session's context if not already available */
- if (! req->sess_ctx) {
- ESP_LOGI(TAG, "/adder PUT allocating new session");
- req->sess_ctx = malloc(sizeof(int));
- req->free_ctx = adder_free_func;
- }
- *(int *)req->sess_ctx = val;
- /* Respond with the reset value */
- snprintf(outbuf, sizeof(outbuf),"%d", *((int *)req->sess_ctx));
- httpd_resp_send(req, outbuf, strlen(outbuf));
- return ESP_OK;
- }
- /* Maintain a variable which stores the number of times
- * the "/adder" URI has been visited */
- static unsigned visitors = 0;
- static const httpd_uri_t adder_post = {
- .uri = "/adder",
- .method = HTTP_POST,
- .handler = adder_post_handler,
- .user_ctx = &visitors
- };
- static const httpd_uri_t adder_get = {
- .uri = "/adder",
- .method = HTTP_GET,
- .handler = adder_get_handler,
- .user_ctx = &visitors
- };
- static const httpd_uri_t adder_put = {
- .uri = "/adder",
- .method = HTTP_PUT,
- .handler = adder_put_handler,
- .user_ctx = &visitors
- };
- static httpd_handle_t start_webserver(void)
- {
- httpd_config_t config = HTTPD_DEFAULT_CONFIG();
- // Start the httpd server
- ESP_LOGI(TAG, "Starting server on port: '%d'", config.server_port);
- httpd_handle_t server;
- if (httpd_start(&server, &config) == ESP_OK) {
- // Set URI handlers
- ESP_LOGI(TAG, "Registering URI handlers");
- httpd_register_uri_handler(server, &adder_get);
- httpd_register_uri_handler(server, &adder_put);
- httpd_register_uri_handler(server, &adder_post);
- return server;
- }
- ESP_LOGI(TAG, "Error starting server!");
- return NULL;
- }
- static void stop_webserver(httpd_handle_t server)
- {
- // Stop the httpd server
- httpd_stop(server);
- }
- static void disconnect_handler(void* arg, esp_event_base_t event_base,
- int32_t event_id, void* event_data)
- {
- httpd_handle_t* server = (httpd_handle_t*) arg;
- if (*server) {
- ESP_LOGI(TAG, "Stopping webserver");
- stop_webserver(*server);
- *server = NULL;
- }
- }
- static void connect_handler(void* arg, esp_event_base_t event_base,
- int32_t event_id, void* event_data)
- {
- httpd_handle_t* server = (httpd_handle_t*) arg;
- if (*server == NULL) {
- ESP_LOGI(TAG, "Starting webserver");
- *server = start_webserver();
- }
- }
- void app_main()
- {
- static httpd_handle_t server = NULL;
- ESP_ERROR_CHECK(nvs_flash_init());
- tcpip_adapter_init();
- ESP_ERROR_CHECK(esp_event_loop_create_default());
- /* This helper function configures Wi-Fi or Ethernet, as selected in menuconfig.
- * Read "Establishing Wi-Fi or Ethernet Connection" section in
- * examples/protocols/README.md for more information about this function.
- */
- ESP_ERROR_CHECK(example_connect());
- /* Register event handlers to stop the server when Wi-Fi or Ethernet is disconnected,
- * and re-start it upon connection.
- */
- #ifdef CONFIG_EXAMPLE_CONNECT_WIFI
- ESP_ERROR_CHECK(esp_event_handler_register(IP_EVENT, IP_EVENT_STA_GOT_IP, &connect_handler, &server));
- ESP_ERROR_CHECK(esp_event_handler_register(WIFI_EVENT, WIFI_EVENT_STA_DISCONNECTED, &disconnect_handler, &server));
- #endif // CONFIG_EXAMPLE_CONNECT_WIFI
- #ifdef CONFIG_EXAMPLE_CONNECT_ETHERNET
- ESP_ERROR_CHECK(esp_event_handler_register(IP_EVENT, IP_EVENT_ETH_GOT_IP, &connect_handler, &server));
- ESP_ERROR_CHECK(esp_event_handler_register(ETH_EVENT, ETHERNET_EVENT_DISCONNECTED, &disconnect_handler, &server));
- #endif // CONFIG_EXAMPLE_CONNECT_ETHERNET
- /* Start the server for the first time */
- server = start_webserver();
- }
|