server-rack-led/main.py

224 lines
7.3 KiB
Python
Raw Normal View History

from base64 import b16decode
2021-11-08 20:36:55 +00:00
import time
import os
from datetime import datetime
from requests import request
2022-03-26 19:14:20 +00:00
try:
from rpi_ws281x import *
import RPi.GPIO as GPIO
from flask import Flask, render_template, request, redirect, url_for, flash, jsonify
import json
from argparse import ArgumentParser
import threading
import paho.mqtt.client as mqtt
2022-03-26 19:14:20 +00:00
except ImportError:
print('Some modules are missing. Try to install them with "pip3 install -r requirements.txt"')
exit()
2021-11-08 20:36:55 +00:00
# LED strip configuration:
LED_COUNT = 60 # Number of LED pixels.
LED_PIN = 18 # GPIO pin connected to the pixels (18 uses PWM!).
#LED_PIN = 10 # GPIO pin connected to the pixels (10 uses SPI /dev/spidev0.0).
LED_FREQ_HZ = 800000 # LED signal frequency in hertz (usually 800khz)
LED_DMA = 10 # DMA channel to use for generating signal (try 10)
LED_BRIGHTNESS = 255 # Set to 0 for darkest and 255 for brightest
LED_INVERT = False # True to invert the signal (when using NPN transistor level shift)
LED_CHANNEL = 0 # set to '1' for GPIOs 13, 19, 41, 45 or 53
def debug_print(message: str):
now = datetime.now()
current_time = now.strftime("%H:%M:%S")
print('[DEBUG]['+current_time+'] '+message)
2021-11-08 20:36:55 +00:00
def wheel(pos):
if pos < 85:
return Color(pos * 3, 255 - pos * 3, 0)
elif pos < 170:
pos -= 85
return Color(255 - pos * 3, 0, pos * 3)
else:
pos -= 170
return Color(0, pos * 3, 255 - pos * 3)
2021-11-09 22:16:38 +00:00
def rainbowCycle(strip, wait_ms=5, iterations=10):
2021-11-08 20:36:55 +00:00
for j in range(256*iterations):
if mode > 0:
break
elif power == "False":
break
2021-11-08 20:36:55 +00:00
for i in range(strip.numPixels()):
strip.setPixelColor(i, wheel((int(i * 256 / strip.numPixels()) + j) & 255))
strip.show()
time.sleep(wait_ms/1000.0)
def setColor(strip, color, wait_ms=10):
2021-11-08 20:36:55 +00:00
for i in range(strip.numPixels()):
strip.setPixelColor(i, color)
strip.show()
time.sleep(wait_ms/1000.0)
def mqtt_on_connect(client, userdata, flags, rc):
client.subscribe("server-rack-led/power")
client.subscribe("server-rack-led/mode")
client.subscribe("server-rack-led/rgb")
if __name__ == '__main__':
app = Flask(__name__)
#api
@app.route('/api/')
def index():
return '''
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Server Rack LED stripe</title>
</head>
<style>
@import url('https://fonts.googleapis.com/css2?family=Roboto&display=swap');
body {
font-family: 'Roboto', sans-serif;
background-color: #1E1E1E;
color: #fff;
}
2021-11-08 20:36:55 +00:00
code {
background-color: #333333;
border: 5px solid #333333;
border-radius: 10px;
2021-11-08 20:36:55 +00:00
.head-text {
font-size: 25px;
#put text in middle of page
margin-left: auto;
margin-right: auto;
left: 50%;
text-align: center;
}
</style>
<body>
<p class="head-text">My Server Rack LED stripe control API</p>
<br>
<p>How to use the API:</p>
<code>/power?power=True</code>
<br>
<code>/power?power=False</code>
<br>
<code>/mode?mode=0</code>
<br>
<code>/mode?mode=1&r=0&g=0&b=0</code>
</body>
</html>
'''
2021-11-09 17:26:35 +00:00
@app.route('/api/mode', methods=['GET'])
def mode():
global mode
global r
global g
global b
args = request.args
mode = str(args.get('mode'))
r = str(args.get('r'))
g = str(args.get('g'))
b = str(args.get('b'))
if mode is None:
return 'No mode specified'
elif mode == "":
return 'No mode specified'
elif mode == "0":
mode = 0
return 'Rainbow Cycle'
elif mode == "1":
if r is None or g is None or b is None:
return 'No color specified'
elif r == "" or g == "" or b == "":
return 'No color specified'
else:
mode = 1
return 'Color: '+str(r)+','+str(g)+','+str(b)
2021-11-08 20:36:55 +00:00
@app.route('/api/power', methods=['GET'])
def power():
global power
power = request.args.get('power')
if power is None:
return 'No power argument specified'
elif power == "":
return 'No power argument specified'
elif power == "True":
power = "True"
return 'Power: ON'
elif power == "False":
power = "False"
return 'Power: OFF'
2022-03-26 19:39:25 +00:00
def mqtt_on_message(client, userdata, msg):
print("topic: " ,str(msg.topic))
print("payload: " ,str(msg.payload.decode("utf-8")))
topic = str(msg.topic)
payload = str(msg.payload.decode("utf-8"))
if topic == "server-rack-led/power":
global power
power = payload
elif topic == "server-rack-led/rgb":
splitted_payload = payload.split(",")
global r
r = str(splitted_payload[0])
global g
g = str(splitted_payload[1])
global b
b = str(splitted_payload[2])
elif topic == "server-rack-led/mode":
if payload == "0":
global mode
mode = 0
elif payload == "1":
mode = 1
strip = Adafruit_NeoPixel(LED_COUNT, LED_PIN, LED_FREQ_HZ, LED_DMA, LED_INVERT, LED_BRIGHTNESS, LED_CHANNEL)
strip.begin()
mode = 0
power = "True"
threading.Thread(target=lambda: app.run(host='0.0.0.0', port=80)).start()
mqttBroker ="homeassistant.ping-mee.local"
client = mqtt.Client("server_rack_led")
client.username_pw_set("mqtt", "pmMQTT_11!")
debug_print("Connecting to MQTT Broker "+str(mqttBroker))
client.connect(mqttBroker)
client.on_connect = mqtt_on_connect
client.on_message = mqtt_on_message
client.publish("server-rack-led/power","True")
client.publish("server-rack-led/mode","0")
2021-11-08 20:36:55 +00:00
try:
client.loop_start()
2021-11-08 20:36:55 +00:00
while True:
if power == "True":
if mode == 0:
rainbowCycle(strip)
elif mode == 1:
setColor(strip, Color(int(r), int(g), int(b)))
time.sleep(1)
2021-11-09 22:16:38 +00:00
else:
setColor(strip, Color(0,0,0))
time.sleep(1)
2021-11-08 20:36:55 +00:00
except KeyboardInterrupt:
client.loop_stop()
2021-11-09 17:29:50 +00:00
setColor(strip, Color(0,0,0))
GPIO.cleanup()