Para integrar esta red a IOT en esquema abierto se ha seleccionado como gestor de gateways a ChirpStack, pues se integra a la gestión de paquetes y al gestor de mensajes MQTT versión Mosquitto.
De esta forma se genera un punto intermedio para integrar las conexiones con otros brokers de forma simplificada.
Dado que el servidor MQTT es parte del IOT Esquema Abierto, las instrucciones de instalación y configuración ya se encuentran descritas en:
Para el módulo HT-M01, el fabricante Heltec publicó una aplicación para gestionar los paquetes denominado packet-forwarder, encargada de reenviar los paquetes a un administrador de gateways.
Los datos recibidos por el módulo gateway son reenviados por SPI o el puerto USB hacia la red local o internet usando el aplicativo instalado en un Raspberry Pi.
En las pruebas con USB se encontró que para reiniciar el módulo HT-M01 es necesario presionar el botón Reset, mientras que en el modo SPI se podía realizar de forma remota, por lo que se prefiere configurar el modo SPI.
Activar interface SPI
En Raspberry OS la interface SPI requiere activarse para su uso con la siguiente instrucción:
sudo raspi-config
Que permite seleccionar de una ventana las opciones de interface
para luego activar SPI
Conexión mediante SPI
Las instrucciones paso a paso se describen más adelante, para la ultima instrucción hay que tener disponible la configuración de región. Para el caso de Ecuador es US915.
Cada instrucción se debe realizar en secuencia, el el penúltimo paso se obtiene el Gateway _id, que será usado para registrar el mini gateway en el servidor de red y aplicaciones.
Nota 2026-03: En Raspbian OS se recomienda usar un nombre de usuario diferente de "pi", por lo que se deben ajustar las direcciones en las instrucciones al usuario en las ultimas instrucciones.
En el directorio "lorasdk" Edite el archivo "install.sh" y "lrgateway.service" para evitar errores de donde se encuentra el archivo. Por ejemplo en install.sh, se editan las líneas con sudo nano install.sh
actualizado los archivos, se puede continuar con las correcciones en los directorios:
# This package will create a "lrgateway" service in Raspberry Pi cd /home/pi/lora/lora_gateway make clean all cd /home/pi/lora/packet_forwarder make clean all cd /home/pi/lora/lorasdk chmod +x install.sh ./install.sh #Run the script. After the script is run, it will create a # service named "lrgateway". The purpose is to make the lora driver # and data forwarding program run automatically at startup. sudo cp -f /home/pi/lora/lorasdk/global_conf_US915.json /home/pi/lora/packet_forwarder/lora_pkt_fwd/global_conf.json #the "global_conf_US915.json" may need change to your need.
Cambios para Raspberry pi OS Trixie 2026-03
En la versión Bookworm y Trixie, las instrucciones para manejar GPIOs han migrado a pinctrl, por lo que se debe actualizar el archivo /lora/packet_forwarder/lora_pkt_fwd.sh
# Force bypassing auto update of Gateway_ID in JSON conf file IOT_SK_GWID_UPDATE=true
# The reset pin of SX1301 is wired with RPi GPIO2 IOT_SK_SX1301_RESET_PIN=17
# set GPIO 17 as output echo "out" #> /sys/class/gpio/gpio$IOT_SK_SX1301_RESET_PIN/direction; WAIT_GPIO
# write output for SX1301 reset pinctrl set 17 op dh WAIT_GPIO pinctrl get 17 echo "1" #> /sys/class/gpio/gpio$IOT_SK_SX1301_RESET_PIN/value; WAIT_GPIO pinctrl set 17 op dl pinctrl get 17 WAIT_GPIO echo "0" #> /sys/class/gpio/gpio$IOT_SK_SX1301_RESET_PIN/value; WAIT_GPIO # set GPIO 17 as input pinctrl set 17 ip pinctrl get 17 WAIT_GPIO echo "in" #> /sys/class/gpio/gpio$IOT_SK_SX1301_RESET_PIN/direction; WAIT_GPIO }
iot_sk_term() { echo "iot_sk_term" # cleanup GPIO 17 pinctrl get 17 echo " term reiniciando" if [ -d /sys/class/gpio/gpio$IOT_SK_SX1301_RESET_PIN ] then pinctrl get 17 echo " iot_sk_term en if "$IOT_SK_SX1301_RESET_PIN" ..." #> /sys/class/gpio/unexport; WAIT_GPIO fi }
Conexión al puerto USB
Las instrucciones son muy semejantes al proceso anterior, para la ultima instrucción hay que tener disponible la configuración de región. Para el caso de Ecuador es US915.
Cada instrucción se debe realizar en secuencia, el el penúltimo paso se obtiene el Gateway _id, que será usado para registrar el mini gateway en el servidor de red y aplicaciones
Si el módulo fue instalado en el proceso anterior, no es necesario ejecutar esta sección
mkdir lora
cd lora
sudo apt-get update
sudo apt-get install git
git clone https://github.com/Lora-net/picoGW_hal.git
git clone https://github.com/Lora-net/picoGW_packet_forwarder.git
git clone https://github.com/HelTecAutomation/picolorasdk.git
cd /home/pi/lora/picoGW_hal
make clean all
cd /home/pi/lora/picoGW_packet_forwarder
make clean all
cd /home/pi/lora/picolorasdk
chmod +x install.sh
./install.sh
#Run this script will create a service named "lrgateway". The purpose is to make the lora driver and data forwarding program run automatically at startup.
sudo cp -f /home/pi/lora/picolorasdk/global_conf_US915.json /home/pi/lora/picoGW_packet_forwarder/lora_pkt_fwd/global_conf.json
#Put the configuration file on the specified path
Estado de Packet-forwarder
Las instrucciones de instalación se encuentran en:
HT-M01 Mini LoRa Gateway Quick Start. Heltec.org. Revisado Septiembre 2023
El archivo local contiene la identificación del gateway obtenida luego de ejecutar la línea ./install.sh del proceso anterior. El archivo global contiene la información de la región y las frecuencias usadas.
El archivo "global_conf.json" se configura el servidor donde se encuentra el gateway-bridge usando el parámetro "server_address". Si se encuentra en el mismo Raspberry Pi que el Packet-forwarder se usa "localhost", sino con la dirección IP respectiva. También hay que actualizar los parámetros para el "gateway_ID" obtenido al final del proceso al instalar el packet-forwarder.
"gateway_conf": {
"gateway_ID": "3532363324003700",
/* change with default server address/ports, or overwrite in local_conf.json */
"server_address": "192.168.10.50",
"serv_port_up": 1700,
"serv_port_down": 1700,
/* adjust the following parameters for your network */
"keepalive_interval": 10,
"stat_interval": 30,
"push_timeout_ms": 100,
/* forward only valid packets */
"forward_crc_valid": true,
"forward_crc_error": false,
"forward_crc_disabled": false
}
Conexión a TTN
El archivo "global_conf.json" se configura para un servidor TTN de la regíon, revisar los datos apropiador para "gateway_ID" y "server_address".
"gateway_conf": {
"gateway_ID": "3532363324003700",
/* change with default server address/ports, or overwrite in local_conf.json */
"server_address": "router.us.thethings.network",
"serv_port_up": 1700,
"serv_port_down": 1700,
/* adjust the following parameters for your network */
"keepalive_interval": 10,
"stat_interval": 30,
"push_timeout_ms": 100,
/* forward only valid packets */
"forward_crc_valid": true,
"forward_crc_error": false,
"forward_crc_disabled": false
}
Durante la implementación realizada, no se disponía del adaptador por lo que se probó construir un adaptador usando una placa perforada, teniendo los mismos resultados que con la placa de HELTEC.
El Kit de conexión del fabricante como referencia se muestra a continuacion:
El esquema abierto para un gateway LoRa de bajo costo, desagrega e interconecta componentes de hardware y software.
El mini-gateway es modular, el componente de software para la gestión de gateways y paquete de datos se implementa sobre un Raspberry Pi, conectado por Ethernet a la red local y con dirección IP fija.
En el manejo de software se prioriza integrar la gestión de dispositivos usando mensajes MQTT y de esta manera simplificar la integración al broker del esquema IoT general.
Componentes
El punto de partida la propuesta es gateway entre LoRa y Ethernet/Wifi. El fabricante Heltec presenta un "mini-Gateway" con el Módulo HT-M01. El módulo de hardware se conecta por medio del software "Packet-forwarder" (en un Raspbery Pi) hacia un administrador de gateways que puede estar en la red local (ChirpStack) o en la nube (The Things Network).
La conexión del módulo HT-M01 se puede realizar con SPI usando una placa de conexión hacia el Raspberry Pi. Si no se tiene la placa, también se la puede construir siguiendo las instrucciones en:
El proceso de instalación del Raspberry Pi se encuentra descrito en la Raspberry Pi OS-Instalar.
Packet-forwarder se instala siguiendo las instrucciones del fabricante.
Inicialmente se usó USB como conexión del módulo Heltec HT-M01, luego se usó SPI solo para comprobar las modalidades de implementación. Se utiliza SPI en la versión de operación regular.
Conexión entre componentes
módulo Heltec HT-M01 y Raspberry, SPI o USB
Ethernet desde la Raspberry Pi , usando dirección fija
La conexión Ethernet facilita la comunicación con el esquema existente y en operación, facilitando la ubicación de los componentes de software en otros "servidores" en los Raspberry Pi.
Hardware con Raspberry pi Zero y adaptador Ethernet
Para el caso de usar mas de un Gateway LoRa con Raspberry Pi Zero que no tiene conector Ethernet, se requiere un adaptador USB a Ethernet.
Algunos adaptadores USB a Ethernet "económicos" tienen la misma dirección Mac que al utilizar varios en una red local (todos en el mismo segmento) genera inconvenientes en la comunicación.
La dirección MAC se puede revisar con la instrucción:
Esta es la última sección a realizar, pues se supone que tiene listo el dispositivo, construido y operativas las partes: Dispositivo, Gateway y Broker.
la visualización de los valores requiere declarar los dispositivos en Home Assistant en el archivo configuration.yaml
1. Incorporar el dispositivo en Home Assistant
Se requiere modificar el archivo configuration.yaml en el raspberry.
Se puede realizar en forma local desde el raspberry que tiene monitor, teclado y mouse conectado, y editar el archivo que se encuentra en el directorio:
Para el envío de los mensajes hacia el broker MQTT y Home Assistant se usa un dispositivo configurado como gateway.
Se usa un dispositivo en lugar de un concentrador en la etapa de prototipo considerando los costos involucrados. La próxima tarea es desarrollar el gateway usando un concentrador, cuyo valor es más alto.
/* Gateway LoRa Lora/Wifi/MQTT/Home-Assistant Envia información por red Lora/Gateway WiFi hacia un broker MQTT y gestionar datos en Home-Assistant edelros@espol.edu.echttp://blog.espol.edu.ec/edelros/ Referencia: Aaron.Lee www.heltec.cnhttps://github.com/Heltec-Aaron-Lee/WiFi_Kit_series*/#include"heltec.h"#include<WiFi.h>#include<PubSubClient.h>// DISPOSITIVO LORA Banda ISM en Región 915Mhz#defineBAND915E6//433E6,868E6,915E6// ranges from 6-12,default 7 see API docsbytespread_factor=8;// LoRa Mensaje a enviar por direccionesStringpaqueteEnv="";bytedir_local=0xC1;// Concentrador 1bytedir_destino=0xD1;// Dispositivo 1bytemsjContador=0;// identificador de mensaje// tiempo entre lecturaslongt_anterior=0;intt_intervalo=4000;// LoRa Mensaje Recibidobytedir_envio=0xC1;// Concentrador 1intdir_remite=0xD0;// Inicia RemitenteStringpaqueteRcb="";bytepaqrcbID=0;bytepaqrcbEstado=0;// 0:vacio, 1: nuevo, 2:incompleto// 3:otro destinatario, 4:Broadcast// Mensajes por Puerto Serialvolatilebooleanserial_msj=true;// WIFI: conexión a Routerchar*ssid="giotirni20";char*password="Anera2020@";// MQTT: Servidorchar*MQTT_IP="192.168.10.50";uint16_tMQTT_puerto=1883;char*MQTT_usuario="usuarioprueba";char*MQTT_contrasena="usuarioclave";// MQTT: Dispositivo Sensorchar*MQTT_ID="LoraGatewayC1";charMQTT_TOPIC_T[50]="invernadero/loraD1/temperatura";charMQTT_TOPIC_H[50]="invernadero/loraD1/humedad";charMQTT_TOPIC_V[50]="invernadero/loraD1/voltaje";charMQTT_SensorEstado[10]="OFF";volatilebooleanmqtt_desconectado=true;// MQTT: Dispositivo Actuadorchar*MQTT_COMMAND="invernadero/loraD1/cambia";charMQTT_ActuadorEstado[10]="OFF";volatilebooleanactuador_estado=false;volatilebooleanactuador_bandera=false;chartemperatura[10]="00.00";charhumedad[10]="00.00";charvoltaje[10]="00.00";// Clientes WiFi y MQTTWiFiClientwificlient;PubSubClientmqttclient(wificlient);voidsetup(){Heltec.begin(false/*DisplayEnable Enable*/,true/*Heltec.Heltec.Heltec.LoRa Disable*/,serial_msj/*Serial Enable*/,true/*PABOOST Enable*/,BAND/*long BAND*/);// ranges from 6-12,default 7 see API docsLoRa.setSpreadingFactor(spread_factor);//LoRa.onReceive(cbk);LoRa.receive();// conexión WIFI y MQTTinicia_wifi();if(WiFi.status()==WL_CONNECTED){inicia_mqtt();}}voidloop(){// parametros de recepciónintrssi_lora=0;intsnr_lora=0;// Revisa mensajes LoRa entrantesintmsjRcbLoRa=LoRa.parsePacket();if(msjRcbLoRa!=0){recibirlora(msjRcbLoRa);rssi_lora=LoRa.packetRssi();snr_lora=LoRa.packetSnr();if(serial_msj==true){Serial.println("remite,msjID,mensaje,estado,Rssi,Snr");Serial.print(String(dir_remite,HEX));Serial.print(",");Serial.print(paqrcbID);Serial.print(",");Serial.print(paqueteRcb);Serial.print(",");Serial.print(paqrcbEstado);Serial.print(",");Serial.print(rssi_lora);Serial.print(",");Serial.println(snr_lora);}yield();// procesa wifi// LED parpadea Rebibido LoradigitalWrite(LED,HIGH);delay(50);digitalWrite(LED,LOW);delay(50);digitalWrite(LED,HIGH);delay(50);digitalWrite(LED,LOW);yield();// procesa wifidelay(100);}// Procesa a MQTT mensaje completoif(msjRcbLoRa!=0&&paqrcbEstado==1){// Separa parámetrosStringt=paqueteRcb.substring(1,6);Stringh=paqueteRcb.substring(8,10);Stringv=paqueteRcb.substring(12);// procesa tópico MQTT// añade dispositivoStringtopico="invernadero/lora";Stringremite=String(dir_remite,HEX);remite.toUpperCase();topico=topico+remite;// procesa topicoStringtopicot=topico+"/temperatura";Stringtopicoh=topico+"/humedad";Stringtopicov=topico+"/voltaje";topicot.toCharArray(MQTT_TOPIC_T,topicot.length()+1);topicoh.toCharArray(MQTT_TOPIC_H,topicoh.length()+1);topicov.toCharArray(MQTT_TOPIC_V,topicov.length()+1);t.toCharArray(temperatura,t.length()+1);h.toCharArray(humedad,h.length()+1);v.toCharArray(voltaje,v.length()+1);Serial.println(topicov);publica_estado();}yield();// procesa wifidelay(20);// reenviar a dispositivoif(actuador_bandera==true){msjContador=msjContador+1;enviarlora(dir_destino,dir_local,msjContador,paqueteEnv);actuador_bandera=false;}yield();// procesa wifidelay(20);if(WiFi.status()!=WL_CONNECTED){inicia_wifi();}else{if(mqttclient.connected()==false){mqtt_desconectado=true;inicia_mqtt();// reintento}if(mqttclient.connected()==true){if(mqtt_desconectado==true){publica_estado();mqtt_desconectado=false;}mqttclient.loop();}}yield();// procesa wifi}voidenviarlora(bytedestino,byteremite,bytepaqueteID,Stringpaquete){// espera que el radio esté listo// para enviar un paquetewhile(LoRa.beginPacket()==0){if(serial_msj==true){Serial.println("Esperando radio disponible...");}yield();// procesa wifidelay(100);}// envio del mensaje LoRaLoRa.beginPacket();LoRa.write(destino);LoRa.write(remite);LoRa.write(paqueteID);LoRa.write(paquete.length());LoRa.print(paquete);LoRa.endPacket();}voidrecibirlora(inttamano){if(tamano==0){paqrcbEstado=0;//vacioreturn;}// lectura de paquetepaqueteRcb="";dir_envio=LoRa.read();dir_remite=LoRa.read();paqrcbID=LoRa.read();bytepaqrcbTamano=LoRa.read();while(LoRa.available()){paqueteRcb+=(char)LoRa.read();}if(paqrcbTamano!=paqueteRcb.length()){paqrcbEstado=2;// Tamaño incompletoreturn;}if(dir_envio!=dir_local){paqrcbEstado=3;// otro destinoreturn;}if(dir_envio==0xFF){paqrcbEstado=4;// Broadcastreturn;}paqrcbEstado=1;// mensaje Nuevo}voidinicia_mqtt(void){intintentosmqtt=5;intcuentamqtt=0;if(serial_msj){Serial.print(" MQTT Conectando a ");Serial.println(MQTT_IP);}mqttclient.setServer(MQTT_IP,MQTT_puerto);mqttclient.connect(MQTT_ID,MQTT_usuario,MQTT_contrasena);mqttclient.setCallback(recibirmqtt);while(!mqttclient.connected()&&(cuentamqtt<=intentosmqtt)){if(serial_msj){Serial.print(".");}cuentamqtt=cuentamqtt+1;// LED Monitor parpadeo MQTTdigitalWrite(LED,HIGH);delay(200);digitalWrite(LED,LOW);delay(200);}if(mqttclient.connected()){publica_estado();}if(serial_msj){//Fin de "...."Serial.println();Serial.print(" MQTT Conectado: ");Serial.print(mqttclient.connected());Serial.print("\t MQTT Estado: ");Serial.println(mqttclient.state());}}voidpublica_estado(){if(mqttclient.connected()==true){mqttclient.publish(MQTT_TOPIC_T,temperatura,true);mqttclient.publish(MQTT_TOPIC_H,humedad,true);mqttclient.publish(MQTT_TOPIC_V,voltaje,true);mqttclient.subscribe(MQTT_COMMAND);}else{mqtt_desconectado=true;}}// llega mensaje MQTT, callback mqttvoidrecibirmqtt(char*p_topic,byte*p_payload,unsignedintp_length){Serial.println("un mensaje mqtt");Serial.println(p_topic);// convierte a textoStringpayload;for(uint8_ti=0;i<p_length;i++){payload.concat((char)p_payload[i]);}// String dispositivo = p_topic[16] + String(p_topic[17]);chardispositivo[3]="D0";dispositivo[1]=p_topic[17];dir_destino=(int)strtol(dispositivo,NULL,16);paqueteEnv=payload;actuador_bandera=true;if(mqttclient.connected()==true){mqttclient.subscribe(MQTT_COMMAND);}else{mqtt_desconectado=true;}}voidinicia_wifi(void){intintentoswifi=10;intcuentawifi=0;if(serial_msj){Serial.print(" WiFi Conectando a ");Serial.println(ssid);}WiFi.disconnect(true);delay(1000);WiFi.mode(WIFI_STA);WiFi.setAutoConnect(true);WiFi.begin(ssid,password);delay(100);while(WiFi.status()!=WL_CONNECTED&&cuentawifi<intentoswifi){if(serial_msj){Serial.print(".");}cuentawifi=cuentawifi+1;// Parpadeo de Monitor WifidigitalWrite(LED,HIGH);delay(300);digitalWrite(LED,LOW);delay(200);}if(serial_msj){// mensaje a serialSerial.println();if(WiFi.status()==WL_CONNECTED){Serial.print(" Estado: ");Serial.println(WiFi.status());Serial.print(" MAC: ");Serial.println(WiFi.macAddress());Serial.print(" IP: ");Serial.println(WiFi.localIP());Serial.print(" RSSI: ");Serial.println(WiFi.RSSI());Serial.println();}if(WiFi.status()!=WL_CONNECTED){WiFi.printDiag(Serial);Serial.println();}}}
Para facilitar la programación, se separan en funciones las acciones para enviar y recibir mensajes Lora y las de manejo de sensor/actuador.
El sensor DTH-11 se conecta al pin 13
El sensor de bateria se conecta al pin 36 como entrada analógica.
/* Dispositivo Sensor Temperatura y Humedad con DHT11 Broker: MQTT/Home-Assistant Red ruta: LoRa/WiFi/Ethernet edelros@espol.edu.echttp://blog.espol.edu.ec/edelros/ Referencia: Ejemplos de Aaron.Lee www.heltec.cn*/#include"heltec.h"#include"DHT.h"// Sensor de Temperatura&Humedad#defineDHTPIN13#defineDHTTYPEDHT11DHTdht(DHTPIN,DHTTYPE);Stringtemperatura="";Stringhumedad="";// Sensor de Bateria# defineBattPIN36StringbattNivel;//Banda LoRa 915Mhz - ISM en Región #defineBAND915E6// 433E6,868E6,915E6// ranges from 6-12,default 7 see API docsbytespread_factor=8;// Mensaje a enviar por direccionesbytedir_local=0xD1;// Dispositivo 1bytedir_destino=0xC1;// Concentrador 1// identificador de mensajebytemsjContador=0;// tiempo entre lecturaslongt_anterior=0;intt_intervalo=5000;// Mensaje Recibidobytedir_envio=0xC1;// Concentrador 1intdir_remite=0xD0;// Inicia RemitenteStringpaqueteRcb="";bytepaqrcbvID=0;bytepaqrcbEstado=0;// 0:vacio, 1: nuevo, 2:incompleto// 3:otro destinatario, 4:Broadcast// Mensajes por Puerto Serialvolatilebooleanserial_msj=true;voidsetup(){Heltec.begin(false/*DisplayEnable Enable*/,true/*Heltec.Heltec.Heltec.LoRa Disable*/,serial_msj/*Serial Enable*/,true/*PABOOST Enable*/,BAND/*long BAND*/);// ranges from 6-12,default 7 see API docsLoRa.setSpreadingFactor(spread_factor);//LoRa.onReceive(cbk);LoRa.receive();//inicializa sensorespinMode(DHTPIN,INPUT);dht.begin();pinMode(BattPIN,INPUT);}voidloop(){// Enviar mensajes entre intervaloslongt_ahora=millis();longt_transcurrido=t_ahora-t_anterior;// parametros de recepciónintrssi_lora=0;intsnr_lora=0;if(t_transcurrido>=t_intervalo){sensorLeeDHT();//actualiza estado del sensorsensorBateria();// actualiza estado de bateria// Construye paquete a enviarStringpaqueteEnv="";paqueteEnv=paqueteEnv+"t"+temperatura;paqueteEnv=paqueteEnv+"|";paqueteEnv=paqueteEnv+"h"+humedad;paqueteEnv=paqueteEnv+"|";paqueteEnv=paqueteEnv+"v"+battNivel;enviarlora(dir_destino,dir_local,msjContador,paqueteEnv);msjContador=msjContador+1;// parametros de recepciónrssi_lora=LoRa.packetRssi();snr_lora=LoRa.packetSnr();yield();// mensaje a serialif(serial_msj==true){Serial.print(String(dir_destino,HEX));Serial.print(",");Serial.print(String(dir_local,HEX));Serial.print(",");Serial.print(msjContador);Serial.print(",");Serial.print(paqueteEnv.length());Serial.print(",");Serial.print(paqueteEnv);Serial.print(",");Serial.print(rssi_lora);Serial.print(",");Serial.println(snr_lora);}t_anterior=millis();t_intervalo=3000+random(2000);// LED parpadea envio loradigitalWrite(LED,HIGH);delay(100);digitalWrite(LED,LOW);delay(100);yield();// procesa wifi}// Revisar mensajes LoRa entrantesintmsjRcbLoRa=LoRa.parsePacket();if(msjRcbLoRa!=0){recibirlora(msjRcbLoRa);rssi_lora=LoRa.packetRssi();snr_lora=LoRa.packetSnr();if(serial_msj==true){if(paqrcbEstado==1){Serial.println("Mensaje: "+paqueteRcb);Serial.println("RSSI: "+String(rssi_lora));Serial.println("Snr: "+String(snr_lora));Serial.println();}else{Serial.print("Paquete recibido Estado: ");Serial.println(paqrcbEstado);}}yield();// procesa wifi// LED parpadea Recibido LoradigitalWrite(LED,HIGH);delay(50);digitalWrite(LED,LOW);delay(50);digitalWrite(LED,HIGH);delay(50);digitalWrite(LED,LOW);}delay(100);yield();// procesa wifi}voidenviarlora(bytedestino,byteremite,bytepaqueteID,Stringpaquete){// espera que el radio esté listo// para enviar un paquetewhile(LoRa.beginPacket()==0){if(serial_msj==true){Serial.println("Esperando radio disponible...");}yield();// procesa wifidelay(100);}// envio del mensaje LoRaLoRa.beginPacket();LoRa.write(destino);LoRa.write(remite);LoRa.write(paqueteID);LoRa.write(paquete.length());LoRa.print(paquete);LoRa.endPacket();}voidrecibirlora(inttamano){if(tamano==0){paqrcbEstado=0;//vacioreturn;}// lectura de paquetepaqueteRcb="";dir_envio=LoRa.read();dir_remite=LoRa.read();paqrcbvID=LoRa.read();bytepaqrcbTamano=LoRa.read();while(LoRa.available()){paqueteRcb+=(char)LoRa.read();}if(paqrcbTamano!=paqueteRcb.length()){paqrcbEstado=2;// Tamaño incompletoreturn;}if(dir_envio!=dir_local){paqrcbEstado=3;// otro destinoreturn;}if(dir_envio==0xFF){paqrcbEstado=4;// Broadcastreturn;}paqrcbEstado=1;// mensaje Nuevo}// Sensor lecturasvoidsensorLeeDHT(){humedad=String(int(dht.readHumidity())).c_str();temperatura=String(dht.readTemperature()).c_str();}voidsensorBateria(){intlectura=analogRead(BattPIN);// convierte a equivalente en voltiosfloatvoltaje=(float(lectura)/4096.0)*4.2*3.0/2.0;battNivel=String(voltaje).c_str();}
El primer prototipo se realiza usando una placa de desarrollo Heltec Lora 32. EL punto de partida es LoRa multipunto añadiendo los siguientes componentes:
Sensor de Temperatura y Humedad DHT-11, puede ser actualizado a DHT-22 para mayor precisión. Se empieza con DHT-11 por tenerlo disponible al inicio.
Bateria para añadir portabilidad
Panel solar para alimentación, usando un módulo de carga de batería.
El sensor numérico más sencillo de implementar es del de temperatura-Humedad con el sensor DHT-11 o DHT-22. Un requerimiento para el dispositivo es operar a batería, con opción de carga con un panel solar.
El uso de la batería limita el consumo de energía, los módulos LoRa al ser de bajo consumo son los seleccionados.
El punto de partida es la configuración LoRa Multipunto, que modificando e valor del sensor enviado y los elementos correspondientes en MQTT, permite visualizar en la página del broker Home-assistant el valor del sensor.
Usando el ejemplo de comunicación multipunto, se revisa la secuencia de paquetes (msjID) enviada por cada uno de los nodos/dispositivos con mensajes que se envían con intervalos aleatorios entre [2-4 segundos].
La base de tiempo de 2 segundos se considera como tiempo que toma un sensor de temperatura DHT11 en dar una nueva lectura.
Para la lectura de los datos desde el gateway se usa la comunicación por puerto serial (USB). de los datos de cada mensaje se usa el identificador de mensaje (msjID) que indica el orden del mensaje enviado.
Ejemplo de mensaje obtenido por puerto serial desde el "gateway simple"
Usando como factor LoRa.setSpreadingFactor(8), se reduce la tasa de errores desde 0.36 en modo predeterminado de los dispositivos.
Ejemplo de resultados obtenidos.
ID [1 2]
cuenta [1879 1871]
errores [254 312]
increm [1 1]
antes [190 173]
%error [0.14 0.17]
Instrucciones en Python
# prueba de recepción de mensajes
# en Gateway LoRa mutipunto
# edelros@espol.edu.ec
import numpy as np
import serial, time
# INGRESO
puerto = 'com8'
baudios = 115200
n = 2
encabezado =['ID','cuenta',
'errores','increm',
'antes']
m = len(encabezado)
tabla = np.zeros(shape=(n+1,m),dtype=int)
d_error = np.zeros(n+1,dtype=float)
for f in range(1,n+1,1):
tabla[f,0]=f
# PROCEDIMIENTO
arduino = serial.Serial(puerto, baudios)
arduino.setDTR(False)
time.sleep(0.3)
# limpia buffer de datos anteriores
arduino.flushInput()
arduino.setDTR()
time.sleep(0.3)
print('\nEstado del puerto: ',arduino.isOpen())
print('Nombre del dispositivo conectado: ', arduino.name)
print('Dump de la configuración:\n ',arduino)
print('\n###############################################\n')
np.set_printoptions(precision=2)
# Lectura de datos
while True:
# espera hasta recibir un dato
while (arduino.inWaiting()==0):
pass
# lee binario del puerto serial
lectura = arduino.readline()
# binario a texto, elimina /r/n
texto = lectura.decode().strip()
print(texto)
tamano = len(texto)
if tamano>=3:
if (texto[0]=='d' and texto[2]==','):
partes = texto.split(',')
msjID = int(partes[0][1])
# incremento
antes = tabla[msjID,4]
ahora = int(partes[1])
tabla[msjID,4] = ahora
incremento = ahora - antes
tabla[msjID,3] = incremento
# cuenta
if antes>0 and incremento>0:
tabla[msjID,1]=tabla[msjID,1] + incremento
# error
if (tabla[msjID,1]>1 and incremento>1):
tabla[msjID,2] = tabla[msjID,2]+incremento-1
if (tabla[msjID,1]>0):
d_error[msjID]=float(tabla[msjID,2])/tabla[msjID,1]
for i in range(0,m,1):
print(encabezado[i]+"\t",tabla[1:,i])
print("%error \t",d_error[1:])
# Cerrar el puerto serial.
serial.Serial.close