2024-02-08 17:49:47 +00:00
|
|
|
from flask import Flask, render_template, request, jsonify
|
2024-02-08 18:24:47 +00:00
|
|
|
import json
|
2024-02-08 21:03:14 +00:00
|
|
|
import MySQLdb
|
2024-02-12 20:55:08 +00:00
|
|
|
import paho.mqtt.client as mqtt
|
|
|
|
import threading
|
2024-02-12 23:45:59 +00:00
|
|
|
import sacn
|
2024-02-12 21:46:38 +00:00
|
|
|
import os
|
2024-02-12 21:50:54 +00:00
|
|
|
from getmac import get_mac_address
|
2024-02-12 22:14:39 +00:00
|
|
|
import netifaces
|
2024-02-08 17:49:47 +00:00
|
|
|
|
|
|
|
app = Flask(__name__)
|
|
|
|
|
2024-02-12 23:12:33 +00:00
|
|
|
wlan_mac_address = str(get_mac_address(interface="wlan0"))
|
2024-02-12 21:46:05 +00:00
|
|
|
|
2024-02-08 18:24:47 +00:00
|
|
|
# Read configuration from config.json
|
|
|
|
try:
|
|
|
|
with open('config.json', 'r') as config_file:
|
|
|
|
config = json.load(config_file)
|
|
|
|
except FileNotFoundError:
|
|
|
|
# Create config.json with default values if it doesn't exist
|
|
|
|
config = {
|
|
|
|
"mysql": {
|
|
|
|
"host": "localhost",
|
2024-02-12 16:42:27 +00:00
|
|
|
"user": "pxm",
|
|
|
|
"password": "pixel",
|
2024-02-08 18:24:47 +00:00
|
|
|
"database": "pixeltube_db"
|
2024-02-12 21:24:16 +00:00
|
|
|
},
|
|
|
|
"artnet": {
|
|
|
|
"universe_count": 1
|
2024-02-08 18:24:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
with open('config.json', 'w') as config_file:
|
|
|
|
json.dump(config, config_file, indent=4)
|
|
|
|
|
2024-02-08 20:58:44 +00:00
|
|
|
database = config['mysql']['database']
|
2024-02-08 17:49:47 +00:00
|
|
|
|
2024-02-08 21:03:14 +00:00
|
|
|
db = MySQLdb.connect(
|
2024-02-08 20:58:44 +00:00
|
|
|
host=config['mysql']['host'],
|
|
|
|
user=config['mysql']['user'],
|
|
|
|
password=config['mysql']['password'],
|
|
|
|
database=config['mysql']['database'],
|
|
|
|
)
|
2024-02-08 17:49:47 +00:00
|
|
|
|
2024-02-12 21:24:16 +00:00
|
|
|
universe_count = config['artnet']['universe_count']
|
2024-02-12 20:55:08 +00:00
|
|
|
|
2024-02-12 21:46:05 +00:00
|
|
|
mqtt_client_id = "PiXelTubeMaster-"+wlan_mac_address
|
2024-02-12 00:27:24 +00:00
|
|
|
|
2024-02-08 17:49:47 +00:00
|
|
|
# Function to register a tube in the database
|
|
|
|
def register_tube(mac_address):
|
2024-02-08 20:58:44 +00:00
|
|
|
cur = db.cursor()
|
2024-02-08 17:49:47 +00:00
|
|
|
|
|
|
|
# Check if the tube already exists in the database
|
|
|
|
cur.execute("SELECT * FROM tubes WHERE mac_address = %s", (mac_address,))
|
|
|
|
existing_tube = cur.fetchone()
|
|
|
|
|
|
|
|
if existing_tube:
|
|
|
|
# Tube already exists, do nothing for now
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
# Tube is new, insert into the database
|
|
|
|
cur.execute("INSERT INTO tubes (mac_address, universe, dmx_address, lamp_power) VALUES (%s, %s, %s, %s)",
|
|
|
|
(mac_address, 0, 1, 0)) # Universe 0, DMX Address 1, Lamp Power Off (False)
|
|
|
|
cur.close()
|
|
|
|
|
|
|
|
# Registration system route
|
|
|
|
@app.route('/register_tube', methods=['POST'])
|
|
|
|
def register_tube_route():
|
|
|
|
mac_address = request.form.get('mac_address')
|
|
|
|
register_tube(mac_address)
|
|
|
|
return jsonify({'success': True, 'message': 'Tube registered successfully.'})
|
|
|
|
|
2024-02-09 16:35:06 +00:00
|
|
|
|
2024-02-12 00:27:24 +00:00
|
|
|
@app.route('/get_assigned_params/<tube_unique_id>', methods=['GET'])
|
|
|
|
def get_assigned_params(tube_unique_id):
|
2024-02-08 18:19:24 +00:00
|
|
|
try:
|
2024-02-08 20:58:44 +00:00
|
|
|
cur = db.cursor()
|
2024-02-12 00:27:24 +00:00
|
|
|
cur.execute("SELECT universe, dmx_address FROM tubes WHERE mac_address = %s", (tube_unique_id,))
|
2024-02-08 18:19:24 +00:00
|
|
|
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-02-12 20:55:08 +00:00
|
|
|
def flask_api():
|
2024-02-08 18:19:24 +00:00
|
|
|
app.run(host='0.0.0.0', port=5000)
|
2024-02-08 17:49:47 +00:00
|
|
|
|
2024-02-12 22:14:39 +00:00
|
|
|
def get_eth0_ip():
|
|
|
|
try:
|
|
|
|
# Get the IP address of the eth0 interface
|
|
|
|
eth0_ip = netifaces.ifaddresses('eth0')[netifaces.AF_INET][0]['addr']
|
|
|
|
return eth0_ip
|
|
|
|
except (KeyError, IndexError, OSError) as e:
|
|
|
|
print(f"Error getting eth0 IP: {e}")
|
|
|
|
return None
|
2024-02-12 23:22:20 +00:00
|
|
|
|
2024-02-12 23:26:26 +00:00
|
|
|
def on_connect(client, userdata, flags, reason_code, properties):
|
|
|
|
if reason_code == 0:
|
2024-02-12 21:46:05 +00:00
|
|
|
print("Connected to MQTT Broker!")
|
|
|
|
else:
|
2024-02-12 23:26:26 +00:00
|
|
|
print("Failed to connect, return code %d\n", str(reason_code))
|
2024-02-12 23:22:20 +00:00
|
|
|
|
|
|
|
def connect_mqtt():
|
2024-02-12 23:11:47 +00:00
|
|
|
# Set Connecting Client ID
|
2024-02-12 23:29:27 +00:00
|
|
|
client = mqtt.Client(mqtt.CallbackAPIVersion.VERSION2)
|
2024-02-12 23:11:47 +00:00
|
|
|
# client.username_pw_set(username, password)
|
2024-02-12 23:00:51 +00:00
|
|
|
client.on_connect = on_connect
|
|
|
|
client.connect("localhost", 1883)
|
2024-02-12 21:46:05 +00:00
|
|
|
return client
|
|
|
|
|
|
|
|
|
2024-02-12 21:24:16 +00:00
|
|
|
def mqtt_publisher(universe):
|
2024-02-12 23:45:59 +00:00
|
|
|
universe = universe-1
|
2024-02-12 21:46:05 +00:00
|
|
|
mqtt_client = connect_mqtt()
|
2024-02-12 21:24:16 +00:00
|
|
|
try:
|
2024-02-12 23:45:59 +00:00
|
|
|
receiver = sacn.sACNreceiver(bind_address=get_eth0_ip())
|
2024-02-12 23:51:45 +00:00
|
|
|
print(str(get_eth0_ip()))
|
2024-02-12 23:45:59 +00:00
|
|
|
receiver.start()
|
|
|
|
|
2024-02-13 12:32:14 +00:00
|
|
|
def on_dmx_frame(packet, universe, channel_data): # packet type: sacn.DataPacket
|
2024-02-12 23:45:59 +00:00
|
|
|
print(packet.dmxData)
|
|
|
|
global dmx_values
|
|
|
|
dmx_values = packet.dmxData
|
2024-02-13 12:32:14 +00:00
|
|
|
|
2024-02-12 23:49:26 +00:00
|
|
|
receiver.join_multicast(universe)
|
2024-02-13 12:34:31 +00:00
|
|
|
receiver.register_listener(universe=universe, func=on_dmx_frame)
|
2024-02-12 23:55:03 +00:00
|
|
|
|
2024-02-12 21:24:16 +00:00
|
|
|
while True:
|
|
|
|
if dmx_values is not None:
|
|
|
|
for channel, value in enumerate(dmx_values):
|
|
|
|
# Create MQTT topic based on the universe and channel
|
2024-02-12 23:10:32 +00:00
|
|
|
topic = f"{universe}/{channel}"
|
2024-02-12 21:24:16 +00:00
|
|
|
|
|
|
|
# Publish the DMX value to the MQTT topic
|
2024-02-12 23:08:55 +00:00
|
|
|
mqtt_client.publish(topic, value)
|
2024-02-12 22:01:00 +00:00
|
|
|
print(str(universe+" "+channel+" "+value))
|
2024-02-12 21:24:16 +00:00
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
print(f"Error in universe {universe}: {e}")
|
|
|
|
|
|
|
|
def start_mqtt_publishers(universe_count):
|
2024-02-12 23:45:59 +00:00
|
|
|
used_universes = universe_count
|
2024-02-12 22:45:37 +00:00
|
|
|
print("universe count: "+str(used_universes))
|
2024-02-12 21:59:21 +00:00
|
|
|
universes_to_publish = list(range(1, used_universes + 1))
|
2024-02-12 21:24:16 +00:00
|
|
|
# Create and start a thread for each universe
|
2024-02-12 23:30:06 +00:00
|
|
|
for universe in universes_to_publish:
|
|
|
|
threads = [threading.Thread(target=mqtt_publisher, args=(universe,))]
|
2024-02-12 21:24:16 +00:00
|
|
|
|
|
|
|
for thread in threads:
|
|
|
|
thread.start()
|
2024-02-12 20:55:08 +00:00
|
|
|
|
2024-02-12 21:24:16 +00:00
|
|
|
for thread in threads:
|
|
|
|
thread.join()
|
2024-02-12 20:55:08 +00:00
|
|
|
|
2024-02-08 17:49:47 +00:00
|
|
|
if __name__ == "__main__":
|
2024-02-12 21:24:16 +00:00
|
|
|
start_mqtt_publishers(universe_count)
|
2024-02-12 22:45:37 +00:00
|
|
|
# flask_thread = threading.Thread(target=flask_api())
|
|
|
|
# flask_thread.start()
|
|
|
|
# flask_thread.join()
|
2024-02-12 22:04:06 +00:00
|
|
|
|