具有自动更新的NodeMCU超小型Web服务器内存不足

时间:2017-09-17 14:06:47

标签: lua nodemcu

我正在测试一个具有自动更新功能的Lua Web服务器。 HTML代码每秒都会重定向到Web服务器本身。因此,客户端的Web浏览器总是从服务器获取新数据而不是 使用浏览器的缓存。

如果我连接一个客户端(我的电脑或我的智能手机)一段时间后,NodeMCU板崩溃了这条消息:

  

PANIC:调用Lua API时出现无保护错误(SO-WebSrv-Test.lua:27:内存不足)

我使用了Marcel Stoer的这段代码,他回答了类似的内容"内存耗尽" question

我修改了Marcel的Lua代码,但是这段代码仍然会耗尽所有堆内存。

我稍微缩小了问题的范围:如果HTML代码的刷新频率低于30秒,则代码会占用堆内存。

那么如何修改此代码以实现常量堆内存使用?

最好的问候。

的Stefan

tmr.alarm(0, 1000, 1, function()
   if wifi.sta.getip() == nil then
      print("trying to connect to AccessPoint...")
   else
      own_ip, netmask, gateway=wifi.sta.getip()
      print("connected to AccessPoint:")
      print("IP Info: \nIP Address of this device: ",own_ip)
      print("Netmask: ",netmask)
      print("Gateway Addr: ",gateway,"\n")
      print("type IP-Address "..own_ip.." into your browser to display SHT-31-website")
      tmr.stop(0)
   end
end)

counter = 0
srv = net.createServer(net.TCP, 28800)
print("Server created... \n")

srv:listen(80, function(conn)
    conn:on("receive", function(sck, request)
        local message = {}
        counter = counter + 1
        message[#message + 1] = "<head> <meta http-equiv=refresh content=1; URL=http://"..own_ip.."> </head>"
        message[#message + 1] = "<h1> ESP8266 SHT-31 Web Server Ver 003</h1>"
        message[#message + 1] = "<h2>some more text blabla blub"..counter.."</h2>"
        local function send(sk)
            if #message > 0 then
                sk:send(table.remove(message, 1))
            else
                sk:close()
                message = nil
                print("Heap Available:" .. node.heap())
            end
        end
        sck:on("sent", send)
        send(sck)
    end)
end)

1 个答案:

答案 0 :(得分:1)

您没有告诉我们您使用的固件版本。我在Chromium浏览器中测试了一个最新版本,并没有看到任何内存问题。我在700多次重载循环后中止了测试,堆耗完全稳定。

今年早些时候我们不得不reduce the TCP TIME_WAIT parameter value,因为太多被遗弃在时间等待状态的被遗弃的插座占用了记忆。说明:

  

TIME-WAIT

     

(服务器或客户端)表示等待足够的时间   传递以确保远程TCP收到其确认   连接终止请求。 [根据RFC 793,连接可以   留在TIME-WAIT最多四分钟,称为两个MSL   (最大段寿命)。]

来源:https://en.wikipedia.org/wiki/Transmission_Control_Protocol#Protocol_operation

更多详情:https://tools.ietf.org/html/rfc7230#section-6.6

然而:

  • 您似乎打算通过HTTP将HTML发送回您的客户,但不要告诉它,它可能不喜欢
  • 如果您的客户(浏览器?)没有按时关闭旧套接字,您也可以告诉它明确这样做

添加适当的HTTP标头可以解决这两个问题。因此,消息部分应该是这样的:

local message = { "HTTP/1.0 200 OK\r\nServer: NodeMCU on ESP8266\r\nConnection: close\r\nContent-Type: text/html\r\n\r\n" }

counter = counter + 1

message[#message + 1] = "<html><head> <meta http-equiv=refresh content=1; URL=http://" .. own_ip .. "> </head>"
message[#message + 1] = "<body><h1> ESP8266 SHT-31 Web Server Ver 003</h1>"
message[#message + 1] = "<h2>some more text blabla blub" .. counter .. "</h2></body></html>"

记下Connection: closeContent-Type: text/html以及结构合理的HTML标记。