Erster Release
This commit is contained in:
1
.gitignore
vendored
1
.gitignore
vendored
@@ -114,3 +114,4 @@ dmypy.json
|
||||
# Pyre type checker
|
||||
.pyre/
|
||||
|
||||
/test/lora.conf
|
||||
|
||||
17
.idea/$CACHE_FILE$
generated
Normal file
17
.idea/$CACHE_FILE$
generated
Normal file
@@ -0,0 +1,17 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="ProjectInspectionProfilesVisibleTreeState">
|
||||
<entry key="Project Default">
|
||||
<profile-state>
|
||||
<expanded-state>
|
||||
<State />
|
||||
</expanded-state>
|
||||
<selected-state>
|
||||
<State>
|
||||
<id>Angular</id>
|
||||
</State>
|
||||
</selected-state>
|
||||
</profile-state>
|
||||
</entry>
|
||||
</component>
|
||||
</project>
|
||||
2
.idea/.gitignore
generated
vendored
Normal file
2
.idea/.gitignore
generated
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
# Default ignored files
|
||||
/workspace.xml
|
||||
6
.idea/inspectionProfiles/profiles_settings.xml
generated
Normal file
6
.idea/inspectionProfiles/profiles_settings.xml
generated
Normal file
@@ -0,0 +1,6 @@
|
||||
<component name="InspectionProjectProfileManager">
|
||||
<settings>
|
||||
<option name="USE_PROJECT_PROFILE" value="false" />
|
||||
<version value="1.0" />
|
||||
</settings>
|
||||
</component>
|
||||
7
.idea/misc.xml
generated
Normal file
7
.idea/misc.xml
generated
Normal file
@@ -0,0 +1,7 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="JavaScriptSettings">
|
||||
<option name="languageLevel" value="ES6" />
|
||||
</component>
|
||||
<component name="ProjectRootManager" version="2" project-jdk-name="Python 3.6" project-jdk-type="Python SDK" />
|
||||
</project>
|
||||
8
.idea/modules.xml
generated
Normal file
8
.idea/modules.xml
generated
Normal file
@@ -0,0 +1,8 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="ProjectModuleManager">
|
||||
<modules>
|
||||
<module fileurl="file://$PROJECT_DIR$/.idea/rn2483lora.iml" filepath="$PROJECT_DIR$/.idea/rn2483lora.iml" />
|
||||
</modules>
|
||||
</component>
|
||||
</project>
|
||||
14
.idea/rn2483lora.iml
generated
Normal file
14
.idea/rn2483lora.iml
generated
Normal file
@@ -0,0 +1,14 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<module type="PYTHON_MODULE" version="4">
|
||||
<component name="NewModuleRootManager">
|
||||
<content url="file://$MODULE_DIR$">
|
||||
<sourceFolder url="file://$MODULE_DIR$" isTestSource="false" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/test" />
|
||||
</content>
|
||||
<orderEntry type="jdk" jdkName="Python 3.6" jdkType="Python SDK" />
|
||||
<orderEntry type="sourceFolder" forTests="false" />
|
||||
</component>
|
||||
<component name="TestRunnerService">
|
||||
<option name="PROJECT_TEST_RUNNER" value="Unittests" />
|
||||
</component>
|
||||
</module>
|
||||
6
.idea/vcs.xml
generated
Normal file
6
.idea/vcs.xml
generated
Normal file
@@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="VcsDirectoryMappings">
|
||||
<mapping directory="$PROJECT_DIR$" vcs="Git" />
|
||||
</component>
|
||||
</project>
|
||||
5
__init__.py
Normal file
5
__init__.py
Normal file
@@ -0,0 +1,5 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""Module for LoRaWAN chip RN2483."""
|
||||
__author__ = "Sven Sager"
|
||||
__copyright__ = "Copyright (C) 2019 Sven Sager"
|
||||
__license__ = "GPLv3"
|
||||
5
lora.conf
Normal file
5
lora.conf
Normal file
@@ -0,0 +1,5 @@
|
||||
[DEFAULT]
|
||||
port = /dev/ttyUSB0
|
||||
appskey =
|
||||
nwkskey =
|
||||
devaddr =
|
||||
153
rn2483lora.py
Normal file
153
rn2483lora.py
Normal file
@@ -0,0 +1,153 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""LoRaWAN module for RN 2483."""
|
||||
from queue import Empty, Queue
|
||||
from threading import Thread
|
||||
|
||||
from serial import Serial
|
||||
|
||||
__author__ = "Sven Sager"
|
||||
__copyright__ = "Copyright (C) 2019 Sven Sager"
|
||||
__license__ = "GPLv3"
|
||||
|
||||
|
||||
class RN2483:
|
||||
|
||||
CRLF = b'\r\n'
|
||||
|
||||
def __init__(self, port, baud=57600):
|
||||
self._exit = False
|
||||
self._serial = Serial(port, baud, timeout=0.5)
|
||||
self._read_q = Queue()
|
||||
self._write_q = Queue()
|
||||
|
||||
# TODO: Clear buffer
|
||||
self.reader = Thread(target=self._reader, daemon=True)
|
||||
self.reader.start()
|
||||
|
||||
def _reader(self) -> None:
|
||||
"""
|
||||
Reader Thread to manage queues for the device.
|
||||
"""
|
||||
while not self._exit:
|
||||
try:
|
||||
cmd = self._write_q.get_nowait()
|
||||
except Empty:
|
||||
pass
|
||||
else:
|
||||
self._serial.write(cmd + self.CRLF)
|
||||
|
||||
buff = self._serial.read_until()
|
||||
buff = buff.decode("ASCII").strip()
|
||||
if buff:
|
||||
self._read_q.put_nowait(buff)
|
||||
|
||||
def close(self) -> None:
|
||||
"""
|
||||
Disconnect from Device.
|
||||
"""
|
||||
self._exit = True
|
||||
self.reader.join()
|
||||
self._serial.close()
|
||||
|
||||
def get_result(self, timeout=None) -> str:
|
||||
"""
|
||||
Get a result fromMo device queue.
|
||||
|
||||
Some commands returns more than one line. You can call this function
|
||||
to get the extra lines from the queue.
|
||||
|
||||
:param timeout: Timeout to wait for result of device
|
||||
:return: Result from device or empty string
|
||||
"""
|
||||
try:
|
||||
result = self._read_q.get(timeout=timeout)
|
||||
except Empty:
|
||||
result = ""
|
||||
return result
|
||||
|
||||
def send(self, cmd: str, get_result=True, timeout=None) -> str:
|
||||
"""
|
||||
Send a command to Device.
|
||||
|
||||
This function can return the result from the device of the send
|
||||
command. If you set get_result to False, you have to use
|
||||
get_result(...) to get the answers of the device.
|
||||
|
||||
:param cmd: Command to send
|
||||
:param get_result: If true, function will return result
|
||||
:param timeout: Timeout to wait for result of device
|
||||
:return: Result from device or empty string
|
||||
"""
|
||||
cmd = cmd.encode("ASCII")
|
||||
self._write_q.put(cmd)
|
||||
if get_result:
|
||||
return self.get_result(timeout)
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
from configparser import ConfigParser
|
||||
from time import sleep
|
||||
|
||||
conf = ConfigParser()
|
||||
conf.read("lora.conf")
|
||||
|
||||
root = RN2483(conf.get("DEFAULT", "port"))
|
||||
|
||||
def first_init() -> None:
|
||||
"""Init our RN2483 after firmware startup."""
|
||||
|
||||
# Configuration: ["cmd", results]
|
||||
cmd_list = [
|
||||
["sys get ver", 1],
|
||||
["mac reset 868", 1],
|
||||
["sys get hweui", 1],
|
||||
|
||||
# Set TTN configurations form .conf file
|
||||
["mac set appskey {0}".format(conf.get("DEFAULT", "appskey")), 1],
|
||||
["mac set nwkskey {0}".format(conf.get("DEFAULT", "nwkskey")), 1],
|
||||
["mac set devaddr {0}".format(conf.get("DEFAULT", "devaddr")), 1],
|
||||
|
||||
["mac join abp", 2],
|
||||
["mac set ar on", 1],
|
||||
]
|
||||
for do_cmd in cmd_list:
|
||||
print("<-", do_cmd[0])
|
||||
print("->", root.send(do_cmd[0]))
|
||||
|
||||
# Get awaited extra results results
|
||||
for i in range(do_cmd[1] - 1):
|
||||
print(" ", root.get_result())
|
||||
|
||||
first_init()
|
||||
|
||||
# Command mode
|
||||
while True:
|
||||
results = 1
|
||||
|
||||
cmd_input = input("<- ")
|
||||
if cmd_input == "exit":
|
||||
break
|
||||
elif cmd_input == "reset":
|
||||
root.send("sys reset")
|
||||
sleep(3)
|
||||
first_init()
|
||||
continue
|
||||
|
||||
# Functions to make live more easier
|
||||
elif cmd_input.find("send ") > -1:
|
||||
lst = cmd_input.split()
|
||||
cmd_input = "mac tx uncnf {port} {bytes}".format(
|
||||
port=lst[2] if len(lst) > 2 else 1,
|
||||
bytes=lst[1] if len(lst) > 1 else "",
|
||||
)
|
||||
results = 2
|
||||
|
||||
# Always send command
|
||||
if cmd_input:
|
||||
print("->", root.send(cmd_input))
|
||||
|
||||
# Get awaited extra results results
|
||||
for i in range(results - 1):
|
||||
print(" ", root.get_result())
|
||||
|
||||
root.close()
|
||||
Reference in New Issue
Block a user