автоматизация с удобными pebble, стабильным noolite и доступной esp8266
Автоматизация с удобными pebble (дада, они ещё живут!), стабильным noolite (от компании Ноотехника, Минск, Беларусь) и доступной esp8266 development board.
Спасибо, за время, которое Вы потратили за чтением моих заметок.
С момента прошлой статьи (ссылка прошел почти год. И за этот год, я переосмыслил некоторые вещи, интернет и получилось что-то вроде IoT:) → интернет вещей)
Постараюсь кратко изложить новую порцию накопленных знаний, описать куда стремлюсь и чего хочу добиться → прошу к прочтению.
Noolite F v2.0
В заголовке у меня проскочили слова: pebble & noolite. Перейдем к ним!
Я ещё управляю с pebble своим светом noolite, гаражом и воротами. Про noolite уже написано достаточно много полезной информации, поиск показывает новые статьи.
Помимо блоков первого поколения системы noolite, у меня появились новые: cиловой блок SLF-1–200 (nooLite-F) с обратной связью (и шифрованием нового поколения)
силовой блок SB-1–150
и обновленный usb адаптера MTRF-64 USB.
Весь свет в моем доме на системе noolite.
Ещё недавно у меня в «системном домашнем корче» красовался noolite USB адаптер первого поколения и на 16 устройств, а теперь на 64 канала (MTRF-64 USB), да ещё и для устройств нового поколения с обратной связью (noolite F). Ноотехника движется в правильном направлении, создавая новый современные устройства.
Есть 2 самые главное опции новых устройств: обратная связь и принцип идентификации устройств по адресам: ID. Теперь не надо писать в 1 канал 1 устройств и управлять каналом. Теперь можно обращаться к конкретному устройств в одном канале по его ID и посылая ему команду исполнения.
Документация для блоков нового поколения доступна на сайте производителя.
# -*- coding: utf-8 -*-
#!/usr/bin/env python
import serial
import time
class NooLiteCommand:
def __init__(self, ch, cmd, mode=0, ctr=0, res=0, fmt=0, d0=0, d1=0, d2=0, d3=0, id0=0, id1=0, id2=0, id3=0):
self.st = 171
self.mode = mode
self.ctr = ctr
self.res = res
self.ch = ch
self.cmd = cmd
self.fmt = fmt
self.d0 = d0
self.d1 = d1
self.d2 = d2
self.d3 = d3
self.id0 = id0
self.id1 = id1
self.id2 = id2
self.id3 = id3
self.sp = 172
@property
def crc(self):
crc = sum([
self.st,
self.mode,
self.ctr,
self.res,
self.ch,
self.cmd,
self.fmt,
self.d0,
self.d1,
self.d2,
self.d3,
self.id0,
self.id1,
self.id2,
self.id3,
])
return crc if crc < 256 else divmod(crc, 256)[1]
def to_bytes(self):
return bytearray([
self.st,
self.mode,
self.ctr,
self.res,
self.ch,
self.cmd,
self.fmt,
self.d0,
self.d1,
self.d2,
self.d3,
self.id0,
self.id1,
self.id2,
self.id3,
self.crc,
self.sp
])
class NooliteSerial:
def __init__(self, tty_name):
self.tty = self._get_tty(tty_name)
def on(self, ch):
self.send_command(ch, 2, 2, 0)
pass
def off(self, ch):
self.send_command(ch, 0, 2, 0)
pass
def status(self, ch):
m = self.send_command(ch, 128, 2, 0)
pass
def send_command(self, ch, cmd, mode=0, ctr=0, res=0, fmt=0, d0=0, d1=0, d2=0, d3=0, id0=0, id1=0, id2=0, id3=0):
command = NooLiteCommand(ch, cmd, mode, ctr, res, fmt, d0, d1, d2, d3, id0, id1, id2, id3)
self.tty.write(command.to_bytes())
while True:
bytes_response = list(self.tty.read(117))
if bytes_response:
all_responses.append(bytes_response)
if bytes_response[3] == 0:
break
else:
break
return all_responses
@staticmethod
def _get_tty(tty_name):
serial_port = serial.Serial(tty_name, timeout=0.1)
if not serial_port.is_open:
serial_port.open()
serial_port.flushInput()
serial_port.flushOutput()
return serial_port
noo_serial = NooliteSerial('/dev/ttyUSB0')
#ch, cmd, mode, ctr
#noo_serial.send_command(1, 15, 2, 0) #включить привязку на канал 1
#noo_serial.send_command(0, 4, 2, 0) #switch
#noo_serial.send_command(0, 2, 2, 0) #turn on
#noo_serial.send_command(0, 3, 2, 0) #Регулировка яркости вверх
#noo_serial.send_command(0, 0, 2, 0) #turn off
#noo_serial.send_command(0, 15, 2, 0) #включить привязку
#noo_serial.send_command(0, 128, 2, 0,0,1) #CMD_Read_State + fmt = 1
#noo_serial.on(0)
#noo_serial.status(0)
#noo_serial.off(0)
#noo_serial.off(0)
#noo_serial.status(0)
#noo_serial.send_command(ch=0,ctr=8, cmd=4, id0=0,id1=0,id2=48,id3=114) #switch noolite ID 0.0.48.114
Используя нужные функции в нужное время, мы можем управлять noolite блоками и получать статус от новых. Так как у меня 99% это старые блоки, мне остается использовать старую схему записи 1 блока в 1 канал, с учетом того, что usb адаптер поддерживает и старые блоки и новые (спасибо разработчикам за совместимость без танцев бубном). На лету можно управлять и получать статус новых блоков и управлять старыми.
До недавнего времени, в доме не было ни одного кнопочного выключателя. Пришлось добавить в гараж свет (я просто добрался туда навести порядок: D), а так же появился выключатель в подсветке зеркала для макияжа. Зная, что есть силовой блок SB-1–150, я поставил выключатель обычный и подключил блок. Подсветку зеркала сделал из белой плиты, 3w лампочек 4000 К, белый свет. SB-1–150 уникален тем, что он может быть размещен в коробке настенного выключателя, (выключателя) в разрез существующей схемы и подключением кнопочного выключателя к самому блоку.
Pebble
Когда я начал увлекаться автоматизацией, одним из главных факторов было: грамотное и удобное управление.
Управлять со смартфона вроде бы и удобно, но… и не удобно, каждый раз грузить приложение, где маленькие неудобные кнопки, левая реклама или пока доберешься до нажатия кнопки… отпадет все желание. Голосовое управление отпадает, так как это тоже неудобно и странно :) — пусть это останется в сюжетах фантастических фильмов.
А я решил задачу по другому — через наручные часы, которые ПОСТОЯННО со мной, не занимают места и сочетают в себе все удобства, мобильность и быстроту управления. Вообщем — Pebble жив! Но самое главное то, что, для часов все ещё можно писать приложения — свои приложения, для управления своими системами автоматизации. Это вообще так-то «бомба».
Трагичность закрытия pebble.com компании (Fitbit выкупил ради патентов весь pebble.com) не отразилось на их производительности и работе. Недавно прошло обновление ios, android — часы отвязали от облачных сервисов pebble (в случае прекращения поддержки) и авторизации.
Визуально — ничего не изменилось, зато остался жить сервис: cloudpebble.net — WTF? Это очень полезная штука. Если Вы можете писать код на JS (и нет желания тратить много времени на C++) — пожалуйста, welcome on board:) — cloud pebble позволяет Вам быстро «накидать» приложение. Ну вот как я сделал. Конечно, производительность JS не стоит сравнивать с C++ (это святое), стерпиться. дА!
var UI = require('ui');
var ajax = require('ajax');
var noolite = [
['ворота','http://your-home-server-ip-address:1183/mqtt/gate/slidegate', 'images/gate.png'],
['гараж','http://your-home-server-ip-address:1183/mqtt/gate/garage', 'images/door.png'],
['приехали!','http://your-home-server-ip-address:1183/noolite/switch/103', 'images/light.png'],
['выкл все','http://your-home-server-ip-address:1183/noolite/switch/100', 'images/system2.png'],
['1 этаж свет','', 'images/light.png'],
['2 этаж свет','', 'images/light.png'],
['выкл 1эт','http://your-home-server-ip-address:1183/noolite/switch/101', 'images/light.png'],
['выкл 2эт','http://your-home-server-ip-address:1183/noolite/switch/102', 'images/light.png'],
['вода','', 'images/water.png'],
['мусорка','http://your-home-server-ip-address:1183/admin/system/1', 'images/system1.png'],
];
var water = [
['сенди вода','http://your-home-server-ip-address:1183/admin/poliv/1', 'images/water.png'],
['баня вода','http://your-home-server-ip-address:1183/admin/poliv/2', 'images/water.png'],
['зона 1 фасад','http://your-home-server-ip-address:1183/noolite/switch/20', 'images/water.png'],
['зона 2 фасад','http://your-home-server-ip-address:1183/noolite/switch/21', 'images/water.png'],
['зона 3 фасад','http://your-home-server-ip-address:1183/noolite/switch/22', 'images/water.png'],
['зона 4 фасад','http://your-home-server-ip-address:1183/noolite/switch/23', 'images/water.png'],
['зона 5 зад','http://your-home-server-ip-address:1183/admin/poliv/switch/5', 'images/water.png'],
['зона 6 зад','http://your-home-server-ip-address:1183/admin/poliv/switch/6', 'images/water.png'],
['зона 7 зад','http://your-home-server-ip-address:1183/admin/poliv/switch/7', 'images/water.png'],
['зона 8 зад','http://your-home-server-ip-address:1183/admin/poliv/switch/8', 'images/water.png'],
['зона 9 зад','http://your-home-server-ip-address:1183/admin/poliv/switch/9', 'images/water.png'],
];
var light1 = [
['коридор свет','http://your-home-server-ip-address:1183/noolite/switch/6', 'images/light.png'],
['кухня свет','http://your-home-server-ip-address:1183/noolite/switch/0', 'images/light.png'],
['кухня стол свет','http://your-home-server-ip-address:1183/noolite/switch/1', 'images/light.png'],
['зал низ свет','http://your-home-server-ip-address:1183/noolite/switch/7', 'images/light.png'],
['зал верх свет','http://your-home-server-ip-address:1183/noolite/switch/8', 'images/light.png'],
['ванная 1 свет','http://your-home-server-ip-address:1183/noolite/switch/2', 'images/light.png'],
['ванная 1 свет','http://your-home-server-ip-address:1183/noolite/switch/3', 'images/light.png'],
['гостевая 1 свет','http://your-home-server-ip-address:1183/noolite/switch/4', 'images/light.png'],
['топочная свет','http://your-home-server-ip-address:1183/noolite/switch/5', 'images/light.png'],
['гараж свет','http://your-home-server-ip-address:1183/noolite/switch/9', 'images/light.png'],
];
var light2 = [
['спальня свет','http://your-home-server-ip-address:1183/noolite/switch/15', 'images/light.png'],
['гардероб свет','http://your-home-server-ip-address:1183/noolite/switch/16', 'images/light.png'],
['коридор 2 свет','http://your-home-server-ip-address:1183/noolite/switch/10', 'images/light.png'],
['ванная 2 свет','http://your-home-server-ip-address:1183/noolite/switch/11', 'images/light.png'],
['гостевая 2 свет','http://your-home-server-ip-address:1183/noolite/switch/12', 'images/light.png'],
['Алиса свет','http://your-home-server-ip-address:1183/noolite/switch/13', 'images/light.png'],
['Алиса картина свет','http://your-home-server-ip-address:1183/noolite/switch/14', 'images/light.png'],
];
var menu = new UI.Menu({
sections: [{
items: [{
title: '',
subtitle: ''
}]
}]
});
var menu1 = new UI.Menu({
sections: [{
items: [{
title: '',
subtitle: ''
}]
}]
});
var menu2 = new UI.Menu({
sections: [{
items: [{
title: '',
subtitle: ''
}]
}]
});
var menu3 = new UI.Menu({
sections: [{
items: [{
title: '',
subtitle: ''
}]
}]
});
var items = [];
for (var i=0; i
Код пишется, компилируется прямо на сайте cloud Pebble и через телефон, скачивая скомпилированное приложение, можно установить на часы — за секунды. Все настолько просто, что справится любой юный автоматизатор ;)
Жаль… очень жаль, что Pebble прекратило существование и на текущий момент альтернативы я не вижу. Pebble Time всегда будут в наших сердцах!
MQTT
В прошлом году я познакомился с таким замечательным протоколом как MQTT (История: Первая версия протокола была разработана доктором Энди Станфорд-Кларком (IBM) и Арлен Ниппер (Arcom) в 1999 году и опубликована под роялти-фри лицензией. Спецификация MQTT 3.1.1 была стандартизирована консорциумом OASIS в 2014 году. ссылка)
Есть много информации как работает протокол и какие есть клиенты… и был сильно разочарован тем, что настоящих клиентов для мобильных приложений — нет, только на android.
Понимая, что отсутствие хорошего мобильного приложения, это весьма большая проблема (дыра) в области IoT (очень многие говорят, а делают очень мало…) — я со своей командой, на работе, решили создать клиент для android, ios, wp, чтобы он соответствовал всем критериям и был удобным. О клиенте отдельно я напишу чуть позже, а так же том, как мы подружили ESP8266+MTRF64 (Noolite) и подготовили Nodemcu прошивку. Будет здорово, удобно, дешево и красиво! Следите за новостями приложения тут.
Управлять Noolite освещением теперь можно будет без всяких домашних серверов и usb адаптеров за 50–100$!
Общая схема работы домашней автоматизации
Все ещё существует следующая схема:
— Видеонаблюдение: xeoma видеонаблюдение [ip камеры]
Сервер на системном блоке: nginx+gunicorn+python+mqtt broker
Управление: Pebble watch + MQTT client
Модули управления (ноды системы): ESP8266 + оптопары \ датчики ds18b20, dht11[22]
— Ввиду того, что я увидел в MQTT протоколе множество удобств, начиная от архитектуры подписки и мгновенным общением с конечными IoT устройствами, заканчивая тем, что MQTT сообщения могут пролетать через роутер, без «проброса» всяких там портов и танцами с бубном. Удобно! Весело! Задорно :)
Все это привело меня к децентрализации системы, то есть уход от одного «домашнего корчвагена» и переход на множество так называемых нодов и использование облачного MQTT брокера (есть бесплатные MQTT брокеры, например: mqtt.ximxim.com (на сайте логин и пасс для доступа)) — так как esp8266 решают все задачи, а она в свою очередь работают через wifi по протоколу MQTT.
Учитывая то, что MQTT Buddy будет предоставлять сервис сценариев, я вообще не буду думать о том, что мне надо писать какие то домашние сценарии, я просто буду создавать их в cloud решении и это и будет настоящий IoT и он работает! От слов к делу!
ESP8266
Когда я познакомился с этой платой разработки, меня сразу подкупило то, что плата маленькая, есть поддержка языка программирования LUA на пошивке Nodemcu. То, что Вам надо + MQTT модуль есть. В автоматизации, простые и надежные решения — это основа. Стоимость от 2$ до 5$ за плату (за 5 модель сразу с usb адаптером на борту) позволяет быстро разворачивать подсистемы.
Подключение по WiFi быстрое, возможностей масса. Например: управление воротами\гаражом\поливом\освещением. У меня 30% полива газонов работает от системы Noolite (блоки сухого контакта), остальные 70% от esp8266 development board с управлением через LUA язык по протоколу MQTT.
--load credentials
dofile("credentials.lua")
function init(name, pass)
wifi.setmode(wifi.STATION)
wifi.sta.config(name, pass)
wifi.sta.connect()
tmr.alarm(0, 1000, 1, function()
if wifi.sta.getip()== nil then
print("IP unavaiable, Waiting...")
else
tmr.stop(0)
print("Config done, IP is "..wifi.sta.getip())
print("mac : "..wifi.sta.getmac())
dofile("mqtt.lua")
end
end)
end
print("START!")
init(SSID,PASSWORD)
— mqtt.lua file
local door = 7 -- gpio13
local window = 6 -- gpio12
local cooler = 5 --gpio14
local led = 4 --GPIO2 board led! LOW == turn ON
local light1 = 8 --gpio 15
local light2 = 1 --gpio 5
local light3 = 2 -- gpio 4
function register_myself()
m:subscribe("mqtt_buddy/#",0,function(conn)
print ("subscribed to Xim mqtt.ximxim.com server")
end)
end
m = mqtt.Client("MQTT_BUDDY_SHOW_ROOM", 120, MQTT_USER, MQTT_PASS)
m:on("connect", function(client) print ("connected to Xim mqtt.ximxim.com server") end)
m:on("offline", function(client) reconnect_mqtt() end)
m:on("message", function(client, topic, data)
--print(topic.." data:"..data)
if topic == "mqtt_buddy/window" and data == "1" then
print "open-close window..."
gpio.write(window, gpio.HIGH)
gpio.write(led, gpio.LOW)
tmr.alarm(1, 1000, tmr.ALARM_SINGLE, function()
gpio.write(window, gpio.LOW)
gpio.write(led, gpio.HIGH)
print "command to close\open window is sent!"
end)
elseif topic == "mqtt_buddy/door" and data == "1" then
print "open-close door..."
gpio.write(door, gpio.HIGH)
gpio.write(led, gpio.LOW)
tmr.alarm(2, 1000, tmr.ALARM_SINGLE, function()
gpio.write(door, gpio.LOW)
gpio.write(led, gpio.HIGH)
print "command to close\open door is sent!"
end)
elseif topic == "mqtt_buddy/cooler" then
if data == "1" then
print "switch on fan..."
gpio.write(cooler, gpio.HIGH)
gpio.write(led, gpio.LOW)
else
print "switch off fan..."
gpio.write(cooler, gpio.LOW)
gpio.write(led, gpio.HIGH)
end
elseif topic == "mqtt_buddy/light1" then
if data == "1" then
print "switch light1 ON..."
gpio.write(light1, gpio.HIGH)
gpio.write(led, gpio.LOW)
else
print "switch light1 off..."
gpio.write(light1, gpio.LOW)
gpio.write(led, gpio.HIGH)
end
elseif topic == "mqtt_buddy/light2" then
if data == "1" then
print "switch light2 ON..."
gpio.write(light2, gpio.HIGH)
gpio.write(led, gpio.LOW)
else
print "switch light2 off..."
gpio.write(light2, gpio.LOW)
gpio.write(led, gpio.HIGH)
end
elseif topic == "mqtt_buddy/light3" then
if data == "1" then
print "switch light3 ON..."
gpio.write(light3, gpio.HIGH)
gpio.write(led, gpio.LOW)
else
print "switch light3 off..."
gpio.write(light3, gpio.LOW)
gpio.write(led, gpio.HIGH)
end
end
end)
m:connect(MQTT_SERVER, MQTT_SERVER_PORT, 0, function(conn) register_myself() end)
Проблема
Обращаясь к огромному сообществу думающих людей, хотел задеть крайне важную тему, а точнее проблему, которую хочу решить, но не хватает времени на более глубокие исследования \ знаний.
Суть: при децентрализации, остается видеонаблюдение, которое так или иначе ТРЕБУЕТ наличие host машины для (минимум) передачи source images либо на сервера видеонаблюдения (например ivideon) или вообще обрабатывать видео на домашней машине, как xeoma (хотя у них тоже есть cloud) — так или иначе — надо «ПРОБРАСЫВАТЬ» видео потоки, картинки на конечные вычислительные мощности (cloud решения) — или наоборот — сделать стриминг простой.
Есть мысли получать source images с камеры (тут тоже есть проблема, не в каждой камере есть URL для получения картинки с камеры и вообще непонятно, как другие узнают этот урл…) и пробрасывать в MQTT канал, где поддерживаются бинарные данные (то есть картинку можно легко передать через mqtt протокол в бинарном виде).
Возможно, кто-то уже пробовал реализовать такие вещи = ESP8266 + IP web cam?
Отзовитесь плиз (bogdanovich.alex[@]gmail.com). Буду очень благодарен!
При реализации проброса, отпадет вопрос домашнего сервера. Зачем? Потому что он электричество кушает и топочную мне греет :)
Всем позитивного дня и хорошего настроения!