2016-07-06 5 views
0

У меня есть сценарий Lua, который отправляет мне электронное письмо через SMTP. Все работает нормально при загрузке в NodeMCU и говорит dofile("sendemail.lua").Тайм-аут NodeMCU при использовании цикла while

-- sendmail.lua  

-- The email and password from the account you want to send emails from 
    MY_EMAIL = "REDACTED" 

EMAIL_PASSWORD = "REDACTED" 

-- The SMTP server and port of your email provider. 
-- If you don't know it google [my email provider] SMTP settings 
SMTP_SERVER = "isp.smtp.server" 
SMTP_PORT = 25 

-- The account you want to send email to 
mail_to = "REDACTED" 

-- Your access point's SSID and password 
SSID = "REDACTED" 
SSID_PASSWORD = "REDACTED" 

-- configure ESP as a station 
wifi.setmode(wifi.STATION) 
wifi.sta.config(SSID,SSID_PASSWORD) 
wifi.sta.autoconnect(1) 

email_subject = "" 
email_body = "" 
count = 0 


local smtp_socket = nil -- will be used as socket to email server 

-- The display() function will be used to print the SMTP server's response 
function display(sck,response) 
    print(response) 
end 

-- The do_next() function is used to send the SMTP commands to the SMTP server in the required sequence. 
-- I was going to use socket callbacks but the code would not run callbacks after the first 3. 
function do_next() 
      if(count == 0)then 
       count = count+1 
       IP_ADDRESS = wifi.sta.getip() 
       smtp_socket:send("HELO "..IP_ADDRESS.."\r\n") 
      elseif(count==1) then 
       count = count+1 
       smtp_socket:send("AUTH LOGIN\r\n") 
      elseif(count == 2) then 
       count = count + 1 
       smtp_socket:send("REDACTED".."\r\n") 
      elseif(count == 3) then 
       count = count + 1 
       smtp_socket:send("REDACTED".."\r\n") 
      elseif(count==4) then 
       count = count+1 
       smtp_socket:send("MAIL FROM:<" .. MY_EMAIL .. ">\r\n") 
      elseif(count==5) then 
       count = count+1 
       smtp_socket:send("RCPT TO:<" .. mail_to ..">\r\n") 
      elseif(count==6) then 
       count = count+1 
       smtp_socket:send("DATA\r\n") 
      elseif(count==7) then 
       count = count+1 
       local message = string.gsub(
       "From: \"".. MY_EMAIL .."\"<"..MY_EMAIL..">\r\n" .. 
       "To: \"".. mail_to .. "\"<".. mail_to..">\r\n".. 
       "Subject: ".. email_subject .. "\r\n\r\n" .. 
       email_body,"\r\n.\r\n","") 

       smtp_socket:send(message.."\r\n.\r\n") 
      elseif(count==8) then 
       count = count+1 
       tmr.stop(0) 
       smtp_socket:send("QUIT\r\n") 
       print("msg sent") 
      else 
       smtp_socket:close() 
      end 
      print(count) 
end 

-- The connectted() function is executed when the SMTP socket is connected to the SMTP server. 
-- This function will create a timer to call the do_next function which will send the SMTP commands 
-- in sequence, one by one, every 5000 seconds. 
-- You can change the time to be smaller if that works for you, I used 5000ms just because. 
function connected(sck) 
    tmr.alarm(0,5000,1,do_next) 
end 

-- @name send_email 
-- @description Will initiated a socket connection to the SMTP server and trigger the connected() function 
-- @param subject The email's subject 
-- @param body The email's body 
function send_email(subject,body) 

    count = 0 
    email_subject = subject 
    email_body = body 
    smtp_socket = net.createConnection(net.TCP,0) 
    smtp_socket:on("connection",connected) 
    smtp_socket:on("receive",display) 
    smtp_socket:connect(SMTP_PORT, SMTP_SERVER)  
end 
-- Send an email 
send_email("ESP8266", "[[Hi, How are your IoT projects coming along? Best Wishes,ESP8266]]") 

Однако я хочу использовать цикл для контроля аналогового входного значения и только послать электронную почту, когда определенные значения аналогового ввода обнаружены. Поэтому я добавил этот код в конце скрипта, после определения sendemail() функции и непосредственно перед функцией sendmail('subject', 'body') называется

vp = 0 
gpio.mode(vp, gpio.INPUT) 

while true do 

    local v = adc.read(vp) 
    if v < 840 or v > 870 then 
     print(v) 
     break 
    end 
    tmr.wdclr() 
end 
sendmail('subject', 'body') 

Петля пока работает отлично, бесконечное ожидание ввода с аналоговым вывода. Как только этот вход найден, он ломается правильно и вызывает функцию sendmail. Однако, как только эта функция вызывается, NodeMCU в конечном счете сбрасывается. Иногда он доходит до успешной аутентификации учетных данных SMTP с сервером, и иногда он даже не сделает HELO до того, как он выключится. Что может быть причиной этого? Почему сценарий sendmail.lua работает нормально, а затем вдруг решит не работать, когда добавляете этот небольшой цикл while, который, кажется, отлично работает сам по себе?

ответ

1

Немного цитата из ссылки NodeMCU:

tmr.wdclr() питающие системы сторожевой таймер.

В общем, если вам когда-либо понадобится использовать эту функцию, вы делаете это неправильно.

Модель NodeMCU, управляемая событиями, означает, что нет необходимости быть , сидящей в жестких петлях, ожидая появления вещей. Скорее просто используйте обратные вызовы, чтобы получать уведомления, когда что-то происходит. При таком подходе никогда не должно быть необходимости вручную подключать систему сторожевого таймера.

Обратите внимание на вторую строчку. :)

Не знаете, в чем проблема, но почему вы используете цикл while в первую очередь? Почему бы не использовать таймерные события для регулярного опроса вашего АЦП?

Возможно, сторожевой таймер вызван, потому что по какой-то причине ваш канал приходит поздно. В случае, если вы не кормите его вообще, выходите из цикла.

1

Даже если это не может быть определенный ответ, я отправляю его как таковой, поскольку ввод комментариев слишком мал.

Во-первых, я предлагаю вам использовать сценарий, который я опубликовал для вашего предыдущего вопроса. Это неправильно настройка WiFi. Вам нужно подождать по таймеру, пока устройство не получит IP-адрес, прежде чем вы сможете продолжить. Помните, что wifi.sta.config не блокирует. И поскольку он использует auto-connect=true, если он не задан явно, он попытается немедленно подключиться к AP. Вот почему wifi.sta.autoconnect(1) является излишним.

Я не понимаю код чтения АЦП, который вы опубликовали.

vp = 0 
gpio.mode(vp, gpio.INPUT) 

Кажется ненужным для меня, потому что а) вы ничего не делать с GPIO 0 и б) adc.read поддерживает только 0.

Вместо того, чтобы использовать петлю занятости и постоянно кормить собаку, which is a very bad sign, я предлагаю вам использовать таймер с интервалом. Кроме того, я думаю, вы не хотите разбить петлю при первом выполнении условия и никогда не возвращаться? Итак, вам нужно оставаться в цикле и продолжать запускать отправку почты, нет? Что-то вроде этого возможно (непроверено):

tmr.alarm(0, 200, tmr.ALARM_AUTO, function() 
    local v = adc.read(0) 
    if v < 840 or v > 870 then 
    node.task.post(function() 
     send_email("ESP8266", "[[Hi, How are your IoT projects coming along? Best Wishes,ESP8266]]") 
    end) 
    end 
end) 
+0

Таким образом, это все еще сбрасывает около десяти секунд в тревогу таймера. Любые идеи, почему это может произойти? – user1173922

+0

На самом деле это происходит случайным образом, когда я пытаюсь запустить скрипты на моем NodeMCU. Кажется, что отдельные части работают нормально, поэтому я думаю, что у меня либо есть плохое оборудование, либо как-то прошивка перепуталась. – user1173922

 Смежные вопросы

  • Нет связанных вопросов^_^