mirror of
https://github.com/nodemcu/nodemcu-firmware.git
synced 2025-01-16 20:52:57 +08:00
295 lines
12 KiB
C
295 lines
12 KiB
C
#include "c_stdio.h"
|
|
#include "c_string.h"
|
|
#include "c_stdlib.h"
|
|
#include "coap.h"
|
|
|
|
#include "lua.h"
|
|
#include "lauxlib.h"
|
|
#include "lualib.h"
|
|
|
|
#include "os_type.h"
|
|
|
|
void build_well_known_rsp(char *rsp, uint16_t rsplen);
|
|
|
|
void endpoint_setup(void)
|
|
{
|
|
coap_setup();
|
|
}
|
|
|
|
static const coap_endpoint_path_t path_well_known_core = {2, {".well-known", "core"}};
|
|
static int handle_get_well_known_core(const coap_endpoint_t *ep, coap_rw_buffer_t *scratch, const coap_packet_t *inpkt, coap_packet_t *outpkt, uint8_t id_hi, uint8_t id_lo)
|
|
{
|
|
outpkt->content.p = (uint8_t *)c_zalloc(MAX_PAYLOAD_SIZE); // this should be free-ed when outpkt is built in coap_server_respond()
|
|
if(outpkt->content.p == NULL){
|
|
NODE_DBG("not enough memory\n");
|
|
return COAP_ERR_BUFFER_TOO_SMALL;
|
|
}
|
|
outpkt->content.len = MAX_PAYLOAD_SIZE;
|
|
build_well_known_rsp(outpkt->content.p, outpkt->content.len);
|
|
return coap_make_response(scratch, outpkt, (const uint8_t *)outpkt->content.p, c_strlen(outpkt->content.p), id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_CONTENT, COAP_CONTENTTYPE_APPLICATION_LINKFORMAT);
|
|
}
|
|
|
|
static const coap_endpoint_path_t path_variable = {2, {"v1", "v"}};
|
|
static int handle_get_variable(const coap_endpoint_t *ep, coap_rw_buffer_t *scratch, const coap_packet_t *inpkt, coap_packet_t *outpkt, uint8_t id_hi, uint8_t id_lo)
|
|
{
|
|
const coap_option_t *opt;
|
|
uint8_t count;
|
|
int n;
|
|
if (NULL != (opt = coap_findOptions(inpkt, COAP_OPTION_URI_PATH, &count)))
|
|
{
|
|
if ((count != ep->path->count ) && (count != ep->path->count + 1)) // +1 for /f/[function], /v/[variable]
|
|
{
|
|
NODE_DBG("should never happen.\n");
|
|
goto end;
|
|
}
|
|
if (count == ep->path->count + 1)
|
|
{
|
|
coap_luser_entry *h = ep->user_entry->next; // ->next: skip the first entry(head)
|
|
while(NULL != h){
|
|
if (opt[count-1].buf.len != c_strlen(h->name))
|
|
{
|
|
h = h->next;
|
|
continue;
|
|
}
|
|
if (0 == c_memcmp(h->name, opt[count-1].buf.p, opt[count-1].buf.len))
|
|
{
|
|
NODE_DBG("/v1/v/");
|
|
NODE_DBG((char *)h->name);
|
|
NODE_DBG(" match.\n");
|
|
if(h->L == NULL)
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_NOT_FOUND, COAP_CONTENTTYPE_NONE);
|
|
if(c_strlen(h->name))
|
|
{
|
|
n = lua_gettop(h->L);
|
|
lua_getglobal(h->L, h->name);
|
|
if (!lua_isnumber(h->L, -1)) {
|
|
NODE_DBG ("should be a number.\n");
|
|
lua_settop(h->L, n);
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_NOT_FOUND, COAP_CONTENTTYPE_NONE);
|
|
} else {
|
|
const char *res = lua_tostring(h->L,-1);
|
|
lua_settop(h->L, n);
|
|
return coap_make_response(scratch, outpkt, (const uint8_t *)res, c_strlen(res), id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_CONTENT, COAP_CONTENTTYPE_TEXT_PLAIN);
|
|
}
|
|
}
|
|
} else {
|
|
h = h->next;
|
|
}
|
|
}
|
|
}else{
|
|
NODE_DBG("/v1/v match.\n");
|
|
goto end;
|
|
}
|
|
}
|
|
NODE_DBG("none match.\n");
|
|
end:
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_CONTENT, COAP_CONTENTTYPE_TEXT_PLAIN);
|
|
}
|
|
|
|
static const coap_endpoint_path_t path_function = {2, {"v1", "f"}};
|
|
static int handle_post_function(const coap_endpoint_t *ep, coap_rw_buffer_t *scratch, const coap_packet_t *inpkt, coap_packet_t *outpkt, uint8_t id_hi, uint8_t id_lo)
|
|
{
|
|
const coap_option_t *opt;
|
|
uint8_t count;
|
|
int n;
|
|
if (NULL != (opt = coap_findOptions(inpkt, COAP_OPTION_URI_PATH, &count)))
|
|
{
|
|
if ((count != ep->path->count ) && (count != ep->path->count + 1)) // +1 for /f/[function], /v/[variable]
|
|
{
|
|
NODE_DBG("should never happen.\n");
|
|
goto end;
|
|
}
|
|
if (count == ep->path->count + 1)
|
|
{
|
|
coap_luser_entry *h = ep->user_entry->next; // ->next: skip the first entry(head)
|
|
while(NULL != h){
|
|
if (opt[count-1].buf.len != c_strlen(h->name))
|
|
{
|
|
h = h->next;
|
|
continue;
|
|
}
|
|
if (0 == c_memcmp(h->name, opt[count-1].buf.p, opt[count-1].buf.len))
|
|
{
|
|
NODE_DBG("/v1/f/");
|
|
NODE_DBG((char *)h->name);
|
|
NODE_DBG(" match.\n");
|
|
|
|
if(h->L == NULL)
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_NOT_FOUND, COAP_CONTENTTYPE_NONE);
|
|
|
|
if(c_strlen(h->name))
|
|
{
|
|
n = lua_gettop(h->L);
|
|
lua_getglobal(h->L, h->name);
|
|
if (lua_type(h->L, -1) != LUA_TFUNCTION) {
|
|
NODE_DBG ("should be a function\n");
|
|
lua_settop(h->L, n);
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_NOT_FOUND, COAP_CONTENTTYPE_NONE);
|
|
} else {
|
|
lua_pushlstring(h->L, inpkt->payload.p, inpkt->payload.len); // make sure payload.p is filled with '\0' after payload.len, or use lua_pushlstring
|
|
lua_call(h->L, 1, 1);
|
|
if (!lua_isnil(h->L, -1)){ /* get return? */
|
|
if( lua_isstring(h->L, -1) ) // deal with the return string
|
|
{
|
|
size_t len = 0;
|
|
const char *ret = luaL_checklstring( h->L, -1, &len );
|
|
if(len > MAX_PAYLOAD_SIZE){
|
|
lua_settop(h->L, n);
|
|
luaL_error( h->L, "return string:<MAX_PAYLOAD_SIZE" );
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_NOT_FOUND, COAP_CONTENTTYPE_NONE);
|
|
}
|
|
NODE_DBG((char *)ret);
|
|
NODE_DBG("\n");
|
|
lua_settop(h->L, n);
|
|
return coap_make_response(scratch, outpkt, ret, len, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_CONTENT, COAP_CONTENTTYPE_TEXT_PLAIN);
|
|
}
|
|
} else {
|
|
lua_settop(h->L, n);
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_CONTENT, COAP_CONTENTTYPE_TEXT_PLAIN);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
h = h->next;
|
|
}
|
|
}
|
|
}else{
|
|
NODE_DBG("/v1/f match.\n");
|
|
goto end;
|
|
}
|
|
}
|
|
NODE_DBG("none match.\n");
|
|
end:
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_NOT_FOUND, COAP_CONTENTTYPE_NONE);
|
|
}
|
|
|
|
extern lua_Load gLoad;
|
|
extern os_timer_t lua_timer;
|
|
extern void dojob(lua_Load *load);
|
|
static const coap_endpoint_path_t path_command = {2, {"v1", "c"}};
|
|
static int handle_post_command(const coap_endpoint_t *ep, coap_rw_buffer_t *scratch, const coap_packet_t *inpkt, coap_packet_t *outpkt, uint8_t id_hi, uint8_t id_lo)
|
|
{
|
|
if (inpkt->payload.len == 0)
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_BAD_REQUEST, COAP_CONTENTTYPE_TEXT_PLAIN);
|
|
if (inpkt->payload.len > 0)
|
|
{
|
|
lua_Load *load = &gLoad;
|
|
if(load->line_position == 0){
|
|
coap_buffer_to_string(load->line, load->len,&inpkt->payload);
|
|
load->line_position = c_strlen(load->line)+1;
|
|
// load->line[load->line_position-1] = '\n';
|
|
// load->line[load->line_position] = 0;
|
|
// load->line_position++;
|
|
load->done = 1;
|
|
NODE_DBG("Get command:\n");
|
|
NODE_DBG(load->line); // buggy here
|
|
NODE_DBG("\nResult(if any):\n");
|
|
os_timer_disarm(&lua_timer);
|
|
os_timer_setfn(&lua_timer, (os_timer_func_t *)dojob, load);
|
|
os_timer_arm(&lua_timer, READLINE_INTERVAL, 0); // no repeat
|
|
}
|
|
return coap_make_response(scratch, outpkt, NULL, 0, id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_CONTENT, COAP_CONTENTTYPE_TEXT_PLAIN);
|
|
}
|
|
}
|
|
|
|
static uint32_t id = 0;
|
|
static const coap_endpoint_path_t path_id = {2, {"v1", "id"}};
|
|
static int handle_get_id(const coap_endpoint_t *ep, coap_rw_buffer_t *scratch, const coap_packet_t *inpkt, coap_packet_t *outpkt, uint8_t id_hi, uint8_t id_lo)
|
|
{
|
|
id = system_get_chip_id();
|
|
return coap_make_response(scratch, outpkt, (const uint8_t *)(&id), sizeof(uint32_t), id_hi, id_lo, &inpkt->tok, COAP_RSPCODE_CONTENT, COAP_CONTENTTYPE_TEXT_PLAIN);
|
|
}
|
|
|
|
coap_luser_entry var_head = {NULL,NULL,NULL};
|
|
coap_luser_entry *variable_entry = &var_head;
|
|
|
|
coap_luser_entry func_head = {NULL,NULL,NULL};
|
|
coap_luser_entry *function_entry = &func_head;
|
|
|
|
const coap_endpoint_t endpoints[] =
|
|
{
|
|
{COAP_METHOD_GET, handle_get_well_known_core, &path_well_known_core, "ct=40", NULL},
|
|
{COAP_METHOD_GET, handle_get_variable, &path_variable, "ct=0", &var_head},
|
|
{COAP_METHOD_POST, handle_post_function, &path_function, NULL, &func_head},
|
|
{COAP_METHOD_POST, handle_post_command, &path_command, NULL, NULL},
|
|
{COAP_METHOD_GET, handle_get_id, &path_id, "ct=0", NULL},
|
|
{(coap_method_t)0, NULL, NULL, NULL, NULL}
|
|
};
|
|
|
|
void build_well_known_rsp(char *rsp, uint16_t rsplen)
|
|
{
|
|
const coap_endpoint_t *ep = endpoints;
|
|
int i;
|
|
uint16_t len = rsplen;
|
|
|
|
c_memset(rsp, 0, len);
|
|
|
|
len--; // Null-terminated string
|
|
|
|
while(NULL != ep->handler)
|
|
{
|
|
if (NULL == ep->core_attr) {
|
|
ep++;
|
|
continue;
|
|
}
|
|
if (NULL == ep->user_entry){
|
|
if (0 < c_strlen(rsp)) {
|
|
c_strncat(rsp, ",", len);
|
|
len--;
|
|
}
|
|
|
|
c_strncat(rsp, "<", len);
|
|
len--;
|
|
|
|
for (i = 0; i < ep->path->count; i++) {
|
|
c_strncat(rsp, "/", len);
|
|
len--;
|
|
|
|
c_strncat(rsp, ep->path->elems[i], len);
|
|
len -= c_strlen(ep->path->elems[i]);
|
|
}
|
|
|
|
c_strncat(rsp, ">;", len);
|
|
len -= 2;
|
|
|
|
c_strncat(rsp, ep->core_attr, len);
|
|
len -= c_strlen(ep->core_attr);
|
|
} else {
|
|
coap_luser_entry *h = ep->user_entry->next; // ->next: skip the first entry(head)
|
|
while(NULL != h){
|
|
if (0 < c_strlen(rsp)) {
|
|
c_strncat(rsp, ",", len);
|
|
len--;
|
|
}
|
|
|
|
c_strncat(rsp, "<", len);
|
|
len--;
|
|
|
|
for (i = 0; i < ep->path->count; i++) {
|
|
c_strncat(rsp, "/", len);
|
|
len--;
|
|
|
|
c_strncat(rsp, ep->path->elems[i], len);
|
|
len -= c_strlen(ep->path->elems[i]);
|
|
}
|
|
|
|
c_strncat(rsp, "/", len);
|
|
len--;
|
|
|
|
c_strncat(rsp, h->name, len);
|
|
len -= c_strlen(h->name);
|
|
|
|
c_strncat(rsp, ">;", len);
|
|
len -= 2;
|
|
|
|
c_strncat(rsp, ep->core_attr, len);
|
|
len -= c_strlen(ep->core_attr);
|
|
|
|
h = h->next;
|
|
}
|
|
}
|
|
ep++;
|
|
}
|
|
}
|
|
|