Тестирование радиомодемов LoRa/LoRaWAN RN2483. Часть 2, LoRaWAN

В предыдущей части рассказывалось, как подключить модемы RN2483 в режиме LoRa. Сейчас перейдем к следующей, более сложной части — подключению к сети LoRaWAN.

Что такое LoRaWAN?


fbc8214665cb4bfb94e1c75021ec502b.png
LoRaWAN — это «условно глобальная» сеть, состоящая из конечных устройств (nodes), отправляющих данные на концентраторы (gateway). Каждый концентратор имеет доступ к сети Интернет, и отправляет принятые данные на сервер, который в свою очередь рассылает их подписанным клиентам. Сеть с одной стороны, «глобальная», т.к. данные может принять любой, находящийся поблизости gateway, с другой стороны, она «условно глобальная», т.к. без интернета/интранета оно все равно работать не будет (хотя никто не мешает иметь собственный gateway и сервер внутри организации).

В качестве «узла» будет использоваться модем RN2483, для подключения будет использоваться бесплатный сервис www.thethingsnetwork.org.
Продолжение под катом.

Сеть The Things Network


Сервис www.thethingsnetwork.org — это открытое сообщество, пользователи которого могут создавать, регистрировать устройства и концентраторы, экспериментировать с тем как все это работает, и т.д. Как говорится, бесплатно и без SMS. Им мы для тестирования и воспользуемся.

Любой желающий может зарегистрировать в сети как node, так и gateway. Сеть The Things Network совершенно бесплатная, многие энтузиасты покупают оборудование за свой счет.
Карта покрытия на момент написания статьи выглядит следующим образом:
18f37b249adb417193d8a3c679287cf0.png

Как можно видеть, русскоязычный сегмент покрытия оставляет желать лучшего, но все же он есть. В этом есть и положительная сторона — в Лондоне или Амстердаме новым концентратором уже никого не удивишь, а в России — вполне есть шанс быть первым.

Подключение RN2483 к LoRaWAN


В отличие от простого p2p-соединения, в LoRaWAN все гораздо сложнее — устройства регистрируются в сети, траффик шифруется различными ключами, и так далее. Т.к. сеть является публичной и общедоступной, вопросы защиты данных здесь весьма актуальны.

В сети LoRaWAN существует 2 вида аутентификации:
— Аутентификация «по воздуху» (OTAA, Over-the-Air Activation). Устройства регистрируются в сети и получают ключ, необходимый для работы.
— Активация ABP (Activation by Personalization). Настройки заранее прописываются в устройстве, дополнительной аутентификации не требуется. Этот режим проще, однако есть и минус — данные просто отправляются в эфир, нет гарантии что они были получены концентратором.
Важно: многие дешевые одноканальные концентраторы поддерживают только режим ABP, так что первый способ может и не сработать.

1. Регистрация устройства — OTAA
Первым делом необходимо зайти на сайт staging.thethingsnetwork.org и зарегистрировать «приложение» (application). Я назвал его Raspberry Pi Home. При регистрации приложения юзер получает первый ключ, который пригодится в дальнейшем.
b338a779084444d193521ef6ce945d04.png
Далее мы видим вкладку Devices, куда можно внести наш модем (их может быть несколько). Нажимаем «register device» и выбираем режим OTAA. Появляется окно, в котором нужно ввести идентификатор девайса.
f23d5697934245b7abffe51bb831a4e1.png
Этот идентификатор можно получить, послав модему команду «sys get hweui». На этом регистрация закончена, как и было обещано, бесплатно и без SMS.
Открыв параметры девайса, можно увидеть все ключи, их необходимо прописать в программе.
9f08c48ff5a54bb7a8bc13086b1815ef.png

Для упрощения кода, я воспользовался библиотекой python-loranode, исходный код для отправки данных приведен ниже. Как можно видеть, в коде есть 3 строки appkey, appeui и deveui, которые используются для аутентификации.

Исходный код
from loranode import RN2483Controller
from commands import *
import time

# LoRaController OTAA based join and message test
if __name__ == "__main__":
    set_debug_level(Level.DEBUG)
    
    port = "COM3"
    appkey = "58FF3007CAED02xxxxxxxxxxxxxxxxxxxx"
    appeui = "70B3D57Exxxxxxxxxxxxxx"
    deveui = "0004A30xxxxxxxxxxx"

    # Test controller
    lc = RN2483Controller(port)
    if lc.test():
        printd("[+] Connected to LoRa RN2483 device", Level.INFO)
        lc.serial_sr(CMD_GET_VERSION)
        lc.serial_sr(CMD_GET_HWEUI)
        lc.get_freq()
    else:
        printd(clr(Color.YELLOW, "[-] Failed to get version from LoRa device"), Level.WARNING)

    lc.set_pwr(15)
    lc.set_adr(False)
    lc.serial_sr(CMD_SET_SF, "sf7") # sf12, sf7

    # Join and send a message
    if lc.join_otaa(appkey, appeui, deveui):
        printd("[+] Connected to gateway", Level.INFO)
        
        # Data-1
        printd("[+] Sending packet #1", Level.INFO)
        timeStr = time.strftime("%H%M", time.gmtime())
        if lc.send(timeStr, ack=False):
            printd(clr(Color.GREEN, "[+] Send-1 succeeded"), Level.CRITICAL)
        else:
            printd(clr(Color.RED, "[+] Send-1 failed"), Level.CRITICAL)
        time.sleep(15)

        # Data-2
        printd("[+] Sending packet #2", Level.INFO)
        timeStr = time.strftime("%H%M", time.gmtime())
        if lc.send(timeStr, ack=False):
          printd(clr(Color.GREEN, "[+] Send-2 succeeded"), Level.CRITICAL)
        else:
          printd(clr(Color.RED, "[+] Send-2 failed"), Level.CRITICAL)
        time.sleep(15)

        del lc
        exit()

    del lc
    printd(clr(Color.RED, "[-] Test failed"), Level.CRITICAL)



