From ec1ac5ffad0147e7020f839b44f9d58eadf0ee57 Mon Sep 17 00:00:00 2001 From: Jeremy Ellis Date: Wed, 8 Jun 2022 20:41:34 -0700 Subject: [PATCH 1/7] Create us915-helium-lorawan-http-integration.ino --- .../us915-helium-lorawan-http-integration.ino | 203 ++++++++++++++++++ 1 file changed, 203 insertions(+) create mode 100644 examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino diff --git a/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino b/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino new file mode 100644 index 0000000..bbdb9fe --- /dev/null +++ b/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino @@ -0,0 +1,203 @@ +/* +* Helium Send And Receive +* +* For Adafruit non-CayeenLPP connections for simple sensor Base 64 encoded +* +* In Helium "functions" section use this simple bsae64 decoder +* +//////////////////////////////////////////////////////////// + function Decoder(bytes, port, uplink_info) { + var decoded={}; + try{ + var result = String.fromCharCode.apply(null, bytes); + decoded.value = result; + return decoded; + } catch (err) { + return 'Decoder: ' + err.name + " : " + err.message;; + } + } +//////////////////////////////////////////////////////////// +* +* +* +* And in the integration section of helium use this JSON +* +//////////////////////////////////////////////////////////// + { "value": {{decoded.payload.value}} } +//////////////////////////////////////////////////////////// +* +* +* +* This sketch demonstrates how to send and receive data with the PortentaH7 LoRa module. +* This example code is in the public domain. +* Note: Helium must be setup for what it does with the data +*/ + +#include // Only needed by https://platformio.org/ +#include + + +LoRaModem modem; +bool connected = false; +bool myWaitForDownlink = false; +bool myDownLink = false; + +unsigned long myStoredMillisA; +unsigned long myStoredMillisB; +const unsigned long myTimerDurationA = 30000; // delay between sending data +const unsigned long myTimerDurationB = 5000; // delay to wait for a downlink + + +// Please enter your sensitive data in the arduino_secrets.h tab +// Note: Best to have the App_Device hard coded. Run the program once to see the value. +#include "arduino_secrets.h" + + +String appEui = SECRET_APP_EUI; // set these in your arduino_secrets.h tab +String appKey = SECRET_APP_KEY; + + +void setup() +{ + // put your setup code here, to run once: + Serial.begin(115200); + pinMode(LEDR,OUTPUT); + pinMode(LEDG,OUTPUT); + pinMode(LEDB,OUTPUT); + digitalWrite(LEDR, HIGH); // new boards HIGH = off + digitalWrite(LEDG, LOW); + digitalWrite(LEDB, HIGH); + + //while (!Serial); // don't wait for serial + + Serial.println("Wait 4"); + delay(3000); // delay instead, so it works when disconnected + digitalWrite(LEDG, HIGH);// allows time to connect serial monitor + + Serial.println("Wait 3"); + delay(3000); + digitalWrite(LEDG, LOW); + + Serial.println("Wait 2"); + delay(3000); + digitalWrite(LEDG, HIGH); + + Serial.println("Wait 1"); + delay(3000); + digitalWrite(LEDG, LOW); + delay(3000); + + // change this to your regional band (eg. US915, AS923, EU868, AU915, AS923, KR920, IN865 ...) + if (!modem.begin(US915)) { + Serial.println("Failed to start module"); + while (1) {} // since error keep program here + }; + + Serial.print("Your module version is: "); + Serial.println(modem.version()); + Serial.print("Your device EUI is: "); + Serial.println(modem.deviceEUI()); + + Serial.println("Now Disabling all channels and enable channel 1 only for Helium "); + + modem.disableChannel(0); + modem.enableChannel(1); // only one enabled for Helium + modem.disableChannel(2); + modem.disableChannel(3); + modem.disableChannel(4); + modem.disableChannel(5); + modem.disableChannel(6); + delay(5000); + Serial.println("Now Joining the Helium Network "); + +} + +void loop() +{ + while (!connected) { + Serial.println("trying to reconnect"); + digitalWrite(LEDR, HIGH); // new boards HIGH = off + digitalWrite(LEDG, LOW); + digitalWrite(LEDB, LOW); + connected = modem.joinOTAA(appEui, appKey); + delay(5000); + digitalWrite(LEDR, HIGH); // new boards HIGH = off + digitalWrite(LEDG, LOW); + digitalWrite(LEDB, HIGH); + delay(1000); + } + + if (myWaitForDownlink){ + + char rcv[64]; + int i = 0; + while (modem.available()) { + rcv[i++] = (char)modem.read(); + myDownLink = true; + } + + if (millis() - myStoredMillisB >= myTimerDurationB){ // Test whether the period has elapsed + myStoredMillisB = millis(); + if (!modem.available()) { + Serial.println("No downlink message received at this time."); + myWaitForDownlink = false; + } + } + + + if (myDownLink) { + myWaitForDownlink = false; + myDownLink = false; + Serial.print("Received: "); + Serial.write(rcv, i); + // for (unsigned int j = 0; j < i; j++) { + // Serial.print(rcv[j] >> 4, HEX); + // Serial.print(rcv[j] & 0xF, HEX); + // Serial.print(" "); + // } + Serial.println(); + digitalWrite(LEDR, LOW); // new boards HIGH = off + digitalWrite(LEDG, HIGH); + digitalWrite(LEDB, LOW); + } + } + + if (millis() - myStoredMillisA >= myTimerDurationA){ // Test whether the period has elapsed + myStoredMillisA = millis(); // IMPORTANT to save the next stored time + + String msg = String(analogRead(A0)); // program expecting numbers! + + Serial.println(); + Serial.print("Sending: " + msg + " - "); + for (unsigned int i = 0; i < msg.length(); i++) { + Serial.print(msg[i] >> 4, HEX); + Serial.print(msg[i] & 0xF, HEX); + Serial.print(" "); + } + Serial.println(); + + int err; + modem.beginPacket(); + modem.print(msg); + err = modem.endPacket(true); + if (err > 0) { + Serial.println("Message sent correctly!"); + digitalWrite(LEDR, HIGH); // new boards HIGH = off + digitalWrite(LEDG, HIGH); + digitalWrite(LEDB, LOW); + } else { + Serial.println("Error sending message :("); + Serial.println("(you may send a limited amount of messages per minute, depending on the signal strength"); + Serial.println("it may vary from 1 message every couple of seconds to 1 message every minute)"); + digitalWrite(LEDR, LOW); // new boards HIGH = off + digitalWrite(LEDG, LOW); + digitalWrite(LEDB, HIGH); + } + myStoredMillisB = millis(); + myWaitForDownlink = true; + // delay(30000); // delay 30 seconds for testing + + } // end timerA + +} + From a5d92fba5492f04a9d410a00ed1e6ebaa69cc9ef Mon Sep 17 00:00:00 2001 From: Jeremy Ellis Date: Wed, 8 Jun 2022 20:42:20 -0700 Subject: [PATCH 2/7] Create arduino_secrets.h --- .../us915-helium-lorawan-http-integration/arduino_secrets.h | 4 ++++ 1 file changed, 4 insertions(+) create mode 100644 examples/us915-helium-lorawan-http-integration/arduino_secrets.h diff --git a/examples/us915-helium-lorawan-http-integration/arduino_secrets.h b/examples/us915-helium-lorawan-http-integration/arduino_secrets.h new file mode 100644 index 0000000..9990d81 --- /dev/null +++ b/examples/us915-helium-lorawan-http-integration/arduino_secrets.h @@ -0,0 +1,4 @@ +// Note: Do not copy the Device-EUI from the Helium Console, it must come from the Portenta +// Run the program once to get the Portenta hard coded Device-EUI +#define SECRET_APP_EUI "0000000000000000" // Copied from Helium Console Devices App-EUI +#define SECRET_APP_KEY "00000000000000000000000000000000" // Copied from Helium Console Devices App-Key From 3ad40cb7e7eaee5db08922476b624a696f4c5867 Mon Sep 17 00:00:00 2001 From: Jeremy Ellis Date: Wed, 8 Jun 2022 20:50:02 -0700 Subject: [PATCH 3/7] Update us915-helium-lorawan-http-integration.ino --- .../us915-helium-lorawan-http-integration.ino | 13 ++++++++++--- 1 file changed, 10 insertions(+), 3 deletions(-) diff --git a/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino b/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino index bbdb9fe..62f480d 100644 --- a/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino +++ b/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino @@ -1,9 +1,9 @@ /* * Helium Send And Receive * -* For Adafruit non-CayeenLPP connections for simple sensor Base 64 encoded +* For Helium and Adafruit.io http integrations sensor data is Base 64 encoded * -* In Helium "functions" section use this simple bsae64 decoder +* In Helium "functions" section use this simple base64 decoder * //////////////////////////////////////////////////////////// function Decoder(bytes, port, uplink_info) { @@ -30,7 +30,14 @@ * * This sketch demonstrates how to send and receive data with the PortentaH7 LoRa module. * This example code is in the public domain. -* Note: Helium must be setup for what it does with the data +* By Jeremy Ellis Twitter @rocksetta +* Helium Video Playlist at +* https://www.youtube.com/watch?v=wt_WZ1sBDeM&list=PL57Dnr1H_egu0PkIBAbbdfQ21O_NJqJg8&index=4 +* +* Helium Arduino Documnetation at +* https://docs.helium.com/use-the-network/devices/development/arduino/lora-vision-shield/arduino/ +* +* Note: Helium and Adafruit.io must be setup for what it does with the data */ #include // Only needed by https://platformio.org/ From bebb5c255ed993ae0e7f4bff044616dd7b0a93c5 Mon Sep 17 00:00:00 2001 From: Jeremy Ellis Date: Wed, 8 Jun 2022 22:14:43 -0700 Subject: [PATCH 4/7] Create us915-helium-lorawan-mqtt-cayenneLPP.ino --- .../us915-helium-lorawan-mqtt-cayenneLPP.ino | 223 ++++++++++++++++++ 1 file changed, 223 insertions(+) create mode 100644 examples/us915-helium-lorawan-mqtt-cayenneLPP/us915-helium-lorawan-mqtt-cayenneLPP.ino diff --git a/examples/us915-helium-lorawan-mqtt-cayenneLPP/us915-helium-lorawan-mqtt-cayenneLPP.ino b/examples/us915-helium-lorawan-mqtt-cayenneLPP/us915-helium-lorawan-mqtt-cayenneLPP.ino new file mode 100644 index 0000000..6ded5e1 --- /dev/null +++ b/examples/us915-helium-lorawan-mqtt-cayenneLPP/us915-helium-lorawan-mqtt-cayenneLPP.ino @@ -0,0 +1,223 @@ +/* +* Helium Send And Receive +* +* For Adafruit CayenneLPP connections using MQTT +* +* In Helium "functions" section use this simple bsae64 decoder +* You simple need to use the CayenneLPP pre-set decoder. +* No other settings are needed +* The only negative is that you can't change the feed names that arrive at Adafruit.io +* +* +* This sketch demonstrates how to send and receive data with the PortentaH7 LoRa module. +* This example code is in the public domain. +* By Jeremy Ellis Twitter @rocksetta +* +* Helium Video Playlist at +* https://www.youtube.com/watch?v=wt_WZ1sBDeM&list=PL57Dnr1H_egu0PkIBAbbdfQ21O_NJqJg8&index=4 +* +* Helium Arduino Documnetation at +* https://docs.helium.com/use-the-network/devices/development/arduino/lora-vision-shield/arduino/ +* +* Note: Helium and Adafruit.io must be setup for what it does with the data +*/ + +#include // Only needed by https://platformio.org/ +#include +#include + + +LoRaModem modem; +CayenneLPP lpp(51); +bool connected = false; +bool myWaitForDownlink = false; +bool myDownLink = false; + +unsigned long myStoredMillisA; +unsigned long myStoredMillisB; +const unsigned long myTimerDurationA = 30000; // delay between sending data +const unsigned long myTimerDurationB = 5000; // delay to wait for a downlink + + + +// Please enter your sensitive data in the arduino_secrets.h tab +// Note: The Portenta has the App_Device hard coded. Run the program once to see the value. +#include "arduino_secrets.h" + + +String appEui = SECRET_APP_EUI; // just strings of the above +String appKey = SECRET_APP_KEY; + + +void setup() +{ + // put your setup code here, to run once: + Serial.begin(115200); + pinMode(LEDR,OUTPUT); + pinMode(LEDG,OUTPUT); + pinMode(LEDB,OUTPUT); + digitalWrite(LEDR, HIGH); // new boards HIGH = off + digitalWrite(LEDG, LOW); + digitalWrite(LEDB, HIGH); + + //while (!Serial); // don't wait for serial + + Serial.println("Wait 4"); + delay(3000); // delay instead, so it works when disconnected + digitalWrite(LEDG, HIGH);// allows time to connect the serial monitor + + Serial.println("Wait 3"); + delay(3000); + digitalWrite(LEDG, LOW); + + Serial.println("Wait 2"); + delay(3000); + digitalWrite(LEDG, HIGH); + + Serial.println("Wait 1"); + delay(3000); + digitalWrite(LEDG, LOW); + + // change this to your regional band (eg. US915, AS923, EU868, AU915, AS923, KR920, IN865 ...) + if (!modem.begin(US915)) { + Serial.println("Failed to start module"); + while (1) {} + }; + + Serial.print("Your module version is: "); + Serial.println(modem.version()); + Serial.print("Your device EUI is: "); + Serial.println(modem.deviceEUI()); + + Serial.println("Now Disabling all channels and enable channel 1 only for Helium "); + modem.disableChannel(0); + modem.enableChannel(1); // only one enabled for Helium + modem.disableChannel(2); + modem.disableChannel(3); + modem.disableChannel(4); + modem.disableChannel(5); + modem.disableChannel(6); + delay(5000); + Serial.println("Now Joining the Helium Network "); + +} + +void loop() +{ + while (!connected) { + Serial.println("trying to reconnect"); + digitalWrite(LEDR, HIGH); // new boards HIGH = off + digitalWrite(LEDG, LOW); + digitalWrite(LEDB, LOW); + connected = modem.joinOTAA(appEui, appKey); + delay(5000); + digitalWrite(LEDR, HIGH); // new boards HIGH = off + digitalWrite(LEDG, LOW); + digitalWrite(LEDB, HIGH); + delay(1000); + } + + if (myWaitForDownlink){ + + char rcv[64]; + int i = 0; + while (modem.available()) { + rcv[i++] = (char)modem.read(); + myDownLink = true; + } + + if (millis() - myStoredMillisB >= myTimerDurationB){ // Test whether the period has elapsed + myStoredMillisB = millis(); + if (!modem.available()) { + Serial.println("No downlink message received at this time."); + myWaitForDownlink = false; + } + } + + if (myDownLink) { + myWaitForDownlink = false; + myDownLink = false; + Serial.print("Received: "); + for (unsigned int j = 0; j < i; j++) { + Serial.print(rcv[j] >> 4, HEX); + Serial.print(rcv[j] & 0xF, HEX); + Serial.print(" "); + } + Serial.println(); + digitalWrite(LEDR, LOW); // new boards HIGH = off + digitalWrite(LEDG, HIGH); + digitalWrite(LEDB, LOW); + } + + } // end if WaitForDownlink + + + if (millis() - myStoredMillisA >= myTimerDurationA){ // Test whether the period has elapsed + myStoredMillisA = millis(); // IMPORTANT to save the next stored time + + lpp.reset(); + float x = analogRead(A0); //analogRead(A0) or rand() / 10000000.0; + lpp.addDigitalOutput(1, x); + + /* + // You can do any of these + // but not necessarily all of them + lpp.reset(); + lpp.addDigitalInput(1, 0); + lpp.addDigitalOutput(2, 1); + lpp.addAnalogInput(3, 1.23f); + lpp.addAnalogOutput(4, 3.45f); + lpp.addLuminosity(5, 20304); + lpp.addPresence(6, 1); + lpp.addTemperature(7, 26.5f); + lpp.addRelativeHumidity(8, 86.6f); + lpp.addAccelerometer(9, 1.234f, -1.234f, 0.567f); + lpp.addBarometricPressure(10, 1023.4f); + lpp.addGyrometer(1, -12.34f, 45.56f, 89.01f); + lpp.addGPS(1, -12.34f, 45.56f, 9.01f); + + lpp.addUnixTime(1, 135005160); + + lpp.addGenericSensor(1, 4294967295); + lpp.addVoltage(1, 3.35); + lpp.addCurrent(1, 0.321); + lpp.addFrequency(1, 50); + lpp.addPercentage(1, 100); + lpp.addAltitude(1 , 50); + lpp.addPower(1 , 50000); + lpp.addDistance(1 , 10.555); + lpp.addEnergy(1 , 19.055); + lpp.addDirection(1 , 90); + lpp.addSwitch(1 , 0); + + lpp.addConcentration(1 , 512); + lpp.addColour(1 , 64, 128, 255); + */ + + Serial.println(); + Serial.println("Sending:" + String(x, 1)); + + int err; + modem.beginPacket(); + modem.write(lpp.getBuffer(), lpp.getSize()); + err = modem.endPacket(true); + if (err > 0) { + Serial.println("Message sent correctly!"); + digitalWrite(LEDR, HIGH); // new boards HIGH = off + digitalWrite(LEDG, HIGH); + digitalWrite(LEDB, LOW); + } else { + Serial.println("Error sending message :("); + Serial.println("(you may send a limited amount of messages per minute, depending on the signal strength"); + Serial.println("it may vary from 1 message every couple of seconds to 1 message every minute)"); + digitalWrite(LEDR, LOW); // new boards HIGH = off + digitalWrite(LEDG, LOW); + digitalWrite(LEDB, HIGH); + } + myStoredMillisB = millis(); + myWaitForDownlink = true; + + // delay(30000); // delay 30 seconds for testing + + } // end timerA +} From 2d6993cdfd4844b3231beaae956940f8f68e167a Mon Sep 17 00:00:00 2001 From: Jeremy Ellis Date: Wed, 8 Jun 2022 22:15:13 -0700 Subject: [PATCH 5/7] Create arduino_secrets.h --- .../us915-helium-lorawan-mqtt-cayenneLPP/arduino_secrets.h | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 examples/us915-helium-lorawan-mqtt-cayenneLPP/arduino_secrets.h diff --git a/examples/us915-helium-lorawan-mqtt-cayenneLPP/arduino_secrets.h b/examples/us915-helium-lorawan-mqtt-cayenneLPP/arduino_secrets.h new file mode 100644 index 0000000..54ea7cb --- /dev/null +++ b/examples/us915-helium-lorawan-mqtt-cayenneLPP/arduino_secrets.h @@ -0,0 +1,5 @@ + +// Note: Do not copy the Device-EUI from the Helium Console, it must come from the Portenta +// Run the program once to get the Portenta hard coded Device-EUI +#define SECRET_APP_EUI "0000000000000000" // Copied from Helium Console Devices App-EUI +#define SECRET_APP_KEY "00000000000000000000000000000000" // Copied from Helium Console Devices App-Key From 1c40dd3f40f72539d617b1bb6c41165e76eca0d9 Mon Sep 17 00:00:00 2001 From: Jeremy Ellis Date: Wed, 8 Jun 2022 22:16:44 -0700 Subject: [PATCH 6/7] Update us915-helium-lorawan-http-integration.ino --- .../us915-helium-lorawan-http-integration.ino | 2 ++ 1 file changed, 2 insertions(+) diff --git a/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino b/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino index 62f480d..cc9284b 100644 --- a/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino +++ b/examples/us915-helium-lorawan-http-integration/us915-helium-lorawan-http-integration.ino @@ -3,6 +3,8 @@ * * For Helium and Adafruit.io http integrations sensor data is Base 64 encoded * +* Run the MKRWAN example MKRWANFWUpdate_standalone to make sure the LoRa module is reset +* * In Helium "functions" section use this simple base64 decoder * //////////////////////////////////////////////////////////// From 3b83b973266c97cd22951a49a4975665cfbe68a5 Mon Sep 17 00:00:00 2001 From: Jeremy Ellis Date: Wed, 8 Jun 2022 22:17:40 -0700 Subject: [PATCH 7/7] Update us915-helium-lorawan-mqtt-cayenneLPP.ino --- .../us915-helium-lorawan-mqtt-cayenneLPP.ino | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/examples/us915-helium-lorawan-mqtt-cayenneLPP/us915-helium-lorawan-mqtt-cayenneLPP.ino b/examples/us915-helium-lorawan-mqtt-cayenneLPP/us915-helium-lorawan-mqtt-cayenneLPP.ino index 6ded5e1..0f32efc 100644 --- a/examples/us915-helium-lorawan-mqtt-cayenneLPP/us915-helium-lorawan-mqtt-cayenneLPP.ino +++ b/examples/us915-helium-lorawan-mqtt-cayenneLPP/us915-helium-lorawan-mqtt-cayenneLPP.ino @@ -3,8 +3,9 @@ * * For Adafruit CayenneLPP connections using MQTT * -* In Helium "functions" section use this simple bsae64 decoder -* You simple need to use the CayenneLPP pre-set decoder. +* Run the MKRWAN example MKRWANFWUpdate_standalone to make sure the LoRa module is reset +* +* In Helium "functions" section you simply need to use the CayenneLPP pre-set decoder. * No other settings are needed * The only negative is that you can't change the feed names that arrive at Adafruit.io *