ULP-ActivitySupport – commitdiff

You can use Git to clone the repository via the web URL. Download snapshot (zip)
Add micro info radiator code
authorJulian Fietkau <git@fietkau.software>
Mon, 17 Oct 2022 23:55:44 +0000 (01:55 +0200)
committerJulian Fietkau <git@fietkau.software>
Mon, 17 Oct 2022 23:55:44 +0000 (01:55 +0200)
28 files changed:
microinforadiator/Audio/audio.py [new file with mode: 0644]
microinforadiator/Bluetooth/__init__.py [new file with mode: 0644]
microinforadiator/Bluetooth/bluetooth_controller.py [new file with mode: 0644]
microinforadiator/LED/__init__.py [new file with mode: 0644]
microinforadiator/LED/arrows.py [new file with mode: 0644]
microinforadiator/LED/led_activities.py [new file with mode: 0644]
microinforadiator/LED/symbols.py [new file with mode: 0644]
microinforadiator/MiR01.env [new file with mode: 0644]
microinforadiator/MiR02.env [new file with mode: 0644]
microinforadiator/MiR03.env [new file with mode: 0644]
microinforadiator/MiR04.env [new file with mode: 0644]
microinforadiator/MiR05.env [new file with mode: 0644]
microinforadiator/MiR06.env [new file with mode: 0644]
microinforadiator/MiR07.env [new file with mode: 0644]
microinforadiator/MiR08.env [new file with mode: 0644]
microinforadiator/MiR09.env [new file with mode: 0644]
microinforadiator/MiR10.env [new file with mode: 0644]
microinforadiator/__init__.py [new file with mode: 0644]
microinforadiator/env.example [new file with mode: 0644]
microinforadiator/interaction.py [new file with mode: 0644]
microinforadiator/joystick_control.py [new file with mode: 0644]
microinforadiator/main.py [new file with mode: 0644]
microinforadiator/mqtt_service.py [new file with mode: 0644]
microinforadiator/proximity.py [new file with mode: 0644]
microinforadiator/requirements.txt [new file with mode: 0644]
microinforadiator/run.sh [new file with mode: 0755]
microinforadiator/run_multi.py [new file with mode: 0755]
microinforadiator/state_factory.py [new file with mode: 0644]