В этом примере на сервер отправляются пакеты, содержащие текущее время (ччмм). Это удобно для контроля того, какой пакет был принят.2. Регистрация устройства — ABP
Здесь все проще — регистрируем устройство как ABP, и получаем несколько ключей, которые необходимо прописать в программе.
7517d9eb74f44a0a8ab6c608c7adc63f.png
Модем отправляет данные «как есть», никакой гарантии что аутентификация была успешной, нет.
Для отправки данных, необходимо вставить в код ключи nwkskey, appskey, devaddr.
Исходный код приведен ниже.

Исходный код
from loranode import RN2483Controller
import platform
from commands import *
import time

# LoRaController ABP based join and ACK test
if __name__ == "__main__":
    set_debug_level(Level.DEBUG)

    port = "COM3"
    nwkskey = "58AA52E96035Axxxxxxxxxxxxxxxxxxxx"
    appskey = "381B1C9206E9BE9xxxxxxxxxxxxxxxxxxx"
    devaddr = "B639xxxx"
    
    lc = None
    
    try:
      # Test controller
      lc = RN2483Controller(port)
      if lc.test():
        printd("[+] Connected to LoRa RN2483 device", Level.INFO)

        lc.serial_sr(CMD_GET_VERSION)
        lc.serial_sr(CMD_GET_HWEUI)
        lc.get_freq()
      else:
        printd(clr(Color.YELLOW, "[-] Failed to get version from LoRa device"), Level.WARNING)

    except Exception, e:
      print "Error: " + str(e)

    if lc is None:
      printd(Color.YELLOW, "Error: cannot connect to device")
      exit()

    lc.set_adr(False)
    lc.set_pwr(15)
    lc.serial_sr(CMD_MAC_PAUSE)
    lc.serial_sr(CMD_SET_SF, "sf12") # sf12, sf7
    lc.serial_sr(CMD_MAC_RESUME)

    # Join and send a message
    if lc.join_abp(nwkskey, appskey, devaddr):
        printd("[+] Connected to gateway", Level.INFO)
        
        # Data-1
        printd("[+] Sending packet #1", Level.INFO)
        timeStr = time.strftime("%H%M", time.gmtime())
        if lc.send(timeStr, ack=False):
          printd(clr(Color.GREEN, "[+] Send-1 succeeded"), Level.CRITICAL)
        else:
          printd(clr(Color.RED, "[+] Send-1 failed"), Level.CRITICAL)
        time.sleep(15)

        # Data-2
        printd("[+] Sending packet #2", Level.INFO)
        timeStr = time.strftime("%H%M", time.gmtime())
        if lc.send(timeStr, ack=False):
          printd(clr(Color.GREEN, "[+] Send-2 succeeded"), Level.CRITICAL)
        else:
          printd(clr(Color.RED, "[+] Send-2 failed"), Level.CRITICAL)
        time.sleep(15)

        del lc
        exit()

    printd(clr(Color.RED, "[-] Test failed"), Level.CRITICAL)
    del lc



Тестирование


Последний шаг, остается одно — взять ноутбук (или Arduino или Raspberry Pi) с модемом с собой и пойти (поехать, полететь) в то место, где находится зона покрытия ближайшего gateway. Мне до ближайшей зоны покрытия где-то полчаса на трамвае, так что процесс не столь долгий и дорогой.

В случае успешного приема данных, пакеты будут отображены на сервере. Сам сервер с данными ничего не делает, на сервере можно настроить собственный код для обработки полученных данных. Данный пример лишь тестовый, с данными не делается ничего, в реальном приложении их можно складывать в базу, отсылать уведомления и пр, все зависит от поставленной задачи.

Зачем это надо?


В идеале, сеть LoRaWAN предоставляет пользователю готовый сервис передачи данных, работающий «из коробки», Достаточно лишь подключить устройство, и данные будут отправлены и обработаны ближайшим концентратором (разумеется если пользователь находится в зоне покрытия сети). Сферы применения достаточно обширны, от датчиков открытия двери, температуры или уровня воды на даче, до gps-маяка на животном или модели самолета.

Несколько видеороликов (на английском).

LoRa Alliance Introduction:

The Things Network Introduction:


Впрочем, если говорить о The Things Network, то это «любительский» сервис, разумеется не гарантирующий постоянного соединения. Существуют ли коммерческие провайдеры LoRaWAN, мне пока неизвестно.

Что делать если покрытия нет?


Может возникнуть вполне резонный вопрос — что делать, если до ближайшего gateway пара суток поездом или пара часов самолетом? В этом есть свой плюс — можно стать первым. Можно создать свой gateway и зарегистрировать его на сервере The Things Network, цена вопроса различна в зависимости от функциональности — от >1000EUR за «фирменные» многоканальные концентраторы, до 15$ за самодельные устройства на базе SX1276 и Raspberry Pi. Здесь есть сравнительная таблица разных маршрутизаторов, желающие могут изучить ее более подробно.

Последний вариант будет рассмотрен в следующей части.

© Geektimes