From 11c17ab5b622db6c7c0014db63e028f4a220b14f Mon Sep 17 00:00:00 2001 From: suda-morris <362953310@qq.com> Date: Fri, 10 May 2019 13:21:14 +0800 Subject: [PATCH] add RESTful API server example --- .../http_server/restful_server/CMakeLists.txt | 7 + .../http_server/restful_server/Makefile | 14 ++ .../http_server/restful_server/README.md | 135 +++++++++++ .../front/web-demo/.browserslistrc | 3 + .../front/web-demo/.editorconfig | 5 + .../front/web-demo/.eslintrc.js | 17 ++ .../restful_server/front/web-demo/.gitignore | 26 ++ .../front/web-demo/babel.config.js | 5 + .../front/web-demo/package.json | 32 +++ .../front/web-demo/postcss.config.js | 5 + .../front/web-demo/public/favicon.ico | Bin 0 -> 6429 bytes .../front/web-demo/public/index.html | 19 ++ .../restful_server/front/web-demo/src/App.vue | 55 +++++ .../front/web-demo/src/assets/logo.png | Bin 0 -> 37327 bytes .../restful_server/front/web-demo/src/main.js | 16 ++ .../front/web-demo/src/plugins/vuetify.js | 7 + .../front/web-demo/src/router.js | 29 +++ .../front/web-demo/src/store.js | 28 +++ .../front/web-demo/src/views/Chart.vue | 41 ++++ .../front/web-demo/src/views/Home.vue | 40 ++++ .../front/web-demo/src/views/Light.vue | 63 +++++ .../front/web-demo/vue.config.js | 11 + .../restful_server/main/CMakeLists.txt | 13 + .../restful_server/main/Kconfig.projbuild | 50 ++++ .../restful_server/main/component.mk | 0 .../restful_server/main/esp_rest_main.c | 134 +++++++++++ .../restful_server/main/rest_server.c | 225 ++++++++++++++++++ .../restful_server/partitions_example.csv | 6 + .../restful_server/sdkconfig.defaults | 9 + 29 files changed, 995 insertions(+) create mode 100644 examples/protocols/http_server/restful_server/CMakeLists.txt create mode 100644 examples/protocols/http_server/restful_server/Makefile create mode 100644 examples/protocols/http_server/restful_server/README.md create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/.browserslistrc create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/.editorconfig create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/.eslintrc.js create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/.gitignore create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/babel.config.js create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/package.json create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/postcss.config.js create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/public/favicon.ico create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/public/index.html create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/src/App.vue create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/src/assets/logo.png create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/src/main.js create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/src/plugins/vuetify.js create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/src/router.js create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/src/store.js create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/src/views/Chart.vue create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/src/views/Home.vue create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/src/views/Light.vue create mode 100644 examples/protocols/http_server/restful_server/front/web-demo/vue.config.js create mode 100644 examples/protocols/http_server/restful_server/main/CMakeLists.txt create mode 100644 examples/protocols/http_server/restful_server/main/Kconfig.projbuild create mode 100644 examples/protocols/http_server/restful_server/main/component.mk create mode 100644 examples/protocols/http_server/restful_server/main/esp_rest_main.c create mode 100644 examples/protocols/http_server/restful_server/main/rest_server.c create mode 100644 examples/protocols/http_server/restful_server/partitions_example.csv create mode 100644 examples/protocols/http_server/restful_server/sdkconfig.defaults diff --git a/examples/protocols/http_server/restful_server/CMakeLists.txt b/examples/protocols/http_server/restful_server/CMakeLists.txt new file mode 100644 index 0000000000..cb143ac640 --- /dev/null +++ b/examples/protocols/http_server/restful_server/CMakeLists.txt @@ -0,0 +1,7 @@ +cmake_minimum_required(VERSION 3.5) + +# This example uses an extra component for common functions such as Wi-Fi and Ethernet connection. +set(EXTRA_COMPONENT_DIRS $ENV{IDF_PATH}/examples/common_components/protocol_examples_common) + +include($ENV{IDF_PATH}/tools/cmake/project.cmake) +project(restful_server) diff --git a/examples/protocols/http_server/restful_server/Makefile b/examples/protocols/http_server/restful_server/Makefile new file mode 100644 index 0000000000..6870cd55f6 --- /dev/null +++ b/examples/protocols/http_server/restful_server/Makefile @@ -0,0 +1,14 @@ +PROJECT_NAME := restful_server + +EXTRA_COMPONENT_DIRS = $(IDF_PATH)/examples/common_components/protocol_examples_common + +include $(IDF_PATH)/make/project.mk + +ifdef CONFIG_WEB_DEPLOY_SF +WEB_SRC_DIR = $(shell pwd)/front/web-demo +ifneq ($(wildcard $(WEB_SRC_DIR)/dist/.*),) +$(eval $(call spiffs_create_partition_image,www,$(WEB_SRC_DIR)/dist,FLASH_IN_PROJECT)) +else +$(error $(WEB_SRC_DIR)/dist doesn't exist. Please run 'npm run build' in $(WEB_SRC_DIR)) +endif +endif diff --git a/examples/protocols/http_server/restful_server/README.md b/examples/protocols/http_server/restful_server/README.md new file mode 100644 index 0000000000..0a2ad7d871 --- /dev/null +++ b/examples/protocols/http_server/restful_server/README.md @@ -0,0 +1,135 @@ +# HTTP Restful API Server Example + +(See the README.md file in the upper level 'examples' directory for more information about examples.) + +## Overview + +This example mainly introduces how to implement a RESTful API server and HTTP server on ESP32, with a frontend browser UI. + +This example designs several APIs to fetch resources as follows: + +| API | Method | Resource Example | Description | Page URL | +| -------------------------- | ------ | ----------------------------------------------------- | ---------------------------------------------------------------------------------------- | -------- | +| `/api/v1/system/info` | `GET` | {
version:"v4.0-dev",
cores:2
} | Used for clients to get system information like IDF version, ESP32 cores, etc | `/` | +| `/api/v1/temp/raw` | `GET` | {
raw:22
} | Used for clients to get raw temperature data read from sensor | `/chart` | +| `/api/v1/light/brightness` | `POST` | {
red:160,
green:160,
blue:160
} | Used for clients to upload control values to ESP32 in order to control LED’s brightness | `/light` | + +**Page URL** is the URL of the webpage which will send a request to the API. + +### About mDNS + +The IP address of an IoT device may vary from time to time, so it’s impracticable to hard code the IP address in the webpage. In this example, we use the `mDNS` to parse the domain name `esp-home.local`, so that we can alway get access to the web server by this URL no matter what the real IP address behind it. See [here](https://docs.espressif.com/projects/esp-idf/en/latest/api-reference/protocols/mdns.html) for more information about mDNS. + +**Notes: mDNS is installed by default on most operating systems or is available as separate package.** + +### About deploy mode + +In development mode, it would be awful to flash the whole webpages every time we update the html, js or css files. So it is highly recommended to deploy the webpage to host PC via `semihost` technology. Whenever the browser fetch the webpage, ESP32 can forward the required files located on host PC. By this mean, it will save a lot of time when designing new pages. + +After developing, the pages should be deployed to one of the following destinations: + +* SPI Flash - which is recommended when the website after built is small (e.g. less than 2MB). +* SD Card - which would be an option when the website after built is very large that the SPI Flash have not enough space to hold (e.g. larger than 2MB). + +### About frontend framework + +Many famous frontend frameworks (e.g. Vue, React, Angular) can be used in this example. Here we just take [Vue](https://vuejs.org/) as example and adopt the [vuetify](https://vuetifyjs.com/) as the UI library. + +## How to use example + +### Hardware Required + +To run this example, you need an ESP32 dev board (e.g. ESP32-WROVER Kit, ESP32-Ethernet-Kit) or ESP32 core board (e.g. ESP32-DevKitC). An extra JTAG adapter might also needed if you choose to deploy the website by semihosting. For more information about supported JTAG adapter, please refer to [select JTAG adapter](https://docs.espressif.com/projects/esp-idf/en/latest/api-guides/jtag-debugging/index.html#jtag-debugging-selecting-jtag-adapter). Or if you choose to deploy the website to SD card, an extra SD slot board is needed. + +#### Pin Assignment: + +Only if you deploy the website to SD card, then the following pin connection is used in this example. + +| ESP32 | SD Card | +| ------ | ------- | +| GPIO2 | D0 | +| GPIO4 | D1 | +| GPIO12 | D2 | +| GPIO13 | D3 | +| GPIO14 | CLK | +| GPIO15 | CMD | + + +### Configure the project + +Enter `make menuconfig` if you are using GNU Make based build system or enter `idf.py menuconfig` if you are using CMake based build system. + +In the `Example Connection Configuration` menu: + +* Choose the network interface in `Connect using` option based on your board. Currently we support both Wi-Fi and Ethernet. +* If you select the Wi-Fi interface, you also have to set: + * Wi-Fi SSID and Wi-Fi password that your esp32 will connect to. +* If you select the Ethernet interface, you also have to set: + * PHY model in `Ethernet PHY` option, e.g. IP101. + * PHY address in `PHY Address` option, which should be determined by your board schematic. + * EMAC Clock mode, GPIO used by SMI. + +In the `Example Configuration` menu: + +* Set the domain name in `mDNS Host Name` option. +* Choose the deploy mode in `Website deploy mode`, currently we support deploy website to host PC, SD card and SPI Nor flash. + * If we choose to `Deploy website to host (JTAG is needed)`, then we also need to specify the full path of the website in `Host path to mount (e.g. absolute path to web dist directory)`. +* Set the mount point of the website in `Website mount point in VFS` option, the default value is `/www`. + +### Build and Flash + +After the webpage design work has been finished, you should compile them by running following commands: + +```bash +cd path_to_this_example/front/web-demo +npm install +npm run build +``` + +After a while, you will see a `dist` directory which contains all the website files (e.g. html, js, css, images). + +Enter `make -j4 flash monitor` if you are using GNU Make based build system or enter `idf.py build flash monitor` if you are using CMake based build system. + +(To exit the serial monitor, type ``Ctrl-]``.) + +See the [Getting Started Guide](https://docs.espressif.com/projects/esp-idf/en/latest/get-started/index.html) for full steps to configure and use ESP-IDF to build projects. + +### Extra steps to do for deploying website by semihost + +We need to run the latest version of OpenOCD which should support semihost feature when we test this deploy mode: + +```bash +openocd-esp32/bin/openocd -s openocd-esp32/share/openocd/scripts -f interface/ftdi/esp32_devkitj_v1.cfg -f board/esp-wroom-32.cfg +``` + +## Example Output + +### Render webpage in browser + +In your browser, enter the URL where the website located (e.g. `http://esp-home.local`). You can also enter the IP address that ESP32 obtained if your operating system currently don't have support for mDNS service. + +![esp_home_local](https://dl.espressif.com/dl/esp-idf/docs/_static/esp_home_local.gif) + +### ESP monitor output + +In the *Light* page, after we set up the light color and click on the check button, the browser will send a post request to ESP32, and in the console, we just print the color value. + +```bash +I (6115) example_connect: Connected to Ethernet +I (6115) example_connect: IPv4 address: 192.168.2.151 +I (6325) esp-home: Partition size: total: 1920401, used: 1587575 +I (6325) esp-rest: Starting HTTP Server +I (128305) esp-rest: File sending complete +I (128565) esp-rest: File sending complete +I (128855) esp-rest: File sending complete +I (129525) esp-rest: File sending complete +I (129855) esp-rest: File sending complete +I (137485) esp-rest: Light control: red = 50, green = 85, blue = 28 +``` + +## Troubleshooting + +1. Error occurred when building example: `...front/web-demo/dist doesn't exit. Please run 'npm run build' in ...front/web-demo`. + * When you choose to deploy website to SPI flash, make sure the `dist` directory has been generated before you building this example. + +(For any technical queries, please open an [issue](https://github.com/espressif/esp-idf/issues) on GitHub. We will get back to you as soon as possible.) diff --git a/examples/protocols/http_server/restful_server/front/web-demo/.browserslistrc b/examples/protocols/http_server/restful_server/front/web-demo/.browserslistrc new file mode 100644 index 0000000000..9dee646463 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/.browserslistrc @@ -0,0 +1,3 @@ +> 1% +last 2 versions +not ie <= 8 diff --git a/examples/protocols/http_server/restful_server/front/web-demo/.editorconfig b/examples/protocols/http_server/restful_server/front/web-demo/.editorconfig new file mode 100644 index 0000000000..7053c49a04 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/.editorconfig @@ -0,0 +1,5 @@ +[*.{js,jsx,ts,tsx,vue}] +indent_style = space +indent_size = 2 +trim_trailing_whitespace = true +insert_final_newline = true diff --git a/examples/protocols/http_server/restful_server/front/web-demo/.eslintrc.js b/examples/protocols/http_server/restful_server/front/web-demo/.eslintrc.js new file mode 100644 index 0000000000..98d043169d --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/.eslintrc.js @@ -0,0 +1,17 @@ +module.exports = { + root: true, + env: { + node: true + }, + 'extends': [ + 'plugin:vue/essential', + '@vue/standard' + ], + rules: { + 'no-console': process.env.NODE_ENV === 'production' ? 'error' : 'off', + 'no-debugger': process.env.NODE_ENV === 'production' ? 'error' : 'off' + }, + parserOptions: { + parser: 'babel-eslint' + } +} diff --git a/examples/protocols/http_server/restful_server/front/web-demo/.gitignore b/examples/protocols/http_server/restful_server/front/web-demo/.gitignore new file mode 100644 index 0000000000..d0d94890c3 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/.gitignore @@ -0,0 +1,26 @@ +.DS_Store +node_modules +/dist + +# local env files +.env.local +.env.*.local + +# Log files +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# Editor directories and files +.idea +.vscode +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? + +# APIs used in this example is simple and stable enough. +# There shouldn't be risk of compatibility unless the major version of some library changed. +# To compress the package size, just exclude the package-lock.json file. +package-lock.json diff --git a/examples/protocols/http_server/restful_server/front/web-demo/babel.config.js b/examples/protocols/http_server/restful_server/front/web-demo/babel.config.js new file mode 100644 index 0000000000..ba179669a1 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/babel.config.js @@ -0,0 +1,5 @@ +module.exports = { + presets: [ + '@vue/app' + ] +} diff --git a/examples/protocols/http_server/restful_server/front/web-demo/package.json b/examples/protocols/http_server/restful_server/front/web-demo/package.json new file mode 100644 index 0000000000..b2c4bc3ad7 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/package.json @@ -0,0 +1,32 @@ +{ + "name": "web-demo", + "version": "0.1.0", + "private": true, + "scripts": { + "serve": "vue-cli-service serve", + "build": "vue-cli-service build", + "lint": "vue-cli-service lint" + }, + "dependencies": { + "axios": "^0.18.0", + "core-js": "^2.6.5", + "vue": "^2.6.10", + "vue-router": "^3.0.3", + "vuetify": "^1.5.14", + "vuex": "^3.0.1" + }, + "devDependencies": { + "@vue/cli-plugin-babel": "^3.7.0", + "@vue/cli-plugin-eslint": "^3.7.0", + "@vue/cli-service": "^3.7.0", + "@vue/eslint-config-standard": "^4.0.0", + "babel-eslint": "^10.0.1", + "eslint": "^5.16.0", + "eslint-plugin-vue": "^5.0.0", + "stylus": "^0.54.5", + "stylus-loader": "^3.0.1", + "vue-cli-plugin-vuetify": "^0.5.0", + "vue-template-compiler": "^2.5.21", + "vuetify-loader": "^1.0.5" + } +} diff --git a/examples/protocols/http_server/restful_server/front/web-demo/postcss.config.js b/examples/protocols/http_server/restful_server/front/web-demo/postcss.config.js new file mode 100644 index 0000000000..961986e2b1 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/postcss.config.js @@ -0,0 +1,5 @@ +module.exports = { + plugins: { + autoprefixer: {} + } +} diff --git a/examples/protocols/http_server/restful_server/front/web-demo/public/favicon.ico b/examples/protocols/http_server/restful_server/front/web-demo/public/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..17b8ff3cb6c3c920c732f563c2507c8472dfb95c GIT binary patch literal 6429 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 + + + + + + + ESP-HOME + + + + + +
+ + + diff --git a/examples/protocols/http_server/restful_server/front/web-demo/src/App.vue b/examples/protocols/http_server/restful_server/front/web-demo/src/App.vue new file mode 100644 index 0000000000..0d79bb1819 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/src/App.vue @@ -0,0 +1,55 @@ + + + diff --git a/examples/protocols/http_server/restful_server/front/web-demo/src/assets/logo.png b/examples/protocols/http_server/restful_server/front/web-demo/src/assets/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..c19919e5cafb7933aa0815c044461ddde4f765ac GIT binary patch literal 37327 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

