2024-02-18 16:00:46 +00:00
|
|
|
from flask import Flask, request, jsonify
|
|
|
|
import json
|
|
|
|
from MySQLdb import connect
|
|
|
|
import paho.mqtt.client as mqtt
|
|
|
|
import python_artnet as Artnet
|
|
|
|
import os
|
|
|
|
from getmac import get_mac_address
|
|
|
|
import time
|
2024-02-19 15:01:38 +00:00
|
|
|
from multiprocessing import Process, Pipe, Queue
|
2024-02-19 15:07:04 +00:00
|
|
|
from queue import Empty
|
2024-02-20 14:55:03 +00:00
|
|
|
import json
|
2024-02-18 16:00:46 +00:00
|
|
|
|
2024-05-12 19:13:10 +00:00
|
|
|
# create flask webaüüs
|
2024-02-18 16:00:46 +00:00
|
|
|
app = Flask(__name__)
|
|
|
|
|
2024-05-12 19:13:10 +00:00
|
|
|
# get macaddress of wlan0 interface
|
2024-02-18 16:00:46 +00:00
|
|
|
wlan_mac_address = str(get_mac_address(interface="wlan0"))
|
|
|
|
|
|
|
|
# Read configuration from config.json
|
2024-02-08 18:24:47 +00:00
|
|
|
try:
|
2024-02-18 16:00:46 +00:00
|
|
|
with open('config.json', 'r') as config_file:
|
|
|
|
config = json.load(config_file)
|
2024-02-08 18:24:47 +00:00
|
|
|
except FileNotFoundError:
|
2024-02-18 16:00:46 +00:00
|
|
|
# Create config.json with default values if it doesn't exist
|
|
|
|
config = {
|
|
|
|
"mysql": {
|
|
|
|
"host": "localhost",
|
|
|
|
"user": "pxm",
|
|
|
|
"password": "pixel",
|
|
|
|
"database": "pixeltube_db"
|
|
|
|
},
|
|
|
|
}
|
|
|
|
with open('config.json', 'w') as config_file:
|
|
|
|
json.dump(config, config_file, indent=4)
|
2024-05-12 19:13:10 +00:00
|
|
|
|
|
|
|
# create DB client from var's
|
2024-02-18 16:00:46 +00:00
|
|
|
database = config['mysql']['database']
|
|
|
|
|
|
|
|
db = connect(
|
|
|
|
host=config['mysql']['host'],
|
|
|
|
user=config['mysql']['user'],
|
|
|
|
password=config['mysql']['password'],
|
|
|
|
database=config['mysql']['database'],
|
|
|
|
)
|
|
|
|
|
|
|
|
db.autocommit(True)
|
|
|
|
|
2024-05-12 19:13:10 +00:00
|
|
|
# set client id via unique id
|
2024-02-18 16:00:46 +00:00
|
|
|
mqtt_client_id = "PiXelTubeMaster-"+wlan_mac_address
|
|
|
|
|
|
|
|
# Function to register a tube in the database
|
|
|
|
def register_tube(mac_address):
|
|
|
|
cur1 = db.cursor()
|
|
|
|
# Check if the tube already exists in the database
|
|
|
|
cur1.execute("SELECT * FROM tubes WHERE mac_address = %s", (mac_address,))
|
|
|
|
existing_tube = cur1.fetchone()
|
|
|
|
|
|
|
|
# Check if the tube exsist. If it doesn't create a new db row
|
|
|
|
if not existing_tube:
|
|
|
|
cur1.execute("INSERT INTO tubes (mac_address, universe, dmx_address) VALUES (%s, %s, %s)",
|
|
|
|
(mac_address, 0, 1))
|
|
|
|
else:
|
|
|
|
pass
|
|
|
|
cur1.close()
|
|
|
|
|
|
|
|
# Registration system route
|
|
|
|
@app.route('/register_tube', methods=['POST'])
|
|
|
|
def register_tube_route():
|
|
|
|
mac_address = request.form.get('mac_address')
|
|
|
|
register_tube(str(mac_address))
|
|
|
|
return jsonify({'success': True, 'message': 'Tube registered successfully.'})
|
|
|
|
|
|
|
|
|
|
|
|
@app.route('/get_assigned_params/<tube_unique_id>', methods=['GET'])
|
|
|
|
def get_assigned_params(tube_unique_id):
|
|
|
|
try:
|
|
|
|
cur = db.cursor()
|
|
|
|
cur.execute("SELECT universe, dmx_address FROM tubes WHERE mac_address = %s", (tube_unique_id,))
|
|
|
|
result = cur.fetchone()
|
|
|
|
cur.close()
|
|
|
|
|
|
|
|
if result:
|
|
|
|
universe, dmx_address = result
|
|
|
|
return jsonify({'success': True, 'universe': universe, 'dmx_address': dmx_address})
|
|
|
|
else:
|
|
|
|
return jsonify({'success': False, 'message': 'Tube not found in the database'})
|
|
|
|
except Exception as e:
|
|
|
|
return jsonify({'success': False, 'message': f'Error: {e}'})
|
|
|
|
|
2024-05-12 19:13:10 +00:00
|
|
|
# run flask api
|
2024-02-18 16:00:46 +00:00
|
|
|
def flask_api():
|
|
|
|
app.run(host='192.168.0.1', port=5000)
|
|
|
|
|
|
|
|
def get_eth0_ip():
|
|
|
|
try:
|
|
|
|
# Get the IP address of the eth0 interface
|
|
|
|
eth0_ip = str(os.system("ip -4 -o addr show eth0 | awk '{print $4}' | cut -d '/' -f 1 "))
|
|
|
|
return eth0_ip
|
|
|
|
except (KeyError, IndexError, OSError) as e:
|
|
|
|
print(f"Error getting eth0 IP: {e}")
|
|
|
|
exit
|
|
|
|
|
|
|
|
def on_connect(client, userdata, flags, reason_code, properties):
|
2024-05-12 19:13:10 +00:00
|
|
|
# print debug message when mqtt client is connected
|
|
|
|
if reason_code == 0:
|
|
|
|
print("Connected to MQTT Broker!")
|
|
|
|
else:
|
|
|
|
print("Failed to connect, return code %d\n", str(reason_code))
|
2024-02-18 16:00:46 +00:00
|
|
|
|
|
|
|
def connect_mqtt():
|
|
|
|
# Set Connecting Client ID
|
|
|
|
client = mqtt.Client(mqtt.CallbackAPIVersion.VERSION2)
|
|
|
|
# client.username_pw_set(username, password)
|
|
|
|
client.on_connect = on_connect
|
|
|
|
client.connect("localhost", 1883)
|
|
|
|
return client
|
|
|
|
|
2024-02-20 15:10:01 +00:00
|
|
|
def mqtt_publisher(ti_queue):
|
2024-05-12 19:13:10 +00:00
|
|
|
# set index and old data index to none as default value
|
2024-02-20 15:10:01 +00:00
|
|
|
tube_index = None
|
2024-02-20 15:53:09 +00:00
|
|
|
tube_index_old = None
|
2024-02-18 16:00:46 +00:00
|
|
|
# Create and start a thread for each universe
|
|
|
|
mqtt_client = connect_mqtt()
|
|
|
|
artnetBindIp = get_eth0_ip()
|
2024-05-12 19:13:10 +00:00
|
|
|
# start arnet listner
|
2024-02-18 16:00:46 +00:00
|
|
|
artNet = Artnet.Artnet(BINDIP = artnetBindIp, DEBUG = True, SHORTNAME = "PiXelTubeMaster", LONGNAME = "PiXelTubeMaster", PORT = 6454)
|
2024-05-12 19:13:10 +00:00
|
|
|
# start publishing process
|
2024-02-18 16:00:46 +00:00
|
|
|
while True:
|
|
|
|
try:
|
2024-05-12 19:13:10 +00:00
|
|
|
# try to get new tube index and if it fails use old index
|
2024-02-20 15:14:05 +00:00
|
|
|
try:
|
|
|
|
tube_index = ti_queue.get(block=False)
|
|
|
|
tube_index_old = tube_index
|
|
|
|
except Empty:
|
|
|
|
if tube_index_old is not None:
|
|
|
|
tube_index = tube_index_old
|
|
|
|
else:
|
|
|
|
tube_index = None
|
2024-02-18 16:00:46 +00:00
|
|
|
# Gets whatever the last Art-Net packet we received is
|
2024-05-12 19:13:10 +00:00
|
|
|
artNetPacket = artNet.readBuffer()
|
2024-02-18 16:00:46 +00:00
|
|
|
# Make sure we actually *have* a packet
|
2024-02-18 16:43:15 +00:00
|
|
|
if artNetPacket is not None:
|
2024-02-19 15:01:38 +00:00
|
|
|
if tube_index is not None:
|
|
|
|
for index_row in tube_index:
|
2024-05-12 19:13:10 +00:00
|
|
|
if artNetPacket[int(index_row[1])]:
|
2024-02-18 16:43:15 +00:00
|
|
|
dmx_address = int(index_row[2])
|
|
|
|
#Define RGB values per pixel
|
2024-02-20 16:35:58 +00:00
|
|
|
p1_g, p1_b, p1_r, p2_g, p2_b, p2_r, p3_g, p3_b, p3_r, p4_g, p4_b, p4_r, p5_g, p5_b, p5_r, p6_g, p6_b, p6_r = artNetPacket.data[dmx_address-1], artNetPacket.data[dmx_address], artNetPacket.data[dmx_address+1], artNetPacket.data[dmx_address+2], artNetPacket.data[dmx_address+3], artNetPacket.data[dmx_address+4], artNetPacket.data[dmx_address+5], artNetPacket.data[dmx_address+6], artNetPacket.data[dmx_address+7], artNetPacket.data[dmx_address+8], artNetPacket.data[dmx_address+9], artNetPacket.data[dmx_address+10], artNetPacket.data[dmx_address+11], artNetPacket.data[dmx_address+12], artNetPacket.data[dmx_address+13], artNetPacket.data[dmx_address+14], artNetPacket.data[dmx_address+15], artNetPacket.data[dmx_address+16]
|
2024-02-18 16:43:15 +00:00
|
|
|
|
|
|
|
# Pixel topics
|
|
|
|
p1_topic = "tube-"+str(index_row[0])+"/pixel_colors"
|
|
|
|
|
|
|
|
# Publish pixel topic
|
2024-02-19 15:24:48 +00:00
|
|
|
colors = [[p1_r, p1_g, p1_b], [p2_r, p2_g, p2_b], [p3_r, p3_g, p3_b], [p4_r, p4_g, p4_b], [p5_r, p5_g, p5_b], [p6_r, p6_g, p6_b]]
|
|
|
|
result_str = [str(color) for color in colors]
|
|
|
|
result = str(result_str)
|
|
|
|
mqtt_client.publish(p1_topic, result)
|
2024-05-12 19:13:10 +00:00
|
|
|
# close artnet if KeyboardInterrupt
|
2024-02-18 16:00:46 +00:00
|
|
|
except KeyboardInterrupt:
|
|
|
|
artNet.close()
|
2024-02-20 15:10:01 +00:00
|
|
|
|
2024-02-19 15:01:38 +00:00
|
|
|
def tube_index_updater(ti_queue):
|
|
|
|
while True:
|
2024-05-12 19:13:10 +00:00
|
|
|
# try to create db cursor and get tube index data tu put in into queue
|
2024-02-19 15:01:38 +00:00
|
|
|
try:
|
|
|
|
cur = db.cursor()
|
|
|
|
cur.execute("SELECT mac_address, universe, dmx_address FROM tubes")
|
|
|
|
tube_index = cur.fetchall()
|
|
|
|
cur.close()
|
2024-02-20 15:01:33 +00:00
|
|
|
ti_queue.put(tube_index)
|
2024-02-19 15:01:38 +00:00
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
time.sleep(5)
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2024-05-12 19:13:10 +00:00
|
|
|
# start pipe as tube index receiver sender
|
2024-02-19 15:55:16 +00:00
|
|
|
(ti_receiver,ti_sender) = Pipe(True)
|
|
|
|
ti_queue = Queue()
|
|
|
|
ti_updater_thread = Process(target=tube_index_updater, args=(ti_queue, ))
|
|
|
|
ti_updater_thread.start()
|
2024-02-20 15:10:01 +00:00
|
|
|
publisher_thread = Process(target=mqtt_publisher, args=(ti_queue, ))
|
2024-02-19 15:55:16 +00:00
|
|
|
publisher_thread.start()
|
2024-02-19 16:18:42 +00:00
|
|
|
flask_thread = Process(target=flask_api)
|
|
|
|
flask_thread.start()
|