diff --git a/microinforadiator/Audio/audio.py b/microinforadiator/Audio/audio.py
new file mode 100644 (file)
index 0000000..06a2358
--- /dev/null
@@ -0,0 +1,28 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+import pygame
+from time import sleep
+
+def sign():
+    pygame.mixer.init()
+    pygame.mixer.music.load("glocke.mp3")
+    pygame.mixer.music.play()
+   # while pygame.mixer.music.get_busy() == True:
+    #    continue
+    
+def birds():
+    pygame.mixer.init()
+    pygame.mixer.music.load("birds.mp3")
+    pygame.mixer.music.play()
+    while pygame.mixer.music.get_busy() == True:
+        continue
+    
+
+if __name__ == '__main__':
+    sign()
+    sleep(2)
+    birds()
diff --git a/microinforadiator/Bluetooth/__init__.py b/microinforadiator/Bluetooth/__init__.py
new file mode 100644 (file)
index 0000000..e69de29
diff --git a/microinforadiator/Bluetooth/bluetooth_controller.py b/microinforadiator/Bluetooth/bluetooth_controller.py
new file mode 100644 (file)
index 0000000..91fee46
--- /dev/null
@@ -0,0 +1,255 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+import sys
+import struct
+import bluetooth
+import bluetooth._bluetooth as bluez
+import csv
+from datetime import datetime
+import paho.mqtt.client as mqtt
+import fcntl
+import array
+
+import time
+import os
+import datetime
+
+#nearby_devices = bluetooth.discover_devices(lookup_names=True)
+#print("found %d devices" %len(nearby_devices))
+
+#for addr, name in nearby_devices:
+#    print("[%s]  %s" %(addr, name))
+    
+def printpacket(pkt):
+    for c in pkt:
+        sys.stdout.write("{%02x} ".format(struct.unpack("B", c)[0]))
+
+def read_inquiry_mode(sock):
+    """returns the current mode, or -1 on failure"""
+    # save current filter
+    old_filter = sock.getsockopt(bluez.SOL_HCI, bluez.HCI_FILTER, 14)
+
+    # Setup socket filter to receive only events related to the
+    # read_inquiry_mode command
+    flt = bluez.hci_filter_new()
+    opcode = bluez.cmd_opcode_pack(bluez.OGF_HOST_CTL,
+                                   bluez.OCF_READ_INQUIRY_MODE)
+    bluez.hci_filter_set_ptype(flt, bluez.HCI_EVENT_PKT)
+    bluez.hci_filter_set_event(flt, bluez.EVT_CMD_COMPLETE)
+    bluez.hci_filter_set_opcode(flt, opcode)
+    sock.setsockopt(bluez.SOL_HCI, bluez.HCI_FILTER, flt)
+
+    # first read the current inquiry mode.
+    bluez.hci_send_cmd(sock, bluez.OGF_HOST_CTL,
+                       bluez.OCF_READ_INQUIRY_MODE)
+
+    pkt = sock.recv(255)
+    status, mode = struct.unpack("xxxxxxBB", pkt)
+
+    # restore old filter
+    sock.setsockopt(bluez.SOL_HCI, bluez.HCI_FILTER, old_filter)
+
+    return mode
+
+def write_inquiry_mode(sock, mode):
+    """returns 0 on success, -1 on failure"""
+    # save current filter
+    old_filter = sock.getsockopt(bluez.SOL_HCI, bluez.HCI_FILTER, 14)
+
+    # Setup socket filter to receive only events related to the
+    # write_inquiry_mode command
+    flt = bluez.hci_filter_new()
+    opcode = bluez.cmd_opcode_pack(bluez.OGF_HOST_CTL,
+                                   bluez.OCF_WRITE_INQUIRY_MODE)
+    bluez.hci_filter_set_ptype(flt, bluez.HCI_EVENT_PKT)
+    bluez.hci_filter_set_event(flt, bluez.EVT_CMD_COMPLETE)
+    bluez.hci_filter_set_opcode(flt, opcode)
+    sock.setsockopt(bluez.SOL_HCI, bluez.HCI_FILTER, flt)
+
+    # send the command!
+    bluez.hci_send_cmd(sock, bluez.OGF_HOST_CTL,
+                       bluez.OCF_WRITE_INQUIRY_MODE,
+                       struct.pack("B", mode))
+
+    pkt = sock.recv(255)
+    status = struct.unpack("xxxxxxB", pkt)[0]
+
+    # restore old filter
+    sock.setsockopt(bluez.SOL_HCI, bluez.HCI_FILTER, old_filter)
+    if not status:
+        return -1
+
+    return 0
+
+def device_inquiry_with_with_rssi(sock):
+    # save current filter
+    old_filter = sock.getsockopt(bluez.SOL_HCI, bluez.HCI_FILTER, 14)
+
+    # perform a device inquiry on bluetooth device #0
+    # The inquiry should last 8 * 1.28 = 10.24 seconds
+    # before the inquiry is performed, bluez should flush its cache of
+    # previously discovered devices
+    flt = bluez.hci_filter_new()
+    bluez.hci_filter_all_events(flt)
+    bluez.hci_filter_set_ptype(flt, bluez.HCI_EVENT_PKT)
+    sock.setsockopt(bluez.SOL_HCI, bluez.HCI_FILTER, flt)
+
+    duration = 4
+    max_responses = 255
+    cmd_pkt = struct.pack("BBBBB", 0x33, 0x8b, 0x9e, duration, max_responses)
+    bluez.hci_send_cmd(sock, bluez.OGF_LINK_CTL, bluez.OCF_INQUIRY, cmd_pkt)
+
+    results = []
+    
+    while True:
+        pkt = sock.recv(255)
+        ptype, event, plen = struct.unpack("BBB", pkt[:3])
+        #print("Event: {}".format(event))
+        if event == bluez.EVT_INQUIRY_RESULT_WITH_RSSI:
+            pkt = pkt[3:]
+            nrsp = bluetooth.get_byte(pkt[0])
+            for i in range(nrsp):
+                addr = bluez.ba2str(pkt[1+6*i:1+6*i+6])
+                rssi = bluetooth.byte_to_signed_int(
+                    bluetooth.get_byte(pkt[1 + 13 * nrsp + i]))
+                timestamp = datetime.now()
+                #results.append(addr)
+                #results.append(rssi)
+                #results.append(timestamp)
+                results.append((addr,rssi,timestamp))
+                #print("[{}] RSSI: {}".format(addr, rssi))
+                              
+        elif event == bluez.EVT_INQUIRY_COMPLETE:
+            break
+        elif event == bluez.EVT_CMD_STATUS:
+            status, ncmd, opcode = struct.unpack("BBH", pkt[3:7])
+            if status:
+                print("Uh oh...")
+                printpacket(pkt[3:7])
+                break
+        elif event == bluez.EVT_INQUIRY_RESULT:
+            pkt = pkt[3:]
+            nrsp = bluetooth.get_byte(pkt[0])
+            for i in range(nrsp):
+                addr = bluez.ba2str(pkt[1+6*i:1+6*i+6])
+                #results.append((addr, -1))
+                print("[{}] (no RRSI)".format(addr))
+        else:
+            print("Unrecognized packet type 0x{:02x}.".format(ptype))
+
+    # restore old filter
+    sock.setsockopt(bluez.SOL_HCI, bluez.HCI_FILTER, old_filter)
+        
+    return results
+
+def get_devices():
+    dev_id = 0
+    writer = None
+    try:
+        sock = bluez.hci_open_dev(dev_id)
+    except:
+        print("Error accessing bluetooth device.")
+        sys.exit(1)
+
+    try:
+        mode = read_inquiry_mode(sock)
+    except Exception as e:
+        print("Error reading inquiry mode.")
+        print("Are you sure this a bluetooth 1.2 device?")
+        print(e)
+        sys.exit(1)
+    print("Current inquiry mode is", mode)
+
+    if mode != 1:
+        print("Writing inquiry mode...")
+        try:
+            result = write_inquiry_mode(sock, 1)
+        except Exception as e:
+            print("Error writing inquiry mode. Are you sure you're root?")
+            print(e)
+            sys.exit(1)
+        if result:
+            print("Error while setting inquiry mode")
+        print("Result:", result)
+
+
+    #get detected devices (MAC address), RSSI distance and timestamp 
+    detected_devices = device_inquiry_with_with_rssi(sock)   
+    #if len(detected_devices) >= 1:
+        #for i in detected_devices:
+            #print("NOW")
+            #print(detected_devices[0])
+            #print(i)
+            
+    return detected_devices
+
+
+def write_BT_toCSV(results):
+    
+    with open('bt-test.csv',mode='w') as file:
+        file_writer = csv.writer(file,delimiter=',',quotechar='"',quoting=csv.QUOTE_MINIMAL)
+        file_writer.writerow(['ADDR','RSSI','Timestamp'])
+              
+        for x,y,z in results:
+            write_to_log = file_writer.writerow((x,y,z))
+        return(write_to_log)
+    
+
+def start_BTdetection():
+    
+    results = get_devices()
+    return results
+
+#get only RSSI value for a specific BD_ADDR
+def bluetooth_rssi(addr):
+    # Open hci socket
+    hci_sock = bluez.hci_open_dev()
+    hci_fd = hci_sock.fileno()
+
+    # Connect to device (to whatever you like)
+    bt_sock = bluetooth.BluetoothSocket(bluetooth.L2CAP)
+    bt_sock.settimeout(10)
+    result = bt_sock.connect_ex((addr, 1))  # PSM 1 - Service Discovery
+#
+#    try:
+    # Get ConnInfo
+    frmt = "6sB17s"
+    t = bluez.str2ba(addr)
+    s = bytes(('\0'),encoding ='utf8')
+    reqstr = struct.pack(frmt, t, bluez.ACL_LINK, s * 17)
+    request = array.array("b", reqstr )
+    try:
+        handle = fcntl.ioctl(hci_fd, bluez.HCIGETCONNINFO, request, 1)
+        handle = struct.unpack("8xH14x", request.tostring())[0]
+        frmt = "H"
+        # Get RSSI
+        cmd_pkt = struct.pack(frmt, handle)
+        rssi = bluez.hci_send_req(hci_sock, bluez.OGF_STATUS_PARAM,
+                bluez.OCF_READ_RSSI, bluez.EVT_CMD_COMPLETE, 4, cmd_pkt)
+        rssi = rssi[3]
+    except:
+        print("Could not detect Bluetooth device")
+        rssi = 99999
+
+    bt_sock.close()
+    hci_sock.close()
+
+    return rssi
+#
+#    except:
+#        return None
+        
+        
+if __name__=='__main__':
+    
+    res = get_devices()
+    #bdaddr = "54:B1:21:F0:80:A3"
+    #for x,y,z in res:
+    #    if x == bdaddr:
+    #res = bluetooth.lookup_name( bdaddr )
+        #print((x,y,z))
+    #write_BT_toCSV(res)
diff --git a/microinforadiator/LED/__init__.py b/microinforadiator/LED/__init__.py
new file mode 100644 (file)
index 0000000..e69de29
diff --git a/microinforadiator/LED/arrows.py b/microinforadiator/LED/arrows.py
new file mode 100644 (file)
index 0000000..f1ef7c6
--- /dev/null
@@ -0,0 +1,76 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+r = (255,0,0)
+b = (0,0,255)
+g = (0,255,0)
+y = (255,255,0)
+m = (255,0,255)
+c = (0,255,255)
+bk = (0,0,0)
+w = (255,255,255) #150,150,150
+cb = bk
+ca = b
+
+def arrow_up (cb, ca):
+    arrow_up = [
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,ca,ca,ca,cb,cb,cb,
+        cb,ca,cb,ca,cb,ca,cb,cb,
+        ca,cb,cb,ca,cb,cb,ca,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb
+        ]
+        
+    return arrow_up
+    
+def arrow_left (cb, ca):
+
+    arrow_l = [
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,ca,cb,cb,cb,cb,cb,
+        cb,ca,cb,cb,cb,cb,cb,cb,
+        ca,ca,ca,ca,ca,ca,ca,ca,
+        cb,ca,cb,cb,cb,cb,cb,cb,
+        cb,cb,ca,cb,cb,cb,cb,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,cb,cb,cb,cb,cb,cb
+    ]
+            
+    return arrow_l
+    
+def arrow_down (cb, ca):
+        
+    arrow_d = [
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        ca,cb,cb,ca,cb,cb,ca,cb,
+        cb,ca,cb,ca,cb,ca,cb,cb,
+        cb,cb,ca,ca,ca,cb,cb,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb
+    ]
+            
+    return arrow_d
+    
+    #def arrow_right (color_background, color_arrow):
+def arrow_right (cb, ca):
+
+    arrow_r = [
+        cb,cb,cb,cb,ca,cb,cb,cb,
+        cb,cb,cb,cb,cb,ca,cb,cb,
+        cb,cb,cb,cb,cb,cb,ca,cb,
+        ca,ca,ca,ca,ca,ca,ca,ca,
+        cb,cb,cb,cb,cb,cb,ca,cb,
+        cb,cb,cb,cb,cb,ca,cb,cb,
+        cb,cb,cb,cb,ca,cb,cb,cb,
+        cb,cb,cb,cb,cb,cb,cb,cb
+    ]
+            
+    return arrow_r
diff --git a/microinforadiator/LED/led_activities.py b/microinforadiator/LED/led_activities.py
new file mode 100644 (file)
index 0000000..b9d2394
--- /dev/null
@@ -0,0 +1,72 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+from arrows import arrow_up, arrow_down, arrow_left, arrow_right
+from symbols import hook, bee
+from time import sleep
+import os
+
+if os.getenv('ENABLE_SENSEHAT') == 'true':
+    from sense_hat import SenseHat
+    sense = SenseHat()
+
+def blinkingArrow(direction,cb,ca):
+    a = []
+    if(direction == "right"):
+        a = arrow_right(cb,ca)
+    elif(direction =="left"):
+        a = arrow_left(cb,ca)
+    elif(direction =="up"):
+        a = arrow_up(cb,ca)
+    elif(direction == "down"):
+        a = arrow_down(cb,ca)
+    
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        sense.set_pixels(a)
+    sleep(1)
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        sense.clear()
+    sleep(1)    
+                   
+def lighting(color):
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        sense.clear(color)
+    sleep(15)
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        sense.clear()
+            
+def blinking(color):
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        sense.clear(color)
+    sleep(1)
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        sense.clear()
+    sleep(1)    
+
+def show_hook(ca):
+    
+    pixels = hook(ca)
+    
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        
+        sense.set_pixels(pixels)
+    sleep(1)
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        sense.clear()
+    sleep(1)    
+                   
+
+def show_bee():
+    
+    pixels = bee()
+    
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        sense.set_pixels(pixels)
+    sleep(1)
+    if os.getenv('ENABLE_SENSEHAT') == 'true':
+        sense.clear()
+    sleep(1)
+    
diff --git a/microinforadiator/LED/symbols.py b/microinforadiator/LED/symbols.py
new file mode 100644 (file)
index 0000000..8b90acc
--- /dev/null
@@ -0,0 +1,145 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+r = (255,0,0)
+b = (0,0,255)
+g = (0,255,0)
+y = (255,255,0)
+m = (255,0,255)
+c = (0,255,255)
+bk = (0,0,0)
+w = (255,255,255) #150,150,150
+
+
+def rainy_cloud():
+    cb = bk
+    ca = b
+    c = [
+        
+            cb,cb,cb,cb,cb,cb,cb,cb,
+            cb,cb,ca,ca,cb,cb,cb,ca,
+            cb,ca,cb,cb,ca,cb,cb,cb,
+            ca,cb,cb,cb,ca,cb,ca,cb,
+            ca,cb,cb,cb,ca,cb,cb,cb,
+            cb,ca,cb,cb,ca,cb,cb,ca,
+            cb,ca,cb,cb,ca,cb,cb,cb,
+            cb,cb,ca,ca,cb,cb,ca,cb
+
+        ]
+    
+    cr = [
+        
+            cb,cb,cb,cb,cb,cb,cb,cb,
+            cb,cb,ca,ca,cb,cb,ca,cb,
+            cb,ca,ca,ca,ca,cb,cb,cb,
+            ca,ca,ca,ca,ca,cb,cb,ca,
+            ca,ca,ca,ca,ca,cb,cb,cb,
+            cb,ca,ca,ca,ca,cb,ca,cb,
+            cb,ca,ca,ca,ca,cb,cb,cb,
+            cb,cb,ca,ca,cb,cb,cb,ca
+
+        ]
+    
+    return [c, cr]
+    
+def sun():
+    sun = [
+            
+        y,b,b,b,b,b,b,y,
+        b,y,b,b,b,b,y,b,
+        b,b,b,y,y,b,b,b,
+        b,b,y,y,y,y,b,b,
+        b,b,y,y,y,y,b,b,
+        b,b,b,y,y,b,b,b,
+        b,y,b,b,b,b,y,b,
+        y,b,b,b,b,b,b,y          
+            
+        ]
+    sun2 = [
+            
+        y,b,b,b,y,b,b,y,
+        b,y,b,b,y,b,y,b,
+        b,b,y,y,y,y,b,b,
+        y,y,y,y,y,y,b,b,
+        b,b,y,y,y,y,y,y,
+        b,b,y,y,y,y,b,b,
+        b,y,b,y,b,b,y,b,
+        y,b,b,y,b,b,b,y          
+            
+        ]
+    
+    return [sun, sun2]
+        
+def escooter():
+    e = [
+            
+        b,b,b,b,b,b,y,b,
+        b,b,y,y,b,y,y,y,
+        b,y,b,b,b,b,y,b,
+        b,b,b,y,b,b,y,b,
+        b,b,b,y,b,b,y,b,
+        b,y,y,y,b,y,y,y,
+        b,b,b,b,b,b,y,b,
+        b,b,b,b,b,b,b,b   
+            
+        ]
+    
+    return e
+
+def warning():
+    b = bk
+    
+    w = [
+        
+        b,b,r,r,r,r,b,b,
+        b,r,b,b,b,b,r,b,
+        r,b,b,b,b,b,b,r,
+        r,b,r,r,b,r,b,r,
+        r,b,r,r,b,r,b,r,
+        r,b,b,b,b,b,b,r,
+        b,r,b,b,b,b,r,b,
+        b,b,r,r,r,r,b,b
+        
+        ]
+    return w
+
+def hook(ca):
+    cb = bk
+      
+    hook = [
+        cb,cb,cb,cb,cb,cb,cb,cb,
+        cb,cb,cb,cb,cb,cb,cb,ca,
+        cb,cb,cb,cb,cb,cb,ca,ca,
+        cb,cb,cb,cb,cb,ca,ca,cb,
+        ca,cb,cb,cb,ca,ca,cb,cb,
+        ca,ca,cb,ca,ca,cb,cb,cb,
+        cb,ca,ca,ca,cb,cb,cb,cb,
+        cb,cb,ca,cb,cb,cb,cb,cb
+    ]
+
+    return hook
+
+def bee():
+    cb = bk
+    cb = (5,5,5)
+    ca = y
+
+    bee = [
+        ca,cb,ca,cb,ca,cb,ca,cb,
+        ca,cb,ca,cb,ca,cb,ca,cb,
+        ca,cb,ca,cb,ca,cb,ca,cb,
+        ca,cb,ca,cb,ca,cb,ca,cb,
+        ca,cb,ca,cb,ca,cb,ca,cb,
+        ca,cb,ca,cb,ca,cb,ca,cb,
+        ca,cb,ca,cb,ca,cb,ca,cb,
+        ca,cb,ca,cb,ca,cb,ca,cb
+    ]
+    return bee
+
+
+if __name__ == '__main__':
+    print(bee())
+
diff --git a/microinforadiator/MiR01.env b/microinforadiator/MiR01.env
new file mode 100644 (file)
index 0000000..15d5163
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 1
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:15.7,25.8
+ROTATION=270
diff --git a/microinforadiator/MiR02.env b/microinforadiator/MiR02.env
new file mode 100644 (file)
index 0000000..5d349b7
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 2
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:15.7,19.0
+ROTATION=270
diff --git a/microinforadiator/MiR03.env b/microinforadiator/MiR03.env
new file mode 100644 (file)
index 0000000..fb10d0f
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 3
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:14.4,14.6
+ROTATION=180
diff --git a/microinforadiator/MiR04.env b/microinforadiator/MiR04.env
new file mode 100644 (file)
index 0000000..30a4861
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 4
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:26.1,6.0
+ROTATION=190
diff --git a/microinforadiator/MiR05.env b/microinforadiator/MiR05.env
new file mode 100644 (file)
index 0000000..b8460ee
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 5
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:38.0,6.9
+ROTATION=200
diff --git a/microinforadiator/MiR06.env b/microinforadiator/MiR06.env
new file mode 100644 (file)
index 0000000..23b782e
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 6
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:28.1,16.1
+ROTATION=195
diff --git a/microinforadiator/MiR07.env b/microinforadiator/MiR07.env
new file mode 100644 (file)
index 0000000..cd3ce09
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 7
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:36.6,18.9
+ROTATION=0
diff --git a/microinforadiator/MiR08.env b/microinforadiator/MiR08.env
new file mode 100644 (file)
index 0000000..cb04763
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 8
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:44.9,17.4
+ROTATION=270
diff --git a/microinforadiator/MiR09.env b/microinforadiator/MiR09.env
new file mode 100644 (file)
index 0000000..86ce423
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 9
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:28.1,30.3
+ROTATION=0
diff --git a/microinforadiator/MiR10.env b/microinforadiator/MiR10.env
new file mode 100644 (file)
index 0000000..cff20be
--- /dev/null
@@ -0,0 +1,7 @@
+MICROINFORADIATOR_ID=Micro information radiator 10
+ENABLE_SENSEHAT=false
+ENABLE_BLUETOOTH=false
+MQTT_BROKER_URL=127.0.0.1
+BLUETOOTH_ADDR=ac:f6:f7:96:35:09
+POSITION=Point:43.3,30.3
+ROTATION=0
diff --git a/microinforadiator/__init__.py b/microinforadiator/__init__.py
new file mode 100644 (file)
index 0000000..e69de29
diff --git a/microinforadiator/env.example b/microinforadiator/env.example
new file mode 100644 (file)
index 0000000..1fecc9b
--- /dev/null
@@ -0,0 +1,4 @@
+MICROINFORADIATOR_ID=micro-0
+ENABLE_SENSEHAT=true
+ENABLE_BLUETOOTH=true
+MQTT_BROKER_URL=192.168.2.105
diff --git a/microinforadiator/interaction.py b/microinforadiator/interaction.py
new file mode 100644 (file)
index 0000000..066436d
--- /dev/null
@@ -0,0 +1,233 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+from datetime import datetime,timedelta
+import json
+import math
+import os
+import threading
+#from threading import Thread
+import queue
+from time import sleep
+from proximity import get_devices, get_rssi
+#from LED.led_activities import blinkingArrow, lighting, blinking, show_hook, show_bee
+
+import state_factory
+
+
+INTERACTION_THREAD = "activities/interaction"
+################# ADD correct MAC Address ####################
+BD_ADDR = os.getenv('BLUETOOTH_ADDR')
+MY_ID = os.getenv('MICROINFORADIATOR_ID')
+duration = 12 #in seconds
+count = 0
+
+
+class InteractionController:
+
+    def __init__(self, incoming_messages, outgoing_messages):
+        self.incoming_messages = incoming_messages
+        self.outgoing_messages = outgoing_messages
+        self.current_state = None
+        self.rssi_pipe = []
+        self.last_known_user_state = {}
+        self.last_known_user_proximity = {}
+        self.sense_hat_worker = None
+        self.last_outgoing_message = None
+        self.current_content_start_time = datetime.utcnow()
+
+    def interaction_loop(self, ):
+        if self.current_state is None:
+            self.current_state = state_factory.get_idle_state()
+        try:
+            message = self.incoming_messages.get_nowait()
+            self.parse_incoming_message(message)
+            self.incoming_messages.task_done()
+        except queue.Empty:
+            pass
+        result = self.decide_on_interaction()
+        if json.dumps(result) != json.dumps(self.last_outgoing_message):
+            self.outgoing_messages.put(result)
+            self.last_outgoing_message = result
+        sleep(0.1)
+
+    def parse_incoming_message(self, message):
+        # for each user, store the last thing we know about them
+        # TODO: anything more useful to be done here?
+        if 'routing' in message['activity']:
+            self.last_known_user_state[message['user']] = message
+        elif 'proximity' in message['activity']:
+            self.last_known_user_proximity[message['user']] = message
+
+    def decide_on_interaction(self):
+        future_users = []
+        proximity_users = []
+        # look for users that are either in proximity or who are currently on the way according to the activity support service
+        for user_id in self.last_known_user_proximity:
+            proximity_time = datetime.fromisoformat(self.last_known_user_proximity[user_id]['time_to_action'])
+            now = datetime.utcnow()
+            if proximity_time + timedelta(seconds=3) > now and proximity_time - timedelta(seconds=3) < now:
+                proximity_users.append(user_id)
+
+        for user_id in self.last_known_user_state:
+            action_time = datetime.fromisoformat(self.last_known_user_state[user_id]['time_to_action'])
+            now = datetime.utcnow()
+            #print(action_time)
+            #print(now)
+            # a user counts as "active" if they will reach this device in the next 20 seconds or if they left less than 3 seconds ago
+            if action_time + timedelta(seconds=3) > now and action_time - timedelta(seconds=20) < now:
+                future_users.append(user_id)
+
+        active_users = list(proximity_users) + [u for u in future_users if u not in proximity_users]
+
+        new_state = self.current_state
+        if len(active_users) == 0:
+            # no one using the display? switch it off!
+            new_state = state_factory.get_idle_state()
+        else:
+            message = None
+            if active_users[0] in self.last_known_user_state:
+                message = self.last_known_user_state[active_users[0]]
+            else:
+                message = self.last_known_user_proximity[active_users[0]]
+            rec = message['user']
+            info = message['activity']
+            ttA = message['time_to_action']
+            color = message['color']
+            symbol = message['symbol']
+
+            direction = None
+            if 'route' in message and symbol == 'arrow':
+                direction = self.decide_signal_direction(message['route'])
+                if direction is None and os.getenv('FALLBACK_SIGNAL_DIRECTION', None) is not None:
+                    direction = os.getenv('FALLBACK_SIGNAL_DIRECTION')
+
+            if BD_ADDR is not None:
+                avg_rssi = self.check_rssi(BD_ADDR)
+                #print("average rssi:"+avg_rssi)
+            else:
+                avg_rssi = 9999 # if no BT address is configured, create no proximity events
+
+            #print(avg_rssi)
+
+            #conduct interaction
+            if avg_rssi <= 10:
+                if symbol == 'arrow' and direction is not None:
+                    new_state = state_factory.get_arrow_state(direction=direction, fg_color=color, bg_color=(0,0,0), blinking=True)
+                elif symbol in ['hook', 'bee']:
+                    new_state = state_factory.get_symbol_state(symbol, color=color, blinking = True)
+                else:
+                    new_state = state_factory.get_color_state(color=color, blinking=True)
+
+            if avg_rssi > 10:
+                new_state = state_factory.get_idle_state()
+
+        # Only change state if it is different from the currently active one
+        if  json.dumps(new_state) != json.dumps(self.current_state):
+            self.current_state = new_state
+            self.current_content_start_time = datetime.utcnow()
+
+            print("Changed to new state:", new_state["id"])
+            if os.getenv('ENABLE_SENSEHAT') == 'true':
+                if self.sense_hat_worker is not None and self.sense_hat_worker.is_alive():
+                    self.sense_hat_worker.obsolete = True
+                    self.sense_hat_worker.join()
+                self.sense_hat_worker = threading.Thread(target=self.load_new_sense_hat_state, args=(0,), name="Thread-SenseHAT")
+                self.sense_hat_worker.daemon = True
+                self.sense_hat_worker.obsolete = False
+                self.sense_hat_worker.start()
+
+        return (self.current_state, self.current_content_start_time.isoformat(), proximity_users)
+
+    def load_new_sense_hat_state(self, sub_state_index):
+        from sense_hat import SenseHat
+        sense = SenseHat()
+        sense.set_rotation(270) # we deploy MiRs with the HDMI port pointing to the left
+        if self.sense_hat_worker.obsolete:
+            return
+        sub_state = self.current_state["states"][sub_state_index]
+        if sub_state["function"] == "clear":
+            if "color" in sub_state:
+                #print("SenseHAT: clear", sub_state["color"])
+                sense.clear(sub_state["color"])
+            else:
+                #print("SenseHAT: clear")
+                sense.clear()
+        if sub_state["function"] == "set_pixels":
+            #print("SenseHAT: new pixels")
+            sense.set_pixels(sub_state["pixels"])
+        if "duration" in sub_state:
+            sleep(sub_state["duration"])
+
+        # only jump to next state if the current state is not the last one
+        next_sub_state_index = None
+        if sub_state_index < len(self.current_state["states"]) - 1:
+            next_sub_state_index = sub_state_index + 1
+        elif "loop" in self.current_state and self.current_state["loop"]:
+            # or jump back to the beginning of the state sequence if loop is true
+            next_sub_state_index = 0
+        if next_sub_state_index is not None:
+            self.load_new_sense_hat_state(next_sub_state_index)
+
+    def decide_signal_direction(self, route):
+        signal_range = 3.0
+        if route is None:
+            return None
+        my_position = os.getenv('POSITION', None)
+        if my_position is None or not my_position.startswith('Point:'):
+            return None
+        my_rotation = os.getenv('ROTATION', None)
+        if my_rotation is None:
+            return None
+        my_position = [float(coord) for coord in my_position.split(':')[1].split(',')]
+        my_rotation = float(my_rotation)
+        initial_far_point = (float(route[0]['x']), float(route[0]['y']))
+        next_far_point = None
+        in_range_from_here = False
+        for point in route:
+            point = (float(point['x']), float(point['y']))
+            distance = math.sqrt((point[0] - my_position[0]) ** 2 + (point[1] - my_position[1]) ** 2)
+            if distance < signal_range:
+                in_range_from_here = True
+            if in_range_from_here and distance > signal_range:
+                next_far_point = point
+                break
+        if next_far_point is None:
+            next_far_point = initial_far_point
+        angle = math.atan2(next_far_point[1] - my_position[1], next_far_point[0] - my_position[0])
+        angle *= 180 / math.pi
+        angle -= my_rotation
+        angle += 270
+        while angle < 0:
+            angle += 360
+        while angle >= 360:
+            angle -= 360
+        if angle >= 315 or angle < 45:
+            direction = 'up'
+        elif angle >= 45 and angle < 135:
+            direction = 'right'
+        elif angle >= 135 and angle < 225:
+            direction = 'down'
+        else:
+            direction = 'left'
+        return direction
+
+    def check_rssi(self, addr):
+        rssi_real = get_rssi(addr)
+        if rssi_real == None:
+            self.check_rssi(addr)
+        else:
+            rssi = abs(rssi_real)
+
+        #print("MAC: "+rec+" RSSI:"+str(rssi_real))
+        if len(self.rssi_pipe) >= 5:
+            self.rssi_pipe.pop(0)
+        self.rssi_pipe.append(rssi)
+        avg_rssi = sum(self.rssi_pipe) / len(self.rssi_pipe)
+        #print("Average RSSI value is: "+str(avg_rssi))
+
+        return avg_rssi
+
diff --git a/microinforadiator/joystick_control.py b/microinforadiator/joystick_control.py
new file mode 100644 (file)
index 0000000..dbc3caa
--- /dev/null
@@ -0,0 +1,175 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+from sense_hat import SenseHat
+from time import sleep
+from random import randint, choice
+import pygame, sys
+
+sense = SenseHat()
+red = (255,0,0)
+blue = (0,0,255)
+green = (0,255,0)
+yellow = (255,255,0)
+magenta = (255,0,255)
+cyan = (0,255,255)
+black = (0,0,0)
+cb = black
+ca = blue
+
+
+#Arrows
+def arrow_up ():
+        
+    arrow_up = [
+      
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,ca,cb,cb,cb,cb,cb,
+        cb,ca,cb,cb,cb,cb,cb,cb,
+        ca,ca,ca,ca,ca,ca,ca,ca,
+        cb,ca,cb,cb,cb,cb,cb,cb,
+        cb,cb,ca,cb,cb,cb,cb,cb,
+        cb,cb,cb,ca,cb,cb,cb,cb,
+        cb,cb,cb,cb,cb,cb,cb,cb
+
+    ]
+        
+    return arrow_up
+    
+def arrow_left ():
+
+        arrow_l = [
+        
+            cb,cb,cb,ca,cb,cb,cb,cb,
+            cb,cb,cb,ca,cb,cb,cb,cb,
+            cb,cb,cb,ca,cb,cb,cb,cb,
+            cb,cb,cb,ca,cb,cb,cb,cb,
+            ca,cb,cb,ca,cb,cb,ca,cb,
+            cb,ca,cb,ca,cb,ca,cb,cb,
+            cb,cb,ca,ca,ca,cb,cb,cb,
+            cb,cb,cb,ca,cb,cb,cb,cb
+
+        ]
+            
+        return arrow_l
+    
+def arrow_down ():
+        
+        arrow_d = [
+        
+            cb,cb,cb,cb,ca,cb,cb,cb,
+            cb,cb,cb,cb,cb,ca,cb,cb,
+            cb,cb,cb,cb,cb,cb,ca,cb,
+            ca,ca,ca,ca,ca,ca,ca,ca,
+            cb,cb,cb,cb,cb,cb,ca,cb,
+            cb,cb,cb,cb,cb,ca,cb,cb,
+            cb,cb,cb,cb,ca,cb,cb,cb,
+            cb,cb,cb,cb,cb,cb,cb,cb
+
+        ]
+            
+        return arrow_d
+    
+    #def arrow_right (color_background, color_arrow):
+def arrow_right ():
+
+        arrow_r = [
+        
+            cb,cb,cb,ca,cb,cb,cb,cb,
+            cb,cb,ca,ca,ca,cb,cb,cb,
+            cb,ca,cb,ca,cb,ca,cb,cb,
+            ca,cb,cb,ca,cb,cb,ca,cb,
+            cb,cb,cb,ca,cb,cb,cb,cb,
+            cb,cb,cb,ca,cb,cb,cb,cb,
+            cb,cb,cb,ca,cb,cb,cb,cb,
+            cb,cb,cb,ca,cb,cb,cb,cb
+
+        ]
+            
+        return arrow_r
+        
+def blinking():        
+    sense.clear(0,0,255)
+    sleep(1)
+    sense.clear()
+    sleep(1)
+    sense.clear(0,0,255)
+    sleep(1)
+    sense.clear()
+#    
+
+#AUDIO
+def audio():
+    pygame.mixer.init()
+    pygame.mixer.music.load("./Audio/glocke.mp3")
+    pygame.mixer.music.play()
+    while pygame.mixer.music.get_busy() == True:
+      
+      continue
+
+def Setup():          
+    prev_dir = 0
+    new_dir = 0
+    
+    while True:
+        
+        
+        for event in sense.stick.get_events():
+            print(event.direction, event.action)
+            if event.action == "pressed" or event.action == "held" or event.action == "released":
+                
+                if event.direction == "up" and event.action == "released":
+                    new_dir = "up"
+                    if(prev_dir == new_dir):
+                        sense.clear()
+                    else:
+                        a = arrow_right()
+                        sense.set_pixels(a)   
+                    prev_dir = "up"
+                            
+                if event.direction == "down" and event.action == "released":
+                    new_dir = "down"
+                    if(prev_dir == new_dir):
+                        sense.clear()  
+                    else:
+                        a = arrow_left()
+                        sense.set_pixels(a)
+                    prev_dir = "down"
+                        
+                if event.direction == "left" and event.action == "released":
+                    new_dir = "left"
+                    if(prev_dir == new_dir):
+                        sense.clear()   
+                    else:
+                        a = arrow_up()
+                        sense.set_pixels(a)   
+                    prev_dir = "left"
+                        
+                if event.direction == "right" and event.action == "released":
+                    new_dir = "right"
+                    if(prev_dir == new_dir):
+                        sense.clear()
+                    else:
+                        a = arrow_down()
+                        sense.set_pixels(a)
+                    prev_dir = "right"
+                        
+                if event.direction == "middle" and event.action == "released":
+                    new_dir = "middle"
+                    if(prev_dir == new_dir):
+                        sense.clear()
+                    else:
+                        #audio()
+                        blinking()
+                    prev_dir = "middle"
+                
+        pass
+               
+if __name__=='__main__':
+    sense.show_letter('O')
+    sleep(1)
+    sense.clear()
+    Setup()
diff --git a/microinforadiator/main.py b/microinforadiator/main.py
new file mode 100644 (file)
index 0000000..e5694be
--- /dev/null
@@ -0,0 +1,72 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+# initialize Micro-Information Radiator functions
+# set up IoT-Service (register, get ID, listen for actions) / MQTT pub and sub
+# Set-Up: create an IoT-device on IoT-Service Platform with Raspberry Pi data
+# Maintain: inform about your status (e.g. current location), identify Bluetooth Devices nearby and inform Platform about it (POST), check for activities for yourself (GET)
+
+from dotenv import load_dotenv
+import datetime
+import json
+import os
+import threading
+import queue
+import time
+
+import interaction
+
+if os.getenv('ENV_FILE_LOCATION') is not None:
+    load_dotenv(dotenv_path=os.getenv('ENV_FILE_LOCATION'))
+else:
+    load_dotenv()
+
+def notify_service_directory():
+    import requests, socket
+    try:
+        s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
+        s.connect(("8.8.8.8", 80))
+        ip = s.getsockname()[0]
+        s.close()
+        requests.get('http://localhost/mir.php?name=' + os.getenv('MICROINFORADIATOR_ID') + '&value=' + ip)
+    except:
+        pass
+
+notify_service_directory()
+
+from time import sleep
+from mqtt_service import start_mqtt, handle_publish
+
+
+incoming_messages = queue.Queue()
+outgoing_messages = queue.Queue()
+
+#set up MQTT (subscribe to topics, publish commands, listen on messages(receive information about seniors: persons-Bluetooth-ID, timestamp, seconds_until, activity[]))
+mqtt_client = start_mqtt(incoming_messages, outgoing_messages)
+mqtt_thread = threading.Thread(target=mqtt_client.loop_forever, name="Thread-MQTT")
+mqtt_thread.daemon = True
+mqtt_thread.start()
+
+interaction_controller = interaction.InteractionController(incoming_messages, outgoing_messages)
+try:
+    while True:
+        interaction_controller.interaction_loop()
+        try:
+            outgoing = outgoing_messages.get_nowait()
+            message = {
+                'mir_id': os.getenv('MICROINFORADIATOR_ID'),
+                'content': outgoing[0],
+                'start_time': outgoing[1],
+                'proximity_users': outgoing[2],
+            }
+            print(message)
+            mqtt_client.publish("activities/interaction", json.dumps(message))
+            outgoing_messages.task_done()
+        except queue.Empty:
+            pass
+except KeyboardInterrupt:
+    print('Received SIGINT, exiting...')
+
diff --git a/microinforadiator/mqtt_service.py b/microinforadiator/mqtt_service.py
new file mode 100644 (file)
index 0000000..b29ad7a
--- /dev/null
@@ -0,0 +1,86 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+import paho.mqtt.client as mqtt
+from threading import Thread
+from time import sleep
+import os
+import json
+
+##########################################################
+MY_THREAD = os.getenv('MICROINFORADIATOR_ID')
+##########################################################
+
+INTERACTION_THREAD = "activities/interaction"
+ACTIVITY_THREAD = "activities/info/muc"
+MiR_THREAD = "mir/status"
+TEST_PATH = "test_channel"
+
+
+def on_connect(client, userdata, flags, rc):
+    client.subscribe(MY_THREAD)
+    client.subscribe(INTERACTION_THREAD)
+    client.subscribe(ACTIVITY_THREAD)
+    client.subscribe(MiR_THREAD)
+    client.subscribe(TEST_PATH)
+    
+def on_message(client, userdata, msg):
+    message = json.loads(msg.payload.decode("utf-8"))
+    if msg.topic == MY_THREAD:
+        #print("message topic: ",msg.topic)
+        #print("message received: ",message)
+        client.incoming_messages.put(message)
+    
+        #send information about interaction to MiR-Controller via mqtt channel activities/interaction (my ID, the persons ID, interaction conducted, timestamp, duration)
+        #Thread(target=handle_publish,args=(client,INTERACTION_THREAD,interaction)).start()
+        
+def handle_publish(client,thread,msg):
+    client.publish(thread,str(msg))
+
+def on_publish(client,userdata,result):
+    print("data published")#create function for callback
+    pass
+
+def on_disconnect(client, userdata, rc):
+    if rc != 0:
+        print("Unexpected disconnection.")
+
+def start_mqtt(incoming_messages, outgoing_messages):
+
+    incoming_message_queue = incoming_messages
+    outgoing_message_queue = outgoing_messages
+    
+    client = mqtt.Client()
+    client.on_connect = on_connect
+    client.on_message = on_message
+    client.on_publish = on_publish
+    client.on_disconnect = on_disconnect
+
+    client.incoming_messages = incoming_messages
+    client.outgoing_messages = outgoing_messages
+
+    mqtt_broker_ips = []
+    if os.getenv('MQTT_BROKER_URL') is not None:
+        mqtt_broker_ips.append(os.getenv('MQTT_BROKER_URL'))
+    
+    while len(mqtt_broker_ips) > 0:
+        ip = mqtt_broker_ips[0]
+        del mqtt_broker_ips[0]
+        try:
+            client.connect(ip, 1883, 60)
+            print("connected to mqtt broker at " + ip)
+            break
+        except:
+            pass
+    if len(mqtt_broker_ips) == 0:
+        print("could not connect to any known mqtt broker")
+
+    return client
+    
+    
+if __name__=='__main__':
+    
+    start_mqtt()
diff --git a/microinforadiator/proximity.py b/microinforadiator/proximity.py
new file mode 100644 (file)
index 0000000..252c8b5
--- /dev/null
@@ -0,0 +1,40 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Laura Stojko
+
+from dotenv import load_dotenv
+import os
+
+if os.getenv('ENV_FILE_LOCATION') is not None:
+    load_dotenv(dotenv_path=os.getenv('ENV_FILE_LOCATION'))
+else:
+    load_dotenv()
+
+if os.getenv('ENABLE_BLUETOOTH') == 'true':
+    print(os.getenv('ENABLE_BLUETOOTH'))
+    from Bluetooth.bluetooth_controller import bluetooth_rssi,get_devices
+
+
+def get_devices():
+    detected_devices = []
+
+    if os.getenv('ENABLE_BLUETOOTH') == 'true':
+        bt_devices = get_devices()
+        detected_devices.extend(bt_devices)
+
+    # TODO: This is where support for simulated proximity events would probably be added.
+
+    return detected_devices
+
+def get_rssi(bd_addr):
+    rssi = None
+
+    if os.getenv('ENABLE_BLUETOOTH') == 'true':
+        rssi = bluetooth_rssi(bd_addr)
+    else:
+    # TODO: Do we need fictional RSSI values for simulated users?
+        rssi = 1
+    
+    return rssi
diff --git a/microinforadiator/requirements.txt b/microinforadiator/requirements.txt
new file mode 100644 (file)
index 0000000..620295c
--- /dev/null
@@ -0,0 +1,5 @@
+python-dotenv
+mercurial
+gattlib
+pybluez
+paho-mqtt
\ No newline at end of file
diff --git a/microinforadiator/run.sh b/microinforadiator/run.sh
new file mode 100755 (executable)
index 0000000..c052491
--- /dev/null
@@ -0,0 +1,4 @@
+#!/bin/sh
+export ENV_FILE_LOCATION='./MiR05.env'
+python3 main.py
+
diff --git a/microinforadiator/run_multi.py b/microinforadiator/run_multi.py
new file mode 100755 (executable)
index 0000000..79eea59
--- /dev/null
@@ -0,0 +1,79 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Julian Fietkau, Laura Stojko
+
+import collections
+import fcntl
+import glob
+import json
+import os
+import subprocess
+import sys
+import time
+
+if len(sys.argv) > 1:
+    if '--create-configs' in sys.argv:
+        defaults_filename = 'defaults.env'
+        defaults = collections.OrderedDict()
+        if os.path.isfile(defaults_filename):
+            with open(defaults_filename, 'r') as fp:
+                for line in fp.read().splitlines():
+                    if len(line) == 0:
+                        continue
+                    key, value = line.split('=')
+                    defaults[key] = value
+        with open('mir_evaluation_2020_09.json', 'r') as fp:
+            mir_data = json.load(fp)
+        for mir in mir_data["microinforadiators"]:
+            values = defaults.copy()
+            values['POSITION'] = mir['location']['type'] + ':' + ','.join([str(c) for c in mir['location']['coordinates']])
+            values['ROTATION'] = str(mir['rotation'])
+            values['MICROINFORADIATOR_ID'] = mir['name']
+            values.move_to_end('MICROINFORADIATOR_ID', last=False) # misleading method name: this actually moves the ID to the beginning
+            with open(mir['name'] + '.env', 'w') as fp:
+                for key in values:
+                    fp.write(key + '=' + values[key] + '\n')
+        sys.exit(0)
+    else:
+        mir_names = sys.argv[1:]
+else:
+    mir_names = [os.path.splitext(os.path.basename(conf))[0] for conf in glob.glob('*.env')]
+    if 'defaults' in mir_names:
+        mir_names.remove('defaults')
+
+mir_names.sort()
+processes = {}
+python_executable = '/usr/bin/python3' # TODO Windows compatibility?
+for mir_name in mir_names:
+    env = { 'ENV_FILE_LOCATION': './' + mir_name + '.env' }
+    print('Starting', mir_name, '...')
+    processes[mir_name] = subprocess.Popen([python_executable, '-u', 'main.py'], bufsize=0, env=env, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
+    flags = fcntl.fcntl(processes[mir_name].stdout, fcntl.F_GETFL) # get current p.stdout flags
+    fcntl.fcntl(processes[mir_name].stdout, fcntl.F_SETFL, flags | os.O_NONBLOCK)
+try:
+    while len(processes) > 0:
+        exited_processes = []
+        for mir_name in processes:
+            for pipe in 'stdout', 'stderr':
+                output = getattr(processes[mir_name], pipe).read()
+                if output is None:
+                    continue
+                for line in output.decode('utf-8').splitlines():
+                    if len(line) > 0:
+                        print('[' + mir_name + ']', line)
+            result = processes[mir_name].poll()
+            if result is not None:
+                exited_processes.append(mir_name)
+        for mir_name in exited_processes:
+            del processes[mir_name]
+            print(mir_name, 'terminated itself.')
+        time.sleep(0.1)
+    print('All child processes have terminated, exiting...')
+except KeyboardInterrupt:
+    print('Detected SIGINT, exiting...')
+    for mir_name in processes:
+        print('Terminating', mir_name, '...')
+        processes[mir_name].terminate()
+
diff --git a/microinforadiator/state_factory.py b/microinforadiator/state_factory.py
new file mode 100644 (file)
index 0000000..45207ad
--- /dev/null
@@ -0,0 +1,98 @@
+#!/usr/bin/env python3
+
+# SPDX-License-Identifier: GPL-3.0-or-later
+# This file is part of UrbanLife+ Activity Support. <https://fietkau.software/ulp_activity_support>
+# Copyright (C) 2019-2020 Julian Fietkau, Laura Stojko
+
+from LED.arrows import arrow_up, arrow_down, arrow_left, arrow_right
+from LED.symbols import hook, bee
+import os
+
+def get_idle_state():
+    return {
+        "id": "idle",
+        "states": [
+            {
+                "function": "clear"
+            }
+        ]
+    }
+
+def get_color_state(color, blinking):
+
+    result = {
+        "id": "color_"+','.join(map(str, color)),
+        "states": [
+           {
+             "function": "clear",
+             "color": color
+           }
+        ]
+    }
+
+    if blinking:
+        result["id"] += "_blink"
+        result["states"][0]["duration"] = 1.0
+        result['states'].append({"function": "clear", "duration": 1.0 })
+        result["loop"] = True
+
+    return result
+
+def get_arrow_state(direction, fg_color, bg_color, blinking):
+    if direction not in ["up", "down", "left", "right"]:
+        raise ValueError(direction)
+
+    result = {
+        "id": "arrow_"+direction+"_"+",".join(map(str,fg_color)),
+        "states": [
+           {
+             "function": "set_pixels",
+           }
+        ]
+    }
+
+    pixels = None
+    if direction == "up":
+        pixels = arrow_up(bg_color, fg_color)
+    if direction == "down":
+        pixels = arrow_down(bg_color, fg_color)
+    if direction == "left":
+        pixels = arrow_left(bg_color, fg_color)
+    if direction == "right":
+        pixels = arrow_right(bg_color, fg_color)
+    result["states"][0]["pixels"] = pixels
+
+    if blinking:
+        result["id"] += "_blink"
+        result["states"][0]["duration"] = 1.0
+        result['states'].append({"function": "clear", "duration": 1.0 })
+        result["loop"] = True
+
+    return result
+
+def get_symbol_state(symbol, color, blinking):
+    pixels = None
+    
+    if symbol == "hook":
+        pixels = hook(color)
+    elif symbol == "bee":
+        pixels = bee()
+        
+    result = {
+        "id": "symbol_"+symbol+"_"+",".join(map(str,color)),
+        "states": [
+           {
+             "function": "set_pixels",
+           }
+        ]
+    }
+     
+    result["states"][0]["pixels"] = pixels
+
+    if blinking:
+        result["id"] += "_blink"
+        result["states"][0]["duration"] = 1.0
+        result['states'].append({"function": "clear", "duration": 1.0 })
+        result["loop"] = True
+
+    return result