hB-DHf$}=oQdIfrhPXj4tbqj#Pa=cE6wDP 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 h(App) +}).$mount('#app') diff --git a/examples/protocols/http_server/restful_server/front/web-demo/src/plugins/vuetify.js b/examples/protocols/http_server/restful_server/front/web-demo/src/plugins/vuetify.js new file mode 100644 index 0000000000..975696e792 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/src/plugins/vuetify.js @@ -0,0 +1,7 @@ +import Vue from 'vue' +import Vuetify from 'vuetify/lib' +import 'vuetify/src/stylus/app.styl' + +Vue.use(Vuetify, { + iconfont: 'md', +}) diff --git a/examples/protocols/http_server/restful_server/front/web-demo/src/router.js b/examples/protocols/http_server/restful_server/front/web-demo/src/router.js new file mode 100644 index 0000000000..2e6ce9440b --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/src/router.js @@ -0,0 +1,29 @@ +import Vue from 'vue' +import Router from 'vue-router' +import Home from './views/Home.vue' +import Chart from './views/Chart.vue' +import Light from './views/Light.vue' + +Vue.use(Router) + +export default new Router({ + mode: 'history', + base: process.env.BASE_URL, + routes: [ + { + path: '/', + name: 'home', + component: Home + }, + { + path: '/chart', + name: 'chart', + component: Chart + }, + { + path: '/light', + name: 'light', + component: Light + } + ] +}) diff --git a/examples/protocols/http_server/restful_server/front/web-demo/src/store.js b/examples/protocols/http_server/restful_server/front/web-demo/src/store.js new file mode 100644 index 0000000000..62f44f6515 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/src/store.js @@ -0,0 +1,28 @@ +import Vue from 'vue' +import Vuex from 'vuex' +import axios from 'axios' + +Vue.use(Vuex) + +export default new Vuex.Store({ + state: { + chart_value: [8, 2, 5, 9, 5, 11, 3, 5, 10, 0, 1, 8, 2, 9, 0, 13, 10, 7, 16], + }, + mutations: { + update_chart_value(state, new_value) { + state.chart_value.push(new_value); + state.chart_value.shift(); + } + }, + actions: { + update_chart_value({ commit }) { + axios.get("/api/v1/temp/raw") + .then(data => { + commit("update_chart_value", data.data.raw); + }) + .catch(error => { + console.log(error); + }); + } + } +}) diff --git a/examples/protocols/http_server/restful_server/front/web-demo/src/views/Chart.vue b/examples/protocols/http_server/restful_server/front/web-demo/src/views/Chart.vue new file mode 100644 index 0000000000..7d75477cfd --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/src/views/Chart.vue @@ -0,0 +1,41 @@ + + + diff --git a/examples/protocols/http_server/restful_server/front/web-demo/src/views/Home.vue b/examples/protocols/http_server/restful_server/front/web-demo/src/views/Home.vue new file mode 100644 index 0000000000..f3ab672878 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/src/views/Home.vue @@ -0,0 +1,40 @@ + + + diff --git a/examples/protocols/http_server/restful_server/front/web-demo/src/views/Light.vue b/examples/protocols/http_server/restful_server/front/web-demo/src/views/Light.vue new file mode 100644 index 0000000000..bcfbe56e4e --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/src/views/Light.vue @@ -0,0 +1,63 @@ + + + + diff --git a/examples/protocols/http_server/restful_server/front/web-demo/vue.config.js b/examples/protocols/http_server/restful_server/front/web-demo/vue.config.js new file mode 100644 index 0000000000..0322551512 --- /dev/null +++ b/examples/protocols/http_server/restful_server/front/web-demo/vue.config.js @@ -0,0 +1,11 @@ +module.exports = { + devServer: { + proxy: { + '/api': { + target: 'http://esp-home.local:80', + changeOrigin: true, + ws: true + } + } + } +} diff --git a/examples/protocols/http_server/restful_server/main/CMakeLists.txt b/examples/protocols/http_server/restful_server/main/CMakeLists.txt new file mode 100644 index 0000000000..e0a89216f0 --- /dev/null +++ b/examples/protocols/http_server/restful_server/main/CMakeLists.txt @@ -0,0 +1,13 @@ +set(COMPONENT_SRCS "esp_rest_main.c" "rest_server.c") +set(COMPONENT_ADD_INCLUDEDIRS ".") + +register_component() + +if(CONFIG_WEB_DEPLOY_SF) + set(WEB_SRC_DIR "${CMAKE_CURRENT_SOURCE_DIR}/../front/web-demo") + if(EXISTS ${WEB_SRC_DIR}/dist) + spiffs_create_partition_image(www ${WEB_SRC_DIR}/dist FLASH_IN_PROJECT) + else() + message(FATAL_ERROR "${WEB_SRC_DIR}/dist doesn't exit. Please run 'npm run build' in ${WEB_SRC_DIR}") + endif() +endif() diff --git a/examples/protocols/http_server/restful_server/main/Kconfig.projbuild b/examples/protocols/http_server/restful_server/main/Kconfig.projbuild new file mode 100644 index 0000000000..5ae5a36e6f --- /dev/null +++ b/examples/protocols/http_server/restful_server/main/Kconfig.projbuild @@ -0,0 +1,50 @@ +menu "Example Configuration" + + config MDNS_HOST_NAME + string "mDNS Host Name" + default "esp-home" + help + Specify the domain name used in the mDNS service. + Note that webpage also take it as a part of URL where it will send GET/POST requests to. + + choice WEB_DEPLOY_MODE + prompt "Website deploy mode" + default WEB_DEPLOY_SEMIHOST + help + Select website deploy mode. + You can deploy website to host, and ESP32 will retrieve them in a semihost way (JTAG is needed). + You can deploy website to SD card or SPI flash, and ESP32 will retrieve them via SDIO/SPI interface. + Detailed operation steps are listed in the example README file. + config WEB_DEPLOY_SEMIHOST + bool "Deploy website to host (JTAG is needed)" + help + Deploy website to host. + It is recommended to choose this mode during developing. + config WEB_DEPLOY_SD + bool "Deploy website to SD card" + help + Deploy website to SD card. + Choose this production mode if the size of website is too large (bigger than 2MB). + config WEB_DEPLOY_SF + bool "Deploy website to SPI Nor Flash" + help + Deploy website to SPI Nor Flash. + Choose this production mode if the size of website is small (less than 2MB). + endchoice + + if WEB_DEPLOY_SEMIHOST + config HOST_PATH_TO_MOUNT + string "Host path to mount (e.g. absolute path to web dist directory)" + default "PATH-TO-WEB-DIST_DIR" + help + When using semihost in ESP32, you should specify the host path which will be mounted to VFS. + Note that only absolute path is acceptable. + endif + + config WEB_MOUNT_POINT + string "Website mount point in VFS" + default "/www" + help + Specify the mount point in VFS. + +endmenu diff --git a/examples/protocols/http_server/restful_server/main/component.mk b/examples/protocols/http_server/restful_server/main/component.mk new file mode 100644 index 0000000000..e69de29bb2 diff --git a/examples/protocols/http_server/restful_server/main/esp_rest_main.c b/examples/protocols/http_server/restful_server/main/esp_rest_main.c new file mode 100644 index 0000000000..7785a0a41c --- /dev/null +++ b/examples/protocols/http_server/restful_server/main/esp_rest_main.c @@ -0,0 +1,134 @@ +/* HTTP Restful API Server 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_event_loop.h" +#include "driver/sdmmc_host.h" +#include "driver/gpio.h" +#include "esp_vfs_semihost.h" +#include "esp_vfs_fat.h" +#include "esp_spiffs.h" +#include "sdmmc_cmd.h" +#include "nvs_flash.h" +#include "tcpip_adapter.h" +#include "esp_event.h" +#include "esp_log.h" +#include "mdns.h" +#include "protocol_examples_common.h" +#include "sdkconfig.h" + +#define MDNS_INSTANCE "esp home web server" + +static const char *TAG = "example"; + +esp_err_t start_rest_server(const char *base_path); + +static void initialise_mdns(void) +{ + mdns_init(); + mdns_hostname_set(CONFIG_MDNS_HOST_NAME); + mdns_instance_name_set(MDNS_INSTANCE); + + mdns_txt_item_t serviceTxtData[] = { + {"board", "esp32"}, + {"path", "/"} + }; + + ESP_ERROR_CHECK(mdns_service_add("ESP32-WebServer", "_http", "_tcp", 80, serviceTxtData, + sizeof(serviceTxtData) / sizeof(serviceTxtData[0]))); +} + +#if CONFIG_WEB_DEPLOY_SEMIHOST +esp_err_t init_fs(void) +{ + esp_err_t ret = esp_vfs_semihost_register(CONFIG_WEB_MOUNT_POINT, CONFIG_HOST_PATH_TO_MOUNT); + if (ret != ESP_OK) { + ESP_LOGE(TAG, "Failed to register semihost driver (%s)!", esp_err_to_name(ret)); + return ESP_FAIL; + } + return ESP_OK; +} +#endif + +#if CONFIG_WEB_DEPLOY_SD +esp_err_t init_fs(void) +{ + sdmmc_host_t host = SDMMC_HOST_DEFAULT(); + sdmmc_slot_config_t slot_config = SDMMC_SLOT_CONFIG_DEFAULT(); + + gpio_set_pull_mode(15, GPIO_PULLUP_ONLY); // CMD + gpio_set_pull_mode(2, GPIO_PULLUP_ONLY); // D0 + gpio_set_pull_mode(4, GPIO_PULLUP_ONLY); // D1 + gpio_set_pull_mode(12, GPIO_PULLUP_ONLY); // D2 + gpio_set_pull_mode(13, GPIO_PULLUP_ONLY); // D3 + + esp_vfs_fat_sdmmc_mount_config_t mount_config = { + .format_if_mount_failed = true, + .max_files = 4, + .allocation_unit_size = 16 * 1024 + }; + + sdmmc_card_t *card; + esp_err_t ret = esp_vfs_fat_sdmmc_mount(CONFIG_WEB_MOUNT_POINT, &host, &slot_config, &mount_config, &card); + if (ret != ESP_OK) { + if (ret == ESP_FAIL) { + ESP_LOGE(TAG, "Failed to mount filesystem."); + } else { + ESP_LOGE(TAG, "Failed to initialize the card (%s)", esp_err_to_name(ret)); + } + return ESP_FAIL; + } + /* print card info if mount successfully */ + sdmmc_card_print_info(stdout, card); + return ESP_OK; +} +#endif + +#if CONFIG_WEB_DEPLOY_SF +esp_err_t init_fs(void) +{ + esp_vfs_spiffs_conf_t conf = { + .base_path = CONFIG_WEB_MOUNT_POINT, + .partition_label = NULL, + .max_files = 5, + .format_if_mount_failed = false + }; + esp_err_t ret = esp_vfs_spiffs_register(&conf); + + if (ret != ESP_OK) { + if (ret == ESP_FAIL) { + ESP_LOGE(TAG, "Failed to mount or format filesystem"); + } else if (ret == ESP_ERR_NOT_FOUND) { + ESP_LOGE(TAG, "Failed to find SPIFFS partition"); + } else { + ESP_LOGE(TAG, "Failed to initialize SPIFFS (%s)", esp_err_to_name(ret)); + } + return ESP_FAIL; + } + + size_t total = 0, used = 0; + ret = esp_spiffs_info(NULL, &total, &used); + if (ret != ESP_OK) { + ESP_LOGE(TAG, "Failed to get SPIFFS partition information (%s)", esp_err_to_name(ret)); + } else { + ESP_LOGI(TAG, "Partition size: total: %d, used: %d", total, used); + } + return ESP_OK; +} +#endif + +void app_main() +{ + ESP_ERROR_CHECK(nvs_flash_init()); + tcpip_adapter_init(); + ESP_ERROR_CHECK(esp_event_loop_create_default()); + initialise_mdns(); + + ESP_ERROR_CHECK(example_connect()); + ESP_ERROR_CHECK(init_fs()); + ESP_ERROR_CHECK(start_rest_server(CONFIG_WEB_MOUNT_POINT)); +} diff --git a/examples/protocols/http_server/restful_server/main/rest_server.c b/examples/protocols/http_server/restful_server/main/rest_server.c new file mode 100644 index 0000000000..8c74ccec52 --- /dev/null +++ b/examples/protocols/http_server/restful_server/main/rest_server.c @@ -0,0 +1,225 @@ +/* HTTP Restful API Server + + 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 +#include +#include "esp_http_server.h" +#include "esp_system.h" +#include "esp_log.h" +#include "esp_vfs.h" +#include "cJSON.h" + +static const char *REST_TAG = "esp-rest"; +#define REST_CHECK(a, str, goto_tag, ...) \ + do \ + { \ + if (!(a)) \ + { \ + ESP_LOGE(REST_TAG, "%s(%d): " str, __FUNCTION__, __LINE__, ##__VA_ARGS__); \ + goto goto_tag; \ + } \ + } while (0) + +#define FILE_PATH_MAX (ESP_VFS_PATH_MAX + 128) +#define SCRATCH_BUFSIZE (10240) + +typedef struct rest_server_context { + char base_path[ESP_VFS_PATH_MAX + 1]; + char scratch[SCRATCH_BUFSIZE]; +} rest_server_context_t; + +#define CHECK_FILE_EXTENSION(filename, ext) (strcasecmp(&filename[strlen(filename) - strlen(ext)], ext) == 0) + +/* Set HTTP response content type according to file extension */ +static esp_err_t set_content_type_from_file(httpd_req_t *req, const char *filepath) +{ + const char *type = "text/plain"; + if (CHECK_FILE_EXTENSION(filepath, ".html")) { + type = "text/html"; + } else if (CHECK_FILE_EXTENSION(filepath, ".js")) { + type = "application/javascript"; + } else if (CHECK_FILE_EXTENSION(filepath, ".css")) { + type = "text/css"; + } else if (CHECK_FILE_EXTENSION(filepath, ".png")) { + type = "image/png"; + } else if (CHECK_FILE_EXTENSION(filepath, ".ico")) { + type = "image/x-icon"; + } else if (CHECK_FILE_EXTENSION(filepath, ".svg")) { + type = "text/xml"; + } + return httpd_resp_set_type(req, type); +} + +/* Send HTTP response with the contents of the requested file */ +static esp_err_t rest_common_get_handler(httpd_req_t *req) +{ + char filepath[FILE_PATH_MAX]; + + rest_server_context_t *rest_context = (rest_server_context_t *)req->user_ctx; + strlcpy(filepath, rest_context->base_path, sizeof(filepath)); + if (req->uri[strlen(req->uri) - 1] == '/') { + strlcat(filepath, "/index.html", sizeof(filepath)); + } else { + strlcat(filepath, req->uri, sizeof(filepath)); + } + int fd = open(filepath, O_RDONLY, 0); + if (fd == -1) { + ESP_LOGE(REST_TAG, "Failed to open file : %s", filepath); + /* Respond with 500 Internal Server Error */ + httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to read existing file"); + return ESP_FAIL; + } + + set_content_type_from_file(req, filepath); + + char *chunk = rest_context->scratch; + ssize_t read_bytes; + do { + /* Read file in chunks into the scratch buffer */ + read_bytes = read(fd, chunk, SCRATCH_BUFSIZE); + if (read_bytes == -1) { + ESP_LOGE(REST_TAG, "Failed to read file : %s", filepath); + } else if (read_bytes > 0) { + /* Send the buffer contents as HTTP response chunk */ + if (httpd_resp_send_chunk(req, chunk, read_bytes) != ESP_OK) { + close(fd); + ESP_LOGE(REST_TAG, "File sending failed!"); + /* Abort sending file */ + httpd_resp_sendstr_chunk(req, NULL); + /* Respond with 500 Internal Server Error */ + httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to send file"); + return ESP_FAIL; + } + } + } while (read_bytes > 0); + /* Close file after sending complete */ + close(fd); + ESP_LOGI(REST_TAG, "File sending complete"); + /* Respond with an empty chunk to signal HTTP response completion */ + httpd_resp_send_chunk(req, NULL, 0); + return ESP_OK; +} + +/* Simple handler for light brightness control */ +static esp_err_t light_brightness_post_handler(httpd_req_t *req) +{ + int total_len = req->content_len; + int cur_len = 0; + char *buf = ((rest_server_context_t *)(req->user_ctx))->scratch; + int received = 0; + if (total_len >= SCRATCH_BUFSIZE) { + /* Respond with 500 Internal Server Error */ + httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "content too long"); + return ESP_FAIL; + } + while (cur_len < total_len) { + received = httpd_req_recv(req, buf + cur_len, total_len); + if (received <= 0) { + /* Respond with 500 Internal Server Error */ + httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to post control value"); + return ESP_FAIL; + } + cur_len += received; + } + buf[total_len] = '\0'; + + cJSON *root = cJSON_Parse(buf); + int red = cJSON_GetObjectItem(root, "red")->valueint; + int green = cJSON_GetObjectItem(root, "green")->valueint; + int blue = cJSON_GetObjectItem(root, "blue")->valueint; + ESP_LOGI(REST_TAG, "Light control: red = %d, green = %d, blue = %d", red, green, blue); + cJSON_Delete(root); + httpd_resp_sendstr(req, "Post control value successfully"); + return ESP_OK; +} + +/* Simple handler for getting system handler */ +static esp_err_t system_info_get_handler(httpd_req_t *req) +{ + httpd_resp_set_type(req, "application/json"); + cJSON *root = cJSON_CreateObject(); + esp_chip_info_t chip_info; + esp_chip_info(&chip_info); + cJSON_AddStringToObject(root, "version", IDF_VER); + cJSON_AddNumberToObject(root, "cores", chip_info.cores); + const char *sys_info = cJSON_Print(root); + httpd_resp_sendstr(req, sys_info); + free((void *)sys_info); + cJSON_Delete(root); + return ESP_OK; +} + +/* Simple handler for getting temperature data */ +static esp_err_t temperature_data_get_handler(httpd_req_t *req) +{ + httpd_resp_set_type(req, "application/json"); + cJSON *root = cJSON_CreateObject(); + cJSON_AddNumberToObject(root, "raw", esp_random() % 20); + const char *sys_info = cJSON_Print(root); + httpd_resp_sendstr(req, sys_info); + free((void *)sys_info); + cJSON_Delete(root); + return ESP_OK; +} + +esp_err_t start_rest_server(const char *base_path) +{ + REST_CHECK(base_path, "wrong base path", err); + rest_server_context_t *rest_context = calloc(1, sizeof(rest_server_context_t)); + REST_CHECK(rest_context, "No memory for rest context", err); + strncpy(rest_context->base_path, base_path, ESP_VFS_PATH_MAX); + + httpd_handle_t server = NULL; + httpd_config_t config = HTTPD_DEFAULT_CONFIG(); + config.uri_match_fn = httpd_uri_match_wildcard; + + ESP_LOGI(REST_TAG, "Starting HTTP Server"); + REST_CHECK(httpd_start(&server, &config) == ESP_OK, "Start server failed", err_start); + + /* URI handler for fetching system info */ + httpd_uri_t system_info_get_uri = { + .uri = "/api/v1/system/info", + .method = HTTP_GET, + .handler = system_info_get_handler, + .user_ctx = rest_context + }; + httpd_register_uri_handler(server, &system_info_get_uri); + + /* URI handler for fetching temperature data */ + httpd_uri_t temperature_data_get_uri = { + .uri = "/api/v1/temp/raw", + .method = HTTP_GET, + .handler = temperature_data_get_handler, + .user_ctx = rest_context + }; + httpd_register_uri_handler(server, &temperature_data_get_uri); + + /* URI handler for light brightness control */ + httpd_uri_t light_brightness_post_uri = { + .uri = "/api/v1/light/brightness", + .method = HTTP_POST, + .handler = light_brightness_post_handler, + .user_ctx = rest_context + }; + httpd_register_uri_handler(server, &light_brightness_post_uri); + + /* URI handler for getting web server files */ + httpd_uri_t common_get_uri = { + .uri = "/*", + .method = HTTP_GET, + .handler = rest_common_get_handler, + .user_ctx = rest_context + }; + httpd_register_uri_handler(server, &common_get_uri); + + return ESP_OK; +err_start: + free(rest_context); +err: + return ESP_FAIL; +} diff --git a/examples/protocols/http_server/restful_server/partitions_example.csv b/examples/protocols/http_server/restful_server/partitions_example.csv new file mode 100644 index 0000000000..ca4898e596 --- /dev/null +++ b/examples/protocols/http_server/restful_server/partitions_example.csv @@ -0,0 +1,6 @@ +# Name, Type, SubType, Offset, Size, Flags +# Note: if you change the phy_init or app partition offset, make sure to change the offset in Kconfig.projbuild +nvs, data, nvs, 0x9000, 0x6000, +phy_init, data, phy, 0xf000, 0x1000, +factory, app, factory, 0x10000, 1M, +www, data, spiffs, , 2M, diff --git a/examples/protocols/http_server/restful_server/sdkconfig.defaults b/examples/protocols/http_server/restful_server/sdkconfig.defaults new file mode 100644 index 0000000000..599472d848 --- /dev/null +++ b/examples/protocols/http_server/restful_server/sdkconfig.defaults @@ -0,0 +1,9 @@ +CONFIG_HTTPD_MAX_REQ_HDR_LEN=1024 +CONFIG_SPIFFS_OBJ_NAME_LEN=64 +CONFIG_FATFS_LONG_FILENAME=y +CONFIG_FATFS_LFN_HEAP=y +CONFIG_ESPTOOLPY_FLASHSIZE_4MB=y +CONFIG_PARTITION_TABLE_CUSTOM=y +CONFIG_PARTITION_TABLE_CUSTOM_FILENAME="partitions_example.csv" +CONFIG_PARTITION_TABLE_CUSTOM_APP_BIN_OFFSET=0x10000 +CONFIG_PARTITION_TABLE_FILENAME="partitions_example.csv"