Данная статья будет посвящена ESP8266 Wi-Fi модулю, языку программирования LUA и прошивке nodeMCU. SDK от производителя рассматриваться не будет.
Примерно года три назад я пробовал реализовать выключатель по 1-wire шине. Как все работало мне очень не понравилось.
- Единая точка отказа т.к. вся логика на сервере;
- Медленная скорость;
- К каждому выключателю придется тянуть от 2х проводов(идеально «витуху»).
В следствии чего все это было удачно заброшено, другие беспроводные решения рассматривались, но были исключены в виду дороговизны, небезопасного протокола и сложности реализации. Хотелось чего то простого с минимумом компонентов, со своей логикой и дешёвого. Не давно заказал 2 штуки esp8266 просто для забавы, не зная, чего конкретного с ними можно сделать. После 2-х вечеров разборок с чипом вспомнил незавершённое дело с кнопкой и решил довести до логического конца.
На данный модуль уже существует некоторое количество прошивок, так же вы можете писать прошивку под себя, используя SDK, но не стал вникать в подробности написания, т.к. после изучения API nodeMCU понял, что данного функционала мне хватает с запасом и прошил оба модуля.
Железо
Себестоимость важный фактор для простого выключателя, так что пытался использовать как можно меньше частей. Решил сделать из того что было дома, но пришлось купить твердотельное реле. Кстати, «релюшка» стоит дороже wifi модуля и ее можно заменить на оптопару, симистор и обвязку, схемы включения легко ищутся в интернете. Был случай, когда плохой контакт в патроне лампочки выбил симистор на коротко. Посмотрим, как покажет себя оптореле, ведь раньше с ними не работал. Стоит учесть для большой нагрузки установка радиатора обязательна.
Тут сразу столкнулся с проблемой, если на gpio при включении он уходил на землю, плата переходила или в режим прошивки или в непонятный режим, т.к кнопка у нас нормально разомкнута, с ней переделывать нечего не стал и так и оставил замыкаться на землю, а оптореле повесил на плюс через резистор и включал подачей 0, выключал подачей 1, соответственно. В итоге получилась такая схема:
Внимание схему стоит улучшить! Выход на реле стоит подать через транзистор, а кнопку подтянуть через резистор от плюса. Ингредиенты получились такие:
- выключатель;
- пружинка(для переделки выключателя в кнопку);
- сам esp8266;
- твердотельное реле использовал(S202T02);
- платка для конструирования;
- резистор 470 Ом;
- провода;
- разъемы по вкусу;
- зарядка от телефона 400мА 5v;
- стабилизатор 1117 3.3v;
- пара конденсаторов.
Переделка выключателя не заняла много времени, выкинул стандартный светодиод. Протянул провода от модуля в центре выключателя, сам модуль расположил снаружи под пластмассовой кнопкой, а силовая часть внутри. Не много фотографий процесса (фото с телефона):
nodeMCU
Прошивка использует Lua язык программирования, данный язык похож чем то на Javascript. Версия ещё сыроватая, но уже базовый функционал вполне не плохо реализован. Сразу после загрузки модуль начинает исполнять файл скрипт init.lua, в чистой прошивке этого файла нет, вам приходиться его создать руками. Все операции можно осуществлять через консоль подключенным к «com» порту, для упрощения заливки файлов в модуль есть скрипт luatool. Заливка работает следующим образом и данный код полностью показывает процесс записи в файл.
file.open("init.lua","w")
file.writeline([[print("Hello World!")]])
file.writeline([[--comment]])
file.close()
Пример чтения конфигурационного файла. Выглядит не очень. Может есть и другой вариант сериализованных данных.
file.open('config')
c_wifi_ssid = string.gsub(file.readline(), "n", "")
c_wifi_key = file.readline()
file.close()
Пример цикла с использованием API с паузой в 1000 миллисекунд представлен ниже:
tmr.alarm(1000, 1, function()
if wifi.sta.getip()=="0.0.0.0" then --текущий ip
print("connecting to AP..."..c_wifi_ssid.."/"..c_wifi_key)
else
print('ip: ',wifi.sta.getip())
tmr.stop() -- alarm stop
end
end)
Работа с GPIO
Если у вас модель модуля ESP-01 новой ревизии, то вам доступно всего 2 gpio, не прибегая к грязному хаку.
Решил отказаться от такого хака и воспользоваться тем, что есть.
Один gpio кнопка и второй выход на твердотельное реле. Есть еще Tx, но заставить работать как gpio у меня не получилось, и для индикации я просто передаю сообщения в консоль print(). Пока закостылил именно так. Чем длиннее сообщение, тем дольше и ярче вспыхивает светодиод. Владельцы данной модификации пролетают лесом и с такими функциями как (node.key, node.led), т.к. они могут использовать только GPIO16, который тоже не разведен на плате.
Все gpio могут работать в нескольких режимах (OUTPUT, INPUT, INT), но интересно то, что функция gpio.read(), прежде чем считать, подает низкий уровень, даже если установлен режим OUTPUT. То есть, чтобы получить текущее состояние выхода, это не подходит. Пришлось использовать внешнюю переменную и писать две функции для удобства, а уже через переменную определять активность.
function on()
gpio.write(8,gpio.LOW)
oo=1
end
function off()
gpio.write(8,gpio.HIGH)
oo=0
end
В качестве событий можно использовать callback gpio.trig(pin, type, function(level)), второй параметр может принимать следующие значения «up», «down», «both», «low», «high». Тут, кажется, все ясно. Если у вас вывод находится в состоянии 1 и мы его опускаем на землю, срабатывает down, потом при поднятии срабатывает up, но, к моему сожалению, такого не происходило, в консоли я видел только down в зависимости от скорости нажатия кнопки событие срабатывало 1 или 2 раза. Решил поставить цикл с паузой и бряк по 1 на gpio.
for i=1,1000 do
print(i)
tmr.delay(10)
tmr.wdclr() -- сбрасывает счетчик и предотвращая авто перезагрузку
end
Но пауза не отработала, а без паузы устройство уходило в перезагрузку. Зато print(i) вносил хорошую задержку. Сделал через tmr.alarm, но в текущий момент активный цикл может быть только один, что не очень подходит.
function down()
tmr.alarm(100, 1, function()
timer = timer + 1
-- ok
if gpio.read(9) == 1 then
print(timer)
tmr.stop()
if timer < 20 then
switch()
else
-- ...
end
timer = 0
end
tmr.wdclr()
end)
end
gpio.trig(9, "down", function (gp)
if timer == 0 then
timer = 1
down()
end
end)
HTTP сервер
Сервер запускается как 2 пальца, но никакого массива параметров запроса не получите. Пока непонятно, как оптимальнее: или писать свой велосипед, или find по подстроке. Согласитесь, выглядит ужасно. В данном примере ищется 2 параметра key и mode=off,on,party. Последний режим – это простое мигание лампочкой каждые 200мс, можно поставить и побыстрее, но побоялся за лампочку и отпореле.
function HTTPd()
print('start http serv')
srv=net.createServer(net.TCP, 5)
srv:listen(80,function(conn)
conn:on("receive",function(conn,payload)
print(payload)
if string.find(payload, "key="..c_api_key) then
msg = "key_ok"
if string.find(payload,"mode=on") then
on()
else
if string.find(payload,"mode=off") then
tmr.stop()
off()
else
if string.find(payload, "mode=party") then
party(200)
end
end
end
else
msg = "error_key"
end
conn:send("<html><head></head><body><h1> mode=[on,off,party] key='api_key' </h1><p>"..msg.."</p></body></html>")
end)
conn:on("sent",function(conn) conn:close() end)
end)
end
Не так сложно написать простенький веб-интерфейс, а скрипты и стили расположить на внешних серверах. С модуля забирать только index страницу и общаться с ним, допустим, по json, так не будет большой нагрузки и все влезет в файловую систему, но мы становимся зависимыми от наличия интернета.
Wi-Fi
Для подключения в качестве клиента хватит такого кода, ip получим по DHCP:
wifi.setmode(wifi.STATION)
wifi.sta.config(c_wifi_ssid,c_wifi_key)
wifi.sta.autoconnect(1)
В третьем примере видно, как можно определить, подключилась точка или нет. Другие режимы wifi я не использовал, так что возможны подводные камни.
В целом, с небольшими хитростями (костылями) была написана логика простого выключателя. А кнопкой можно управлять прямо из консоли. Поставив программу BetterTouchTool, можно комбинацией клавиш «curl`ить» сервер выключателя. Нашлось применение кнопке EJECT.
Резюме
Нисколько не жалею, что выбрал nodeMCU: не пришлось залезать в дебри официального SDK. Не надо каждый раз компилировать код, а саму программу легко разбить на модули и заливать отдельными кусочками. За это пришлось поплатиться иногда непонятным поведением. Главное, что вся логика находится на самом выключателе, и он при недоступности wifi будет выполнять свою главную функцию, ну, только если сгорит. Самому проекту nodeMCU я желаю скорейшего развития и в дальнейшем попробую сделать что-нибудь ещё, благо остался еще один модуль.
Полезные ссылки
Автор: hav0k
Копипаст ста три с гиктаймс http://geektimes.ru/post/242366/
Это, вообще-то, указано в конце статьи (ссылка на источник)