2024-02-08 17:49:47 +00:00
|
|
|
import socket
|
|
|
|
from flask import Flask, render_template, request, jsonify
|
|
|
|
from pythonosc import udp_client, dispatcher, osc_server
|
2024-02-08 18:24:47 +00:00
|
|
|
import json
|
2024-02-08 18:41:23 +00:00
|
|
|
import pymysql
|
|
|
|
|
|
|
|
pymysql.install_as_MySQLdb()
|
2024-02-08 17:49:47 +00:00
|
|
|
|
|
|
|
app = Flask(__name__)
|
|
|
|
|
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",
|
|
|
|
"user": "root",
|
|
|
|
"password": "",
|
|
|
|
"database": "pixeltube_db"
|
|
|
|
},
|
|
|
|
"network": {
|
|
|
|
"SOURCE_NETWORK_IP": "10.0.0.0/8",
|
|
|
|
"DESTINATION_NETWORK_IP": "192.168.0.0/8"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
with open('config.json', 'w') as config_file:
|
|
|
|
json.dump(config, config_file, indent=4)
|
|
|
|
|
|
|
|
# Use MySQL configuration from the config file
|
|
|
|
app.config['MYSQL_HOST'] = config['mysql']['host']
|
|
|
|
app.config['MYSQL_USER'] = config['mysql']['user']
|
|
|
|
app.config['MYSQL_PASSWORD'] = config['mysql']['password']
|
|
|
|
app.config['MYSQL_DB'] = config['mysql']['database']
|
|
|
|
|
|
|
|
# Use network configuration from the config file
|
|
|
|
SOURCE_NETWORK_IP = config['network']['SOURCE_NETWORK_IP']
|
|
|
|
DESTINATION_NETWORK_IP = config['network']['DESTINATION_NETWORK_IP']
|
2024-02-08 17:49:47 +00:00
|
|
|
|
|
|
|
mysql = MySQL(app)
|
|
|
|
|
|
|
|
# Art-Net settings
|
|
|
|
ARTNET_PORT_IN = 6454 # Standard Art-Net input port
|
|
|
|
ARTNET_PORT_OUT = 6455 # Standard Art-Net output port
|
|
|
|
|
|
|
|
# Create an Art-Net dispatcher
|
|
|
|
artnet_dispatcher = dispatcher.Dispatcher()
|
|
|
|
|
|
|
|
def forward_artnet_handler(address, *args):
|
|
|
|
# Create an Art-Net packet from the received OSC message
|
|
|
|
artnet_packet = b'Art-Net\x00' + b'\x00' * 7 + bytes(args[0], 'utf-8') + b'\x00' * 2 + b'\x00\x00'
|
|
|
|
|
|
|
|
# Send the Art-Net packet to the destination network
|
|
|
|
with socket.socket(socket.AF_INET, socket.SOCK_DGRAM) as sock:
|
|
|
|
sock.sendto(artnet_packet, (DESTINATION_NETWORK_IP, ARTNET_PORT_OUT))
|
|
|
|
|
|
|
|
# Map the Art-Net handler to the OSC address
|
|
|
|
artnet_dispatcher.map('/artnet', forward_artnet_handler)
|
|
|
|
|
|
|
|
# Create an Art-Net server listening on the source network
|
|
|
|
artnet_server = osc_server.ThreadingOSCUDPServer((SOURCE_NETWORK_IP, ARTNET_PORT_IN), artnet_dispatcher)
|
|
|
|
artnet_server_thread = artnet_server.serve_forever()
|
|
|
|
|
|
|
|
# Function to register a tube in the database
|
|
|
|
def register_tube(mac_address):
|
|
|
|
cur = mysql.connection.cursor()
|
|
|
|
|
|
|
|
# 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)
|
|
|
|
mysql.connection.commit()
|
|
|
|
|
|
|
|
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.'})
|
|
|
|
|
|
|
|
# Function to retrieve registered tubes from the database
|
|
|
|
def get_tubes():
|
|
|
|
cur = mysql.connection.cursor()
|
|
|
|
cur.execute("SELECT * FROM tubes")
|
|
|
|
tubes = cur.fetchall()
|
|
|
|
cur.close()
|
|
|
|
return tubes
|
|
|
|
|
2024-02-08 18:19:24 +00:00
|
|
|
@app.route('/get_assigned_params/<tube_id>', methods=['GET'])
|
|
|
|
def get_assigned_params(tube_id):
|
|
|
|
try:
|
|
|
|
cur = mysql.connection.cursor()
|
|
|
|
cur.execute("SELECT universe, dmx_address FROM tubes WHERE mac_address = %s", (tube_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-02-08 17:49:47 +00:00
|
|
|
# Index route for the web interface
|
|
|
|
@app.route('/')
|
|
|
|
def index():
|
|
|
|
tubes = get_tubes()
|
|
|
|
return render_template('index.html', tubes=tubes)
|
|
|
|
|
|
|
|
def main():
|
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
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|