mirror of
https://github.com/daniviga/django-ram.git
synced 2025-08-04 13:17:50 +02:00
Merge pull request #1 from daniviga/feedback
Make operations sync and use bidirectional socket instead of MQTT
This commit is contained in:
Submodule arduino/CommandStation-EX updated: 92591c8a2e...00e3c80b44
Submodule arduino/arduino-cli updated: e63c798cda...4506bf9d36
0
daemons/__init__.py
Normal file
0
daemons/__init__.py
Normal file
44
daemons/net-to-serial.py
Normal file
44
daemons/net-to-serial.py
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import serial
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
|
||||||
|
class SerialDaemon:
|
||||||
|
def __init__(self):
|
||||||
|
self.ser = serial.Serial('/dev/pts/7') # WIP
|
||||||
|
self.ser.baudrate = 115200
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
|
try:
|
||||||
|
self.ser.close()
|
||||||
|
except AttributeError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
async def handle_echo(self, reader, writer):
|
||||||
|
data = await reader.read(100)
|
||||||
|
message = data.decode()
|
||||||
|
addr = writer.get_extra_info('peername')
|
||||||
|
|
||||||
|
print(f"Received {message!r} from {addr!r}")
|
||||||
|
|
||||||
|
self.ser.write(data)
|
||||||
|
response = self.read_until()
|
||||||
|
print(f"Send: {response!r}")
|
||||||
|
writer.write(response)
|
||||||
|
await writer.drain()
|
||||||
|
|
||||||
|
print("Close the connection")
|
||||||
|
writer.close()
|
||||||
|
|
||||||
|
|
||||||
|
async def main():
|
||||||
|
sd = SerialDaemon()
|
||||||
|
server = await asyncio.start_server(
|
||||||
|
sd.handle_echo, '127.0.0.1', 2560) # WIP
|
||||||
|
addrs = ', '.join(str(sock.getsockname()) for sock in server.sockets)
|
||||||
|
print(f'Serving on {addrs}')
|
||||||
|
|
||||||
|
async with server:
|
||||||
|
await server.serve_forever()
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
asyncio.run(main())
|
@@ -1,42 +0,0 @@
|
|||||||
import serial
|
|
||||||
import logging
|
|
||||||
import socket
|
|
||||||
import asyncio
|
|
||||||
import time
|
|
||||||
import paho.mqtt.client as mqtt
|
|
||||||
from asyncio_mqtt import Client
|
|
||||||
|
|
||||||
MQTT_HOST = "127.0.0.1"
|
|
||||||
MQTT_PORT = 1883
|
|
||||||
|
|
||||||
|
|
||||||
async def mqtt_broker(ser):
|
|
||||||
async with Client(MQTT_HOST, port=MQTT_PORT) as client:
|
|
||||||
await client.subscribe("dcc/commands")
|
|
||||||
async with client.unfiltered_messages() as messages:
|
|
||||||
async for message in messages:
|
|
||||||
print(message.payload.decode())
|
|
||||||
# ser.write(message.payload)
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
client = mqtt.Client()
|
|
||||||
# ser = serial.Serial('/dev/pts/7')
|
|
||||||
# ser.baudrate = 9600
|
|
||||||
ser = None # remove me
|
|
||||||
while True:
|
|
||||||
try:
|
|
||||||
client.connect(MQTT_HOST, MQTT_PORT)
|
|
||||||
break
|
|
||||||
except (socket.gaierror, ConnectionRefusedError):
|
|
||||||
logging.warning('Broker not available')
|
|
||||||
time.sleep(5)
|
|
||||||
|
|
||||||
logging.info('Broker subscribed')
|
|
||||||
client.disconnect()
|
|
||||||
asyncio.run(mqtt_broker(ser))
|
|
||||||
# ser.close()
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
@@ -39,6 +39,7 @@ INSTALLED_APPS = [
|
|||||||
'django.contrib.messages',
|
'django.contrib.messages',
|
||||||
'django.contrib.staticfiles',
|
'django.contrib.staticfiles',
|
||||||
'django_countries',
|
'django_countries',
|
||||||
|
'solo',
|
||||||
'rest_framework',
|
'rest_framework',
|
||||||
'dcc',
|
'dcc',
|
||||||
'driver',
|
'driver',
|
||||||
|
@@ -1,3 +1,6 @@
|
|||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
|
from solo.admin import SingletonModelAdmin
|
||||||
|
|
||||||
# Register your models here.
|
from driver.models import DriverConfiguration
|
||||||
|
|
||||||
|
admin.site.register(DriverConfiguration, SingletonModelAdmin)
|
||||||
|
@@ -1,22 +1,27 @@
|
|||||||
import paho.mqtt.client as mqtt
|
import socket
|
||||||
|
|
||||||
|
from driver.models import DriverConfiguration
|
||||||
|
|
||||||
|
|
||||||
class Connector:
|
class Connector:
|
||||||
MQTT_HOST = "127.0.0.1"
|
|
||||||
MQTT_PORT = 1883
|
|
||||||
|
|
||||||
@classmethod
|
def __init__(self):
|
||||||
def __mqtt_pub(self, message):
|
config = DriverConfiguration.get_solo()
|
||||||
|
self.remote_host = config.remote_host
|
||||||
|
self.remote_port = config.remote_port
|
||||||
|
|
||||||
|
def __send_data(self, message):
|
||||||
# to be encoded
|
# to be encoded
|
||||||
client = mqtt.Client()
|
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as sock:
|
||||||
client.connect(self.MQTT_HOST, self.MQTT_PORT)
|
sock.connect((self.remote_host, self.remote_port))
|
||||||
client.publish("dcc/commands", payload=message)
|
sock.sendall(message)
|
||||||
client.disconnect()
|
resp = sock.recv(1024)
|
||||||
print(message)
|
|
||||||
|
print(resp)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def passthrough(self, data):
|
def passthrough(self, data):
|
||||||
self.__mqtt_pub(data)
|
self.__send_data(data)
|
||||||
|
|
||||||
def ops(self, address, data, function=False):
|
def ops(self, address, data, function=False):
|
||||||
if function:
|
if function:
|
||||||
@@ -25,7 +30,7 @@ class Connector:
|
|||||||
else:
|
else:
|
||||||
message = "<t 1 {0} {1} {2}>".format(address, data['speed'],
|
message = "<t 1 {0} {1} {2}>".format(address, data['speed'],
|
||||||
data['direction'])
|
data['direction'])
|
||||||
self.__mqtt_pub(message)
|
self.__send_data(message)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def infra(self, data):
|
def infra(self, data):
|
||||||
@@ -36,9 +41,9 @@ class Connector:
|
|||||||
track = ""
|
track = ""
|
||||||
|
|
||||||
if power:
|
if power:
|
||||||
self.__mqtt_pub('<1{}>'.format(track))
|
self.__send_data('<1{}>'.format(track))
|
||||||
else:
|
else:
|
||||||
self.__mqtt_pub('<0{}>'.format(track))
|
self.__send_data('<0{}>'.format(track))
|
||||||
|
|
||||||
def emergency(self):
|
def emergency(self):
|
||||||
self.__mqtt_pub('<!>')
|
self.__send_data('<!>')
|
||||||
|
25
dcc/driver/migrations/0001_initial.py
Normal file
25
dcc/driver/migrations/0001_initial.py
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
# Generated by Django 4.0 on 2021-12-31 10:41
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
initial = True
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='DriverConfiguration',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('remote_host', models.GenericIPAddressField(default='192.168.4.1', protocol='IPv4')),
|
||||||
|
('remote_port', models.SmallIntegerField(default=2560)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'Driver Configuration',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
@@ -1,3 +1,14 @@
|
|||||||
from django.db import models
|
from django.db import models
|
||||||
|
from solo.models import SingletonModel
|
||||||
|
|
||||||
# Create your models here.
|
|
||||||
|
class DriverConfiguration(SingletonModel):
|
||||||
|
remote_host = models.GenericIPAddressField(
|
||||||
|
protocol="IPv4", default="192.168.4.1")
|
||||||
|
remote_port = models.SmallIntegerField(default=2560)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return "Driver Configuration"
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = "Driver Configuration"
|
||||||
|
@@ -8,11 +8,8 @@ class CabSerializer(serializers.ModelSerializer):
|
|||||||
manufacturer = ManufacturerSerializer()
|
manufacturer = ManufacturerSerializer()
|
||||||
decoder = DecoderSerializer()
|
decoder = DecoderSerializer()
|
||||||
company = CompanySerializer()
|
company = CompanySerializer()
|
||||||
# manufacturer = serializers.StringRelatedField()
|
|
||||||
# decoder = serializers.StringRelatedField()
|
|
||||||
# company = serializers.StringRelatedField()
|
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Cab
|
model = Cab
|
||||||
fields = "__all__"
|
fields = "__all__"
|
||||||
read_only_fields = ("identifier", "creation_time", "updated_time")
|
read_only_fields = ("creation_time", "updated_time")
|
||||||
|
@@ -1,27 +1,27 @@
|
|||||||
from rest_framework.generics import ListCreateAPIView, RetrieveUpdateAPIView
|
from rest_framework.generics import ListAPIView, RetrieveAPIView
|
||||||
|
|
||||||
from roster.models import Cab
|
from roster.models import Cab
|
||||||
from roster.serializers import CabSerializer
|
from roster.serializers import CabSerializer
|
||||||
|
|
||||||
|
|
||||||
class RosterList(ListCreateAPIView):
|
class RosterList(ListAPIView):
|
||||||
queryset = Cab.objects.all()
|
queryset = Cab.objects.all()
|
||||||
serializer_class = CabSerializer
|
serializer_class = CabSerializer
|
||||||
|
|
||||||
|
|
||||||
class RosterGet(RetrieveUpdateAPIView):
|
class RosterGet(RetrieveAPIView):
|
||||||
queryset = Cab.objects.all()
|
queryset = Cab.objects.all()
|
||||||
serializer_class = CabSerializer
|
serializer_class = CabSerializer
|
||||||
lookup_field = 'uuid'
|
lookup_field = 'uuid'
|
||||||
|
|
||||||
|
|
||||||
class RosterAddress(RetrieveUpdateAPIView):
|
class RosterAddress(RetrieveAPIView):
|
||||||
queryset = Cab.objects.all()
|
queryset = Cab.objects.all()
|
||||||
serializer_class = CabSerializer
|
serializer_class = CabSerializer
|
||||||
lookup_field = 'address'
|
lookup_field = 'address'
|
||||||
|
|
||||||
|
|
||||||
class RosterIdentifier(RetrieveUpdateAPIView):
|
class RosterIdentifier(RetrieveAPIView):
|
||||||
queryset = Cab.objects.all()
|
queryset = Cab.objects.all()
|
||||||
serializer_class = CabSerializer
|
serializer_class = CabSerializer
|
||||||
lookup_field = 'identifier'
|
lookup_field = 'identifier'
|
||||||
|
@@ -1,8 +1,9 @@
|
|||||||
pytz
|
pytz
|
||||||
Django
|
Django
|
||||||
djangorestframework
|
djangorestframework
|
||||||
|
django-solo
|
||||||
django-health-check
|
django-health-check
|
||||||
# psycopg2-binary
|
# psycopg2-binary
|
||||||
paho-mqtt
|
# paho-mqtt
|
||||||
asyncio-mqtt
|
# asyncio-mqtt
|
||||||
pySerial
|
pySerial
|
||||||
|
Reference in New Issue
Block a user