feat(tcp_server): schema in message pack and more command
This commit is contained in:
parent
080a1d81e2
commit
1428ec2d11
7 changed files with 225 additions and 84 deletions
|
|
@ -2,9 +2,10 @@ file(GLOB_RECURSE MY_LIB_SOURCES
|
|||
"ESP32epdx/src/*.cpp"
|
||||
"ESP32epdx/src/GUI/*.cpp"
|
||||
"pubsubclient/src/*.cpp",
|
||||
"mpack/src/mpack/*.c"
|
||||
)
|
||||
|
||||
idf_component_register(
|
||||
SRCS "main.cpp" "EPD.cpp" "mesh_main.c" "mesh_netif.c" "tcphelper.c" ${MY_LIB_SOURCES}
|
||||
INCLUDE_DIRS "" "ESP32epdx/src" "ESP32epdx/src/GUI"
|
||||
INCLUDE_DIRS "" "ESP32epdx/src" "ESP32epdx/src/GUI" "mpack/src/mpack"
|
||||
)
|
||||
|
|
|
|||
146
main/main.cpp
146
main/main.cpp
|
|
@ -1,8 +1,10 @@
|
|||
#include "Arduino.h"
|
||||
#include "stdlib.h"
|
||||
|
||||
#include "Arduino.h"
|
||||
#include "nvs_flash.h"
|
||||
#include <mutex>
|
||||
|
||||
#include "mpack.h"
|
||||
#include "ESP32epdx.h"
|
||||
#include "EPD.h"
|
||||
#include "mesh_netif.h"
|
||||
|
|
@ -16,46 +18,142 @@ SPI: SCK = 6 ; MISO = 2 ; MOSI = 7 ; SS = 16
|
|||
EPD: RES = 22 ; DC = 23 ; CS = 1 ; BUSY = 0
|
||||
*/
|
||||
|
||||
#define IMAGE_BUF_SIZE 5624
|
||||
uint8_t *image_buf = nullptr;
|
||||
|
||||
extern "C" int tcp_server_callback(char *buf, size_t len) {
|
||||
if (!strncmp(buf, "ping", 4)) { // ping
|
||||
return 0;
|
||||
} else if (!strncmp(buf, "image", 5) && len == 5 + 5624) { // update image
|
||||
if (epd_mtx.try_lock())
|
||||
{
|
||||
// work
|
||||
EPD_HW_Init_Fast();
|
||||
EPD_Display((unsigned char *)(buf+5));
|
||||
EPD_DeepSleep();
|
||||
typedef struct request_t {
|
||||
// command : update_image, push_image, ping
|
||||
char *command;
|
||||
|
||||
//vTaskDelay(pdMS_TO_TICKS(1000));
|
||||
// arguments for update_image
|
||||
uint64_t index;
|
||||
char *data;
|
||||
size_t data_len;
|
||||
} request_t;
|
||||
|
||||
// unlock
|
||||
epd_mtx.unlock();
|
||||
|
||||
return 0;
|
||||
extern "C" request_t* parse_data(char *buf, size_t len) {
|
||||
// variables
|
||||
char *command = NULL;
|
||||
uint64_t index = 0;
|
||||
char *data = NULL;
|
||||
size_t data_len = 0;
|
||||
|
||||
// init reader
|
||||
mpack_reader_t reader;
|
||||
mpack_reader_init_data(&reader, buf, len);
|
||||
|
||||
size_t count = mpack_expect_map_max(&reader, 100);
|
||||
if (mpack_reader_error(&reader) != mpack_ok) {
|
||||
return NULL;
|
||||
}
|
||||
else
|
||||
|
||||
// read data
|
||||
for (size_t i = 0; i < count; i++) {
|
||||
char *key = mpack_expect_cstr_alloc(&reader, 100);
|
||||
|
||||
if (strcmp(key, "command") == 0) {
|
||||
command = mpack_expect_cstr_alloc(&reader, 32);
|
||||
} else if (strcmp(key, "index") == 0) {
|
||||
index = mpack_expect_u64(&reader);
|
||||
} else if (strcmp(key, "data") == 0) {
|
||||
data = mpack_expect_bin_alloc(&reader, CONFIG_IMAGE_BUF_SLICE_SIZE, &data_len);
|
||||
} else {
|
||||
mpack_discard(&reader);
|
||||
}
|
||||
|
||||
free(key);
|
||||
}
|
||||
|
||||
// finish
|
||||
mpack_done_map(&reader);
|
||||
if (mpack_reader_destroy(&reader) != mpack_ok) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
// return
|
||||
request_t *result = (request_t *)malloc(sizeof(request_t));
|
||||
if (!result) {
|
||||
return NULL;
|
||||
}
|
||||
memset(result, 0, sizeof(request_t));
|
||||
result->command = command;
|
||||
result->index = index;
|
||||
result->data = data;
|
||||
result->data_len = data_len;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
extern "C" int command_update_image(uint64_t index, char *buf, size_t len) {
|
||||
if (!epd_mtx.try_lock()) {
|
||||
return -3; // locked
|
||||
}
|
||||
|
||||
if (
|
||||
index < 0 || index >= CONFIG_IMAGE_BUF_SIZE ||
|
||||
len < 0 || len > CONFIG_IMAGE_BUF_SIZE ||
|
||||
(index+len) < 0 || (index+len) > CONFIG_IMAGE_BUF_SIZE
|
||||
)
|
||||
{
|
||||
// locked
|
||||
epd_mtx.unlock();
|
||||
return -2;
|
||||
}
|
||||
} else { // invalid request
|
||||
|
||||
memcpy(image_buf + index, buf, len);
|
||||
|
||||
epd_mtx.unlock();
|
||||
return 0;
|
||||
}
|
||||
|
||||
extern "C" int command_push_image() {
|
||||
if (epd_mtx.try_lock()) {
|
||||
EPD_HW_Init_Fast();
|
||||
EPD_Display((unsigned char *)(image_buf));
|
||||
EPD_DeepSleep();
|
||||
|
||||
epd_mtx.unlock();
|
||||
return 0;
|
||||
} else {
|
||||
return -3;
|
||||
}
|
||||
}
|
||||
|
||||
extern "C" void app_main()
|
||||
{
|
||||
extern "C" int tcp_server_callback(char *buf, size_t len) {
|
||||
request_t *req = parse_data(buf, len);
|
||||
if (!req) { // error -> invalid data
|
||||
return -2;
|
||||
}
|
||||
|
||||
// execute command
|
||||
if (strcmp(req->command, "ping") == 0)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
else if (strcmp(req->command, "update_image") == 0)
|
||||
{
|
||||
return command_update_image(req->index, req->data, req->data_len);
|
||||
}
|
||||
else if (strcmp(req->command, "push_image") == 0)
|
||||
{
|
||||
return command_push_image();
|
||||
}
|
||||
else
|
||||
{
|
||||
return -2;
|
||||
}
|
||||
}
|
||||
|
||||
extern "C" void app_main() {
|
||||
initArduino();
|
||||
|
||||
// init image_buf
|
||||
image_buf = (uint8_t *)malloc(IMAGE_BUF_SIZE);
|
||||
memset(image_buf, 0, IMAGE_BUF_SIZE);
|
||||
image_buf = (uint8_t *)malloc(CONFIG_IMAGE_BUF_SIZE);
|
||||
if (!image_buf) {
|
||||
return;
|
||||
}
|
||||
memset(image_buf, 0, CONFIG_IMAGE_BUF_SIZE);
|
||||
|
||||
//Initialize NVS
|
||||
// initialize NVS
|
||||
esp_err_t ret = nvs_flash_init();
|
||||
if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND) {
|
||||
ESP_ERROR_CHECK(nvs_flash_erase());
|
||||
|
|
|
|||
|
|
@ -22,9 +22,12 @@ static const uint8_t MESH_ID[6] = { 0x77, 0x77, 0x77, 0x77, 0x77, 0x76 };
|
|||
#define CONFIG_MESH_AP_CONNECTIONS 6 // number of nodes
|
||||
#define CONFIG_MESH_NON_MESH_AP_CONNECTIONS 0 // number of non-node devices
|
||||
|
||||
#define CONFIG_TCP_DEBUG 0
|
||||
#define CONFIG_TCP_SERVER_BIND_ADDRESS "0.0.0.0"
|
||||
#define CONFIG_TCP_SERVER_BIND_PORT "8888"
|
||||
#define CONFIG_TCP_SERVER_RXBUFFER_SIZE 8192
|
||||
#define CONFIG_TCP_SERVER_RXBUFFER_SIZE 4096 // tcp raw data max size
|
||||
#define CONFIG_IMAGE_BUF_SLICE_SIZE 2048 // map["data"] max size
|
||||
#define CONFIG_IMAGE_BUF_SIZE 5624
|
||||
|
||||
/*******************************************************
|
||||
* Function Declarations
|
||||
|
|
|
|||
|
|
@ -320,9 +320,12 @@ void mesh_main(void)
|
|||
cfg.router.ssid_len = strlen(CONFIG_MESH_ROUTER_SSID);
|
||||
memcpy((uint8_t *) &cfg.router.ssid, CONFIG_MESH_ROUTER_SSID, cfg.router.ssid_len);
|
||||
|
||||
#if CONFIG_TCP_DEBUG
|
||||
memcpy((uint8_t *) &cfg.router.password, CONFIG_MESH_ROUTER_PASSWD,
|
||||
strlen(CONFIG_MESH_ROUTER_PASSWD));
|
||||
//memset((uint8_t *) &cfg.router.password, 0, 64);
|
||||
#else
|
||||
memset((uint8_t *) &cfg.router.password, 0, 64);
|
||||
#endif
|
||||
|
||||
/* mesh softAP */
|
||||
ESP_ERROR_CHECK(esp_mesh_set_ap_authmode(CONFIG_MESH_AP_AUTHMODE));
|
||||
|
|
|
|||
1
main/mpack
Submodule
1
main/mpack
Submodule
|
|
@ -0,0 +1 @@
|
|||
Subproject commit a94ff1129d41a470ad9425ce6f730b3d8e99983f
|
||||
|
|
@ -250,14 +250,11 @@ static void tcp_server_task(void *pvParameters)
|
|||
case 0:
|
||||
strcpy(tx_buffer, "ok\n");
|
||||
break;
|
||||
case -1:
|
||||
strcpy(tx_buffer, "invalid data\n");
|
||||
break;
|
||||
case -2:
|
||||
strcpy(tx_buffer, "locked\n");
|
||||
strcpy(tx_buffer, "invalid request\n");
|
||||
break;
|
||||
case -3:
|
||||
strcpy(tx_buffer, "invalid request\n");
|
||||
strcpy(tx_buffer, "locked\n");
|
||||
break;
|
||||
default:
|
||||
strcpy(tx_buffer, "unknown error\n");
|
||||
|
|
|
|||
|
|
@ -1,38 +1,76 @@
|
|||
from typing import Dict, Any
|
||||
import socket
|
||||
|
||||
import msgpack
|
||||
|
||||
import tcp_client_image
|
||||
|
||||
SERVER_ADDRESS = ('172.16.0.249', 8888)
|
||||
|
||||
client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
||||
client_socket.connect(SERVER_ADDRESS)
|
||||
def send_request(data:Dict[str, Any]):
|
||||
client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
||||
client_socket.connect(SERVER_ADDRESS)
|
||||
|
||||
try:
|
||||
while True:
|
||||
message = ""
|
||||
choice = int(input("image > "))
|
||||
if choice == 1:
|
||||
message = b"image" + bytes(tcp_client_image.image1)
|
||||
elif choice == 2:
|
||||
message = b"image" + bytes(tcp_client_image.image2)
|
||||
elif choice == 3:
|
||||
message = b"ping"
|
||||
else:
|
||||
message = b"something_invalid"
|
||||
try:
|
||||
message:bytes = msgpack.packb(data)
|
||||
|
||||
# 4. 傳送資料 (需要編碼成 bytes)
|
||||
client_socket.sendall(message)
|
||||
|
||||
# 5. 接收伺服器的回應
|
||||
# 接收最多 1024 bytes
|
||||
data = client_socket.recv(1024)
|
||||
rx = client_socket.recv(1024)
|
||||
if data:
|
||||
print(f"收到伺服器訊息: {data.decode('utf-8')}")
|
||||
print(f"Received message: {rx.decode('utf-8').strip()}")
|
||||
else:
|
||||
print("伺服器已關閉連線。")
|
||||
break
|
||||
except ConnectionResetError:
|
||||
print("連線被伺服器重置。")
|
||||
finally:
|
||||
print("Connection closed")
|
||||
return
|
||||
except ConnectionResetError:
|
||||
print("Connection reset")
|
||||
finally:
|
||||
client_socket.close()
|
||||
print("客戶端連線已關閉。")
|
||||
print("Connection closed")
|
||||
|
||||
def image():
|
||||
image = b""
|
||||
try:
|
||||
choice = int(input("image > "))
|
||||
if choice == 1:
|
||||
image = tcp_client_image.image1
|
||||
elif choice == 2:
|
||||
image = tcp_client_image.image2
|
||||
else:
|
||||
raise ValueError
|
||||
except:
|
||||
print("invalid choice")
|
||||
return
|
||||
|
||||
chunk_size = 2048
|
||||
for i in range(0, len(image), chunk_size):
|
||||
data = {
|
||||
"command": "update_image",
|
||||
"index": i,
|
||||
"data": bytes(image[i:i+chunk_size])
|
||||
}
|
||||
send_request(data)
|
||||
|
||||
send_request({"command":"push_image"})
|
||||
|
||||
def main():
|
||||
while True:
|
||||
print("Command:")
|
||||
print("(1) update + push image")
|
||||
print("(2) ping")
|
||||
print("(3) invalid request")
|
||||
print("(4) exit")
|
||||
|
||||
choice:int = int(input("> "))
|
||||
data = {}
|
||||
if choice == 1:
|
||||
image()
|
||||
elif choice == 2:
|
||||
send_request({"command":"ping"})
|
||||
elif choice == 3:
|
||||
send_request({"command":"invalid_command"})
|
||||
else:
|
||||
break
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
Loading…
Add table
Reference in a new issue