2017-03-14 14:57:02 +01:00
#!/usr/bin/env python3
2017-03-11 14:04:02 +01:00
#
# c4ctrl: Command line client for AutoC4
2017-04-06 16:33:32 +02:00
#
# Author: Shy
"""
This is a command line client for Autoc4 , the home automation system of the C4 .
Some parts of it * * may * * be useful as python module for simple tasks .
"""
2017-03-11 14:04:02 +01:00
import sys
2017-04-06 16:33:32 +02:00
2017-03-11 14:04:02 +01:00
class C4Interface ( ) :
2017-04-06 16:33:32 +02:00
""" Interaction with the C4 home automation system. """
2017-03-11 14:04:02 +01:00
port = 1883
broker = " autoc4.labor.koeln.ccc.de "
2017-04-06 16:33:32 +02:00
qos = 2
2017-03-11 14:04:02 +01:00
retain = True
debug = False
def __init__ ( self , topic = None ) :
# Set a default topic
if topic : self . topic = topic
2017-03-31 17:55:54 +02:00
def push ( self , cmd , topic = None , retain = None ) :
2017-04-06 16:33:32 +02:00
""" Send a message to the MQTT broker.
cmd may a byte encoded payload or a list of byte encoded
payloads . """
2017-03-11 14:04:02 +01:00
from paho . mqtt import publish
2017-03-14 15:50:06 +01:00
# Overwrite defaults
2017-03-11 14:04:02 +01:00
if topic : self . topic = topic
2017-03-31 17:55:54 +02:00
if retain != None : self . retain = retain
2017-03-11 14:04:02 +01:00
if type ( cmd ) == list :
# Add <qos> and <retain> to every message
2017-03-12 16:24:01 +01:00
for item in cmd . copy ( ) :
2017-03-11 14:04:02 +01:00
if type ( item ) == dict :
item [ " qos " ] = self . qos
item [ " retain " ] = self . retain
2017-03-12 15:49:12 +01:00
elif type ( item ) == tuple :
2017-03-12 16:24:01 +01:00
new_item = (
2017-03-12 15:49:12 +01:00
item [ 0 ] or self . topic , # topic
item [ 1 ] , # payload
self . qos , # qos
self . retain # retain
)
2017-03-12 16:24:01 +01:00
cmd . remove ( item )
cmd . append ( new_item )
2017-03-12 15:49:12 +01:00
2017-03-12 15:13:41 +01:00
if self . debug : return print ( " [DEBUG] inhibited messages: " , cmd )
2017-03-11 14:04:02 +01:00
publish . multiple ( cmd ,
hostname = self . broker ,
port = self . port )
else :
if self . debug :
return print ( " [DEBUG] inhibited message to ' {} ' : ' {} ' " . format (
self . topic , cmd ) )
publish . single ( self . topic ,
payload = cmd ,
qos = self . qos ,
retain = self . retain ,
hostname = self . broker ,
port = self . port )
2017-03-15 09:53:00 +01:00
def pull ( self , topic = [ ] ) :
2017-04-06 16:33:32 +02:00
""" Return the state of a topic.
topic may be a list of topics or a single topic given as string . """
2017-03-11 14:04:02 +01:00
from paho . mqtt import subscribe
2017-03-21 16:12:00 +01:00
topic = topic or self . topic
2017-03-13 12:35:14 +01:00
# <topic> must be a list
if type ( topic ) == str :
topic = [ topic ]
2017-03-11 14:04:02 +01:00
if self . debug :
print ( " [DEBUG] inhibited query for: " , topic )
return [ ]
return subscribe . simple ( topic ,
msg_count = len ( topic ) ,
qos = self . qos ,
hostname = self . broker ,
port = self . port )
def status ( self ) :
2017-04-06 16:33:32 +02:00
""" Returns current status (string " open " or " closed " ) of the club. """
2017-03-15 09:53:00 +01:00
st = self . pull ( " club/status " )
2017-03-12 15:13:41 +01:00
# Produce fake result to prevent errors if in debug mode
if C4Interface . debug :
print ( " [DEBUG] Warning: handing over fake data to allow further execution! " )
class st : pass
st . payload = b ' \x00 '
2017-03-11 14:04:02 +01:00
if st . payload == b ' \x01 ' :
2017-04-06 16:33:32 +02:00
return " open "
2017-03-11 14:04:02 +01:00
else :
2017-04-06 16:33:32 +02:00
return " closed "
2017-03-11 14:04:02 +01:00
def open_gate ( self ) :
""" Open the gate. """
2017-03-15 09:53:00 +01:00
self . push ( cmd = b ' \x01 ' , topic = " club/gate " , retain = False )
2017-03-11 14:04:02 +01:00
def shutdown ( self , force = False ) :
""" Invoke the shutdown routine. """
if force :
payload = b ' \x44 '
else :
payload = b ' \x00 '
2017-03-15 09:53:00 +01:00
self . push ( cmd = payload , topic = " club/shutdown " , retain = False )
2017-03-11 14:04:02 +01:00
class Dmx :
2017-04-06 16:33:32 +02:00
""" Abstraction of the 3 channel LED cans in the club. """
2017-03-11 14:04:02 +01:00
2017-03-13 12:35:14 +01:00
template = " 000000 "
2017-03-11 14:04:02 +01:00
def __init__ ( self , topic , color = None ) :
self . topic = topic
2017-03-13 12:35:14 +01:00
self . set_color ( color or self . template )
2017-03-17 23:27:54 +01:00
self . is_master = topic . rfind ( " /master " ) == len ( topic ) - 7 # 7 = len("/master")
2017-03-11 14:04:02 +01:00
2017-03-13 12:35:14 +01:00
def _pad_color ( self , color ) :
2017-04-06 16:33:32 +02:00
""" Merge hex color value into hex template.
2017-03-15 00:05:45 +01:00
2017-04-06 16:33:32 +02:00
Expand 4 bit hex code notation ( eg . #f0f) and pad with template. """
2017-03-27 16:01:55 +02:00
if len ( color ) > len ( self . template ) :
# Silently truncate
2017-03-13 12:35:14 +01:00
return color [ : len ( self . template ) ]
2017-03-11 14:04:02 +01:00
# Expand 3 char codes and codes of half the required length.
# Yet, lets presume that a 6-char code should never be expanded.
2017-03-13 12:35:14 +01:00
if len ( color ) != 6 and len ( color ) == 3 or len ( color ) == ( len ( self . template ) / 2 ) :
2017-03-11 14:04:02 +01:00
expanded = " "
for c in color :
expanded + = c * 2
color = expanded
2017-03-15 00:05:45 +01:00
if len ( color ) == len ( self . template ) : # Nothing more to do
2017-03-11 14:04:02 +01:00
return color
# Add padding
2017-03-15 00:05:45 +01:00
color = color + self . template [ len ( color ) : ]
2017-03-11 14:04:02 +01:00
return color
def set_color ( self , color ) :
2017-04-06 16:33:32 +02:00
""" Set color (hex) for this instance.
2017-03-12 16:24:01 +01:00
2017-04-06 16:33:32 +02:00
The color is then available via its color variable . """
2017-03-13 12:35:14 +01:00
color = self . _pad_color ( color )
2017-03-11 14:04:02 +01:00
self . color = color
self . payload = bytearray . fromhex ( color )
2017-03-13 20:42:53 +01:00
class Dmx4 ( Dmx ) :
2017-04-06 16:33:32 +02:00
""" Abstraction of the 4 channel LED cans in the club. """
2017-03-11 14:04:02 +01:00
2017-03-13 20:42:53 +01:00
template = " 000000ff "
2017-03-15 00:05:45 +01:00
2017-03-11 14:04:02 +01:00
class Dmx7 ( Dmx ) :
2017-04-06 16:33:32 +02:00
""" Abstraction of the 7 channel LED cans in the club. """
2017-03-11 14:04:02 +01:00
2017-03-13 12:35:14 +01:00
template = " 000000000000ff "
2017-03-15 00:05:45 +01:00
2017-03-11 14:04:02 +01:00
class C4Room :
2017-04-06 16:33:32 +02:00
""" Methods of rooms in the club. """
2017-03-11 14:04:02 +01:00
2017-03-15 00:05:45 +01:00
def __init__ ( self ) :
self . c4 = C4Interface ( )
2017-03-11 14:04:02 +01:00
2017-03-15 00:05:45 +01:00
def _interactive_light_switch ( self ) :
# Interactively ask for input
2017-03-25 23:45:42 +01:00
if sys . stdin . isatty ( ) :
print ( " [ {} ] " . format ( self . name ) )
print ( " Please enter 0 or 1 for every light: " )
for level in range ( len ( self . switches ) ) :
print ( ( level * ' | ' ) + " ,- " + self . switches [ level ] [ 0 ] )
# Current state of switches
state = " "
req = [ ]
for t in self . switches :
req . append ( t [ 1 ] )
responce = self . c4 . pull ( req )
for sw in self . switches :
for r in responce :
if r . topic == sw [ 1 ] :
state = state + str ( int . from_bytes ( r . payload ,
byteorder = " little " ) )
print ( state ) # Present current state
2017-03-15 00:05:45 +01:00
try :
userinput = sys . stdin . readline ( ) . rstrip ( ' \n ' )
except KeyboardInterrupt :
print ( " \r Interrupted by user. " )
return " "
return userinput
def light_switch ( self , userinput = " " ) :
2017-04-06 16:33:32 +02:00
""" Switch lamps in a room on or off. """
2017-03-15 00:05:45 +01:00
if not userinput :
userinput = self . _interactive_light_switch ( )
2017-03-25 23:45:42 +01:00
if userinput == " " : return
if not userinput . isdecimal ( ) :
print ( " You ' re not paying attention! " , file = sys . stderr )
return
2017-03-11 14:04:02 +01:00
if len ( userinput ) != len ( self . switches ) :
2017-03-25 23:45:42 +01:00
if len ( bin ( int ( userinput ) ) ) < = len ( self . switches ) + 2 :
2017-03-21 18:24:46 +01:00
# +2 because bin() returns something like 'b0...'
2017-03-21 18:14:00 +01:00
# Try to interpret as integer
binary = bin ( int ( userinput ) ) [ 2 : ]
userinput = str ( len ( self . switches ) * ' 0 ' ) [ : - len ( binary ) ] + binary
else :
print ( " Error: wrong number of digits (expected {} , got {} )! " . format (
len ( self . switches ) , len ( userinput ) ) )
return False
2017-03-11 14:04:02 +01:00
cmd = [ ]
for si in range ( len ( self . switches ) ) :
if userinput [ si ] not in " 01 " :
print ( " Error: invalid digit: " + userinput [ si ] )
return False
cmd . append ( {
" topic " : self . switches [ si ] [ 1 ] ,
" payload " : bytearray ( [ int ( userinput [ si ] ) ] )
} )
2017-03-15 00:05:45 +01:00
2017-03-15 09:53:00 +01:00
return self . c4 . push ( cmd )
2017-03-11 14:04:02 +01:00
2017-04-06 16:33:32 +02:00
def set_colorscheme ( self , colorscheme , no_magic ) :
""" Apply colorscheme to the LED Cans in this room. """
2017-03-11 14:04:02 +01:00
cmd = [ ]
for light in self . lights :
if colorscheme . color_for ( light . topic ) :
2017-03-24 12:04:00 +01:00
2017-04-06 16:33:32 +02:00
# Update internal state of this Dmx object, so we can query
# <object>.payload later
light . set_color ( colorscheme . color_for ( light . topic ) )
if no_magic :
# Send data to the real lanterns, not fluffyd.
2017-03-17 23:27:54 +01:00
cmd . append ( {
2017-04-06 16:33:32 +02:00
" topic " : light . topic ,
" payload " : light . payload
2017-03-17 23:27:54 +01:00
} )
else :
2017-04-06 16:33:32 +02:00
# Send color to ghost instead of the "real" light
# Generate the ghost topic for topic
ghost = " ghosts " + light . topic [ light . topic . find ( ' / ' ) : ]
2017-03-17 23:27:54 +01:00
cmd . append ( {
2017-04-06 16:33:32 +02:00
" topic " : ghost ,
2017-03-17 23:27:54 +01:00
" payload " : light . payload
} )
2017-03-31 17:55:54 +02:00
if cmd == [ ] : return
2017-04-06 16:33:32 +02:00
if no_magic :
2017-03-31 17:55:54 +02:00
return self . c4 . push ( cmd )
2017-04-06 16:33:32 +02:00
else : # Do not retain "magic" messages
return self . c4 . push ( cmd , retain = False )
2017-03-11 14:04:02 +01:00
class Wohnzimmer ( C4Room ) :
2017-04-06 16:33:32 +02:00
""" Description of the Wohnzimmer. """
2017-03-11 14:04:02 +01:00
name = " Wohnzimmer "
2017-03-21 14:23:04 +01:00
switches = (
2017-03-11 14:04:02 +01:00
( " Tür " , " licht/wohnzimmer/tuer " ) ,
( " Mitte " , " licht/wohnzimmer/mitte " ) ,
( " Flur " , " licht/wohnzimmer/gang " ) ,
( " Küche " , " licht/wohnzimmer/kueche " )
2017-03-21 14:23:04 +01:00
)
2017-03-11 14:04:02 +01:00
master = Dmx7 ( " dmx/wohnzimmer/master " )
2017-03-21 14:23:04 +01:00
lights = (
2017-03-11 14:04:02 +01:00
Dmx7 ( " dmx/wohnzimmer/master " ) ,
Dmx7 ( " dmx/wohnzimmer/tuer1 " ) ,
Dmx7 ( " dmx/wohnzimmer/tuer2 " ) ,
Dmx7 ( " dmx/wohnzimmer/tuer3 " ) ,
Dmx7 ( " dmx/wohnzimmer/mitte1 " ) ,
Dmx7 ( " dmx/wohnzimmer/mitte2 " ) ,
Dmx7 ( " dmx/wohnzimmer/mitte3 " ) ,
Dmx7 ( " dmx/wohnzimmer/gang " ) ,
Dmx7 ( " dmx/wohnzimmer/baellebad " ) ,
2017-03-16 19:32:52 +01:00
Dmx ( " led/kitchen/sink " )
2017-03-21 14:23:04 +01:00
)
2017-03-11 14:04:02 +01:00
class Plenarsaal ( C4Room ) :
2017-04-06 16:33:32 +02:00
""" Description of the Plenarsaal. """
2017-03-11 14:04:02 +01:00
name = " Plenarsaal "
2017-03-21 14:23:04 +01:00
switches = (
2017-03-11 14:04:02 +01:00
( " Vorne/Wand " , " licht/plenar/vornewand " ) ,
( " Vorne/Fenster " , " licht/plenar/vornefenster " ) ,
( " Hinten/Wand " , " licht/plenar/hintenwand " ) ,
( " Hinten/Fenster " , " licht/plenar/hintenfenster " )
2017-03-21 14:23:04 +01:00
)
2017-03-11 14:04:02 +01:00
master = Dmx7 ( " dmx/plenar/master " )
2017-03-21 14:23:04 +01:00
lights = (
2017-03-11 14:04:02 +01:00
Dmx7 ( " dmx/plenar/master " ) ,
Dmx7 ( " dmx/plenar/vorne1 " ) ,
Dmx7 ( " dmx/plenar/vorne2 " ) ,
Dmx7 ( " dmx/plenar/vorne3 " ) ,
Dmx7 ( " dmx/plenar/hinten1 " ) ,
Dmx7 ( " dmx/plenar/hinten2 " ) ,
Dmx7 ( " dmx/plenar/hinten3 " ) ,
Dmx7 ( " dmx/plenar/hinten4 " )
2017-03-21 14:23:04 +01:00
)
2017-03-11 14:04:02 +01:00
class Fnordcenter ( C4Room ) :
2017-04-06 16:33:32 +02:00
""" Description of the Fnordcenter. """
2017-03-11 14:04:02 +01:00
name = " Fnordcenter "
2017-03-21 14:23:04 +01:00
switches = (
2017-03-11 14:04:02 +01:00
( " Links (Fairydust) " , " licht/fnord/links " ) ,
( " Rechts (SCUMM) " , " licht/fnord/rechts " )
2017-03-21 14:23:04 +01:00
)
2017-03-13 20:42:53 +01:00
master = Dmx4 ( " dmx/fnord/master " )
2017-03-21 14:23:04 +01:00
lights = (
2017-03-13 20:42:53 +01:00
Dmx4 ( " dmx/fnord/master " ) ,
Dmx4 ( " dmx/fnord/scummfenster " ) ,
Dmx4 ( " dmx/fnord/schranklinks " ) ,
Dmx4 ( " dmx/fnord/fairyfenster " ) ,
Dmx4 ( " dmx/fnord/schrankrechts " )
2017-03-21 14:23:04 +01:00
)
2017-03-11 14:04:02 +01:00
class Keller ( C4Room ) :
2017-04-06 16:33:32 +02:00
""" Description of the Keller. """
2017-03-11 14:04:02 +01:00
name = " Keller "
2017-03-21 14:23:04 +01:00
switches = (
2017-03-11 14:04:02 +01:00
( " Außen " , " licht/keller/aussen " ) ,
( " Innen " , " licht/keller/innen " ) ,
( " Vorne " , " licht/keller/vorne " )
2017-03-21 14:23:04 +01:00
)
2017-03-31 17:14:03 +02:00
master = None
2017-03-24 12:04:00 +01:00
lights = ( )
2017-03-11 14:04:02 +01:00
class Kitchenlight :
2017-04-06 16:33:32 +02:00
""" The Kitchenlight. """
2017-03-11 14:04:02 +01:00
2017-03-15 00:05:45 +01:00
_END = " little " # Endianess
2017-03-11 14:04:02 +01:00
_available_modes = """
2017-03-12 15:13:41 +01:00
off turn off
2017-03-11 14:04:02 +01:00
checker [ , DELAY [ , COLOR_1 [ , COLOR_2 ] ] ]
Checker
matrix [ , LINES ] Matrix
mood [ , MODE ] ( 1 = Colorwheel , 2 = Random )
Moodlight
oc [ , DELAY ] Open Chaos
pacman Pacman
sine Sine
text [ , TEXT [ , DELAY ] ] Text
flood Flood
clock Clock """
def __init__ ( self , topic = " kitchenlight/change_screen " ,
powertopic = " power/wohnzimmer/kitchenlight " ,
autopower = True ) :
self . topic = topic # Kitchenlight topic
self . powertopic = powertopic # Topic for power on
self . autopower = autopower # Power on on every mode change?
def _switch ( self , data , poweron = False , poweroff = False ) :
2017-04-06 16:33:32 +02:00
""" Send commands via a C4Interface to the MQTT broker. """
2017-03-11 14:04:02 +01:00
if self . autopower or poweron or poweroff :
c4 = C4Interface ( self . topic )
cmd = [ ]
cmd . append ( {
" topic " : self . topic ,
" payload " : data } )
if poweroff :
cmd . append ( {
" topic " : self . powertopic ,
2017-04-06 16:33:32 +02:00
" payload " : b ' \x00 ' } )
2017-03-11 14:04:02 +01:00
elif self . autopower or poweron :
cmd . append ( {
" topic " : self . powertopic ,
2017-04-06 16:33:32 +02:00
" payload " : b ' \x01 ' } )
2017-03-15 09:53:00 +01:00
c4 . push ( cmd )
2017-03-11 14:04:02 +01:00
else :
c4 = C4Interface ( self . topic )
2017-03-15 09:53:00 +01:00
c4 . push ( data )
2017-03-11 14:04:02 +01:00
def set_mode ( self , mode , opts = [ ] ) :
""" Switch to given mode. """
mode = mode . lower ( )
if mode == " off " :
return self . empty ( )
if mode == " checker " :
return self . checker ( * opts )
if mode == " matrix " :
return self . matrix ( * opts )
if mode == " mood " :
return self . moodlight ( * opts )
if mode == " oc " or mode == " openchaos " :
return self . openchaos ( * opts )
if mode == " pacman " :
return self . pacman ( )
if mode == " sine " :
return self . sine ( )
if mode == " text " :
return self . text ( * opts )
if mode == " flood " :
return self . flood ( )
if mode == " clock " :
return self . clock ( )
2017-03-15 00:05:45 +01:00
print ( " Error: unknown Kitchenlight mode {} ! " . format ( mode ) )
2017-03-11 14:04:02 +01:00
return False
def empty ( self ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " empty " and turn off Kitchenlight. """
2017-03-11 14:04:02 +01:00
# Screen 0
2017-03-15 00:05:45 +01:00
d = int ( 0 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
self . _switch ( d , poweroff = True )
def checker ( self , delay = 500 , colA = " 0000ff " , colB = " 00ff00 " ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " checker " . """
2017-03-11 14:04:02 +01:00
# Kind of a hack: lets treat the two colors as DMX lights
ca = Dmx ( " checker/a " , colA . lstrip ( ' # ' ) )
cb = Dmx ( " checker/b " , colB . lstrip ( ' # ' ) )
d = bytearray ( 20 )
v = memoryview ( d )
# Screen 1
2017-03-15 00:05:45 +01:00
v [ 0 : 4 ] = int ( 1 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
# Delay
2017-03-15 00:05:45 +01:00
v [ 4 : 8 ] = int ( delay ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
# ColorA R/G/B
2017-03-15 00:05:45 +01:00
v [ 8 : 10 ] = int ( ca . color [ 0 : 2 ] , base = 16 ) . to_bytes ( 2 , self . _END )
v [ 10 : 12 ] = int ( ca . color [ 2 : 4 ] , base = 16 ) . to_bytes ( 2 , self . _END )
v [ 12 : 14 ] = int ( ca . color [ 4 : 6 ] , base = 16 ) . to_bytes ( 2 , self . _END )
2017-03-11 14:04:02 +01:00
# ColorB R/G/B
2017-03-15 00:05:45 +01:00
v [ 14 : 16 ] = int ( cb . color [ 0 : 2 ] , base = 16 ) . to_bytes ( 2 , self . _END )
v [ 16 : 18 ] = int ( cb . color [ 2 : 4 ] , base = 16 ) . to_bytes ( 2 , self . _END )
v [ 18 : 20 ] = int ( cb . color [ 4 : 6 ] , base = 16 ) . to_bytes ( 2 , self . _END )
2017-03-11 14:04:02 +01:00
self . _switch ( d )
def matrix ( self , lines = 8 ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " matrix " . """
2017-03-13 18:45:17 +01:00
if int ( lines ) > 31 : lines = 31 # Maximal line count
2017-03-11 14:04:02 +01:00
d = bytearray ( 8 )
v = memoryview ( d )
# Screen 2
2017-03-15 00:05:45 +01:00
v [ 0 : 4 ] = int ( 2 ) . to_bytes ( 4 , self . _END )
v [ 4 : 8 ] = int ( lines ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
self . _switch ( d )
def moodlight ( self , mode = 1 ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " moodlight " . """
2017-03-11 14:04:02 +01:00
if mode == 1 : # Mode "Colorwheel"
d = bytearray ( 19 )
v = memoryview ( d )
# Screen 3
2017-03-15 00:05:45 +01:00
v [ 0 : 4 ] = int ( 3 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
# Mode
2017-03-15 00:05:45 +01:00
v [ 4 : 5 ] = int ( mode ) . to_bytes ( 1 , self . _END )
2017-03-11 14:04:02 +01:00
# Step
2017-03-15 00:05:45 +01:00
v [ 5 : 9 ] = int ( 1 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
# Fade delay
2017-03-15 00:05:45 +01:00
v [ 9 : 13 ] = int ( 10 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
# Pause
2017-03-15 00:05:45 +01:00
v [ 13 : 17 ] = int ( 10000 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
# Hue step
2017-03-15 00:05:45 +01:00
v [ 17 : 19 ] = int ( 30 ) . to_bytes ( 2 , self . _END )
2017-03-11 14:04:02 +01:00
else : # Mode "Random"
d = bytearray ( 17 )
v = memoryview ( d )
# Screen 3
2017-03-15 00:05:45 +01:00
v [ 0 : 4 ] = int ( 3 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
# Mode
2017-03-15 00:05:45 +01:00
v [ 4 : 5 ] = int ( mode ) . to_bytes ( 1 , self . _END )
2017-03-11 14:04:02 +01:00
# Step
2017-03-15 00:05:45 +01:00
v [ 5 : 9 ] = int ( 1 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
# Fade delay
2017-03-15 00:05:45 +01:00
v [ 9 : 13 ] = int ( 10 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
# Pause
2017-03-15 00:05:45 +01:00
v [ 13 : 17 ] = int ( 10000 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
self . _switch ( d )
2017-03-13 12:35:14 +01:00
def openchaos ( self , delay = 1000 ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " openchaos " . """
2017-03-11 14:04:02 +01:00
d = bytearray ( 8 )
v = memoryview ( d )
# Screen 4
2017-03-15 00:05:45 +01:00
v [ 0 : 4 ] = int ( 4 ) . to_bytes ( 4 , self . _END )
v [ 4 : 8 ] = int ( delay ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
self . _switch ( d )
def pacman ( self ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " pacman " . """
2017-03-11 14:04:02 +01:00
# Screen 5
2017-03-15 00:05:45 +01:00
d = int ( 5 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
self . _switch ( d )
def sine ( self ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " sine " . """
2017-03-11 14:04:02 +01:00
# Screen 6
2017-03-15 00:05:45 +01:00
d = int ( 6 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
self . _switch ( d )
# Screen 7 is Strobo, which is disabled because it seems to do harm to
# the Kitchenlight. Evil strobo!
2017-03-13 18:45:17 +01:00
def text ( self , text = " Hello World " , delay = 250 ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " text " . """
2017-03-13 12:43:59 +01:00
text = text . encode ( " ascii " , " ignore " )
2017-03-14 17:15:16 +01:00
if len ( text ) > 256 : # Maximum text length
2017-03-25 20:16:58 +01:00
print ( " Warning: text length must not exceed 256 characters! " , file = sys . stderr )
2017-03-14 17:15:16 +01:00
text = text [ : 256 ]
2017-03-11 14:04:02 +01:00
d = bytearray ( 8 + len ( text ) + 1 )
v = memoryview ( d )
# Screen 8
2017-03-15 00:05:45 +01:00
v [ 0 : 4 ] = int ( 8 ) . to_bytes ( 4 , self . _END )
v [ 4 : 8 ] = int ( delay ) . to_bytes ( 4 , self . _END )
2017-03-13 12:35:14 +01:00
v [ 8 : 8 + len ( text ) ] = text
v [ len ( d ) - 1 : len ( d ) ] = bytes ( 1 )
2017-03-11 14:04:02 +01:00
self . _switch ( d )
def flood ( self ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " flood " . """
2017-03-11 14:04:02 +01:00
# Screen 9
2017-03-15 00:05:45 +01:00
d = int ( 9 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
self . _switch ( d )
def clock ( self ) :
2017-04-06 16:33:32 +02:00
""" Set to mode " clock " . """
2017-03-11 14:04:02 +01:00
# Screen 11
2017-03-15 00:05:45 +01:00
d = int ( 11 ) . to_bytes ( 4 , self . _END )
2017-03-11 14:04:02 +01:00
self . _switch ( d )
class ColorScheme :
2017-04-06 16:33:32 +02:00
""" Abstraction of a colorscheme. """
2017-03-11 14:04:02 +01:00
2017-04-06 16:33:32 +02:00
# Names of virtual presets. These are always listed as available and the
# user may not save presets under this name.
2017-03-11 14:04:02 +01:00
_virtual_presets = [ " off " , " random " ]
def __init__ ( self , autoinit = " " ) :
self . mapping = { }
self . single_color = False
self . return_random_color = False
self . available = None # List of available presets
if autoinit :
# Load or generate preset
if autoinit [ 0 ] == ' # ' :
2017-03-15 00:05:45 +01:00
return self . from_color ( autoinit )
2017-03-11 14:04:02 +01:00
elif self . _expand_preset ( autoinit ) == " off " :
2017-03-15 00:05:45 +01:00
# Virtual preset: set masters to #000000
2017-03-11 14:04:02 +01:00
return self . from_color ( " 000000 " )
elif self . _expand_preset ( autoinit ) == " random " :
2017-03-15 00:05:45 +01:00
# Virtual preset: return random color on every query
2017-03-11 14:04:02 +01:00
return self . from_random ( )
else :
2017-03-15 00:05:45 +01:00
# Load preset file
2017-03-11 14:04:02 +01:00
return self . from_file ( autoinit )
def __bool__ ( self ) :
# Return true if color_for has a chance to present anything useful
if self . mapping : return True
if self . single_color : return True
if self . return_random_color : return True
else : return False
def _get_cfg_dir ( self , quiet = False , create = False ) :
2017-04-06 16:33:32 +02:00
""" Returns path of the config dir. """
2017-03-11 14:04:02 +01:00
import os
# The name of our config directory
XDG_NAME = " c4ctrl "
# Get XDG_CONFIG_DIR from environment or set default
if " XDG_CONFIG_DIR " in os . environ :
XDG_CONFIG_DIR = os . environ [ " XDG_CONFIG_DIR " ]
else :
XDG_CONFIG_DIR = os . path . expanduser ( " ~/.config " )
# Does our config dir exist?
cfg_dir = os . path . join ( XDG_CONFIG_DIR , XDG_NAME )
if not os . path . isdir ( cfg_dir ) :
if create :
print ( " Creating config directory \" {} \" " . format ( cfg_dir ) )
os . mkdir ( cfg_dir )
elif quiet :
return None
else :
2017-03-25 20:16:58 +01:00
print ( " Warning: config dir \" {} \" does not exist! " . format (
cfg_dir ) , file = sys . stderr )
2017-03-11 14:04:02 +01:00
return None
return cfg_dir
def _expand_preset ( self , preset ) :
2017-04-06 16:33:32 +02:00
""" Tries to expand given string to a valid preset name. """
2017-03-11 14:04:02 +01:00
import os
if not self . available :
cfg_dir = self . _get_cfg_dir ( quiet = True )
if not cfg_dir :
self . available = self . _virtual_presets . copy ( )
else :
self . available = os . listdir ( cfg_dir )
self . available . extend ( self . _virtual_presets )
# Search for an exact match first
for a in self . available :
if a == preset : return a
# Return anything which begins with the name given
for a in self . available :
if a . find ( preset ) == 0 : return a
# Fallback
return preset
def _topic_is_master ( self , topic ) :
2017-04-06 16:33:32 +02:00
""" Does the given topic look like a master topic? """
2017-03-11 14:04:02 +01:00
return topic . lower ( ) . rfind ( " /master " ) == len ( topic ) - 7 # 7 = len("/master")
def _random_color ( self ) :
2017-04-06 16:33:32 +02:00
""" Returns a 3*4 bit pseudo random color in 6 char hex notation. """
2017-03-18 11:31:54 +01:00
from random import randint , sample
chls = [ 15 ]
chls . append ( randint ( 0 , 15 ) )
2017-03-18 16:13:48 +01:00
chls . append ( randint ( 0 , 15 ) - chls [ 1 ] )
2017-03-18 11:31:54 +01:00
if chls [ 2 ] < 0 : chls [ 2 ] = 0
2017-03-11 14:04:02 +01:00
color = " "
2017-03-18 11:31:54 +01:00
for ch in sample ( chls , k = 3 ) :
color + = hex ( ch ) [ 2 : ] * 2
2017-03-11 14:04:02 +01:00
return color
2017-03-25 16:24:01 +01:00
def _single_color ( self ) :
# Check if there is a range in the color string. If yes, replace it
# by a random color.
if self . single_color . find ( ' - ' , 1 , - 1 ) == - 1 :
return self . single_color
from random import randint
color = " "
for i in range ( len ( self . single_color ) ) :
if self . single_color [ i ] != ' - ' :
try :
if self . single_color [ i - 1 ] == ' - ' :
continue
elif self . single_color [ i + 1 ] == ' - ' :
continue
except IndexError : pass
color + = self . single_color [ i ]
else :
f , t = self . single_color [ i - 1 ] , self . single_color [ i + 1 ]
color + = hex ( randint ( int ( f , base = 16 ) , int ( t , base = 16 ) ) ) [ 2 : ]
return color
2017-03-11 14:04:02 +01:00
def color_for ( self , topic ) :
2017-04-06 16:33:32 +02:00
""" Returns the color (in hexadecimal notation) this ColorScheme assumes
for the given topic . """
2017-03-28 17:47:03 +02:00
# We need to take care not to return colors for both "normal" topics
2017-04-06 16:33:32 +02:00
# and masters, as setting masters would override other settings.
# If this ColorScheme has been read from a file though, we asssume that
# the user has taken care of this and apply what we are told to apply.
2017-03-11 14:04:02 +01:00
if self . mapping :
if topic in self . mapping . keys ( ) :
return self . mapping [ topic ]
elif self . single_color :
2017-03-31 17:55:54 +02:00
if not self . _topic_is_master ( topic ) :
2017-03-28 17:47:03 +02:00
return self . _single_color ( )
2017-03-11 14:04:02 +01:00
elif self . return_random_color :
2017-03-31 17:55:54 +02:00
if not self . _topic_is_master ( topic ) :
2017-03-11 14:04:02 +01:00
return self . _random_color ( )
2017-03-15 00:05:45 +01:00
# Fallback
return None
2017-03-11 14:04:02 +01:00
def from_file ( self , preset ) :
2017-04-06 16:33:32 +02:00
""" Load ColorScheme from file. """
2017-03-25 14:54:54 +01:00
if preset == ' - ' :
fd = sys . stdin
else :
import os
cfg_dir = self . _get_cfg_dir ( )
if not cfg_dir :
print ( " Error: could not load preset! " )
2017-03-25 23:45:42 +01:00
return
2017-03-11 14:04:02 +01:00
2017-03-25 14:54:54 +01:00
# Expand preset name
preset = self . _expand_preset ( preset )
# Try to open the preset file
fn = os . path . join ( cfg_dir , preset )
try :
fd = open ( fn )
except OSError :
print ( " Error: could not load preset \" {} \" (file could not be accessed)! " . format ( preset ) )
2017-03-25 23:45:42 +01:00
return
2017-03-11 14:04:02 +01:00
# Parse the preset file
self . mapping = { }
self . name = preset
for line in fd . readlines ( ) :
# Skip every line which does not begin with an alphabetic character
try :
if not line . lstrip ( ) [ 0 ] . isalpha ( ) : continue
except IndexError : continue # Empty line
# Strip spaces and split
k , v = line . replace ( ' ' , ' ' ) . replace ( ' \t ' , ' ' ) . split ( ' = ' )
# Convert #fff to fff and remove trailing comments, nl and cr chars
vl = v . rstrip ( " \n \r " ) . split ( ' # ' )
v = vl [ 0 ] or vl [ 1 ]
# Validate hex code
for c in v . lower ( ) :
if c not in " 0123456789abcdef " :
2017-03-28 17:47:03 +02:00
print ( " Error: invalid color code \" {} \" in preset \" {} \" ! " . format ( v , preset ) , file = sys . stderr )
2017-03-11 14:04:02 +01:00
sys . exit ( 1 )
self . mapping [ k ] = v
fd . close ( )
def from_color ( self , color ) :
2017-04-06 16:33:32 +02:00
""" Derive ColorScheme from a single hex color. """
2017-03-25 16:24:01 +01:00
self . single_color = color . lstrip ( ' # ' ) . strip ( ' - ' )
2017-03-11 14:04:02 +01:00
def from_random ( self ) :
2017-04-06 16:33:32 +02:00
""" Derive ColorScheme from random colors. """
2017-03-11 14:04:02 +01:00
self . return_random_color = True
def list_available ( self ) :
2017-04-06 16:33:32 +02:00
""" List available presets. """
2017-03-11 14:04:02 +01:00
import os
cfg_dir = self . _get_cfg_dir ( )
if not cfg_dir :
self . available = self . _virtual_presets . copy ( )
if not self . available :
self . available = os . listdir ( cfg_dir )
self . available . extend ( self . _virtual_presets )
self . available . sort ( )
2017-03-16 19:32:52 +01:00
print ( " Available presets: \n " )
2017-03-11 14:04:02 +01:00
for entry in self . available :
if entry [ 0 ] == ' . ' or entry [ - 1 : ] == ' ~ ' : continue
2017-03-12 15:13:41 +01:00
print ( " " + entry )
2017-03-11 14:04:02 +01:00
def store ( self , name ) :
2017-04-06 16:33:32 +02:00
""" Store the current state of all lights as preset. """
2017-03-11 14:04:02 +01:00
# First of all, refuse to override virtual presets
if name in self . _virtual_presets :
print ( " I ' m sorry Dave. I ' m afraid I can ' t do that. The name \" {} \" is reserved. Please choose a different one. " . format ( name ) )
return False
2017-03-25 14:54:54 +01:00
if name == ' - ' :
fd = sys . stdout
else :
import os
cfg_dir = self . _get_cfg_dir ( create = True ) # Create config dir if missing
2017-03-11 14:04:02 +01:00
2017-03-25 14:54:54 +01:00
fn = os . path . join ( cfg_dir , name )
try :
fd = open ( fn , ' xt ' )
except FileExistsError :
print ( " A preset with this name already exists, overwrite? [y/N] " ,
end = ' ' , flush = True )
if sys . stdin . read ( 1 ) . lower ( ) == ' y ' :
fd = open ( fn , ' wt ' )
else :
return False
2017-03-11 14:04:02 +01:00
# Get current states
c4 = C4Interface ( )
2017-03-25 23:45:42 +01:00
if name == ' - ' :
2017-03-28 17:47:03 +02:00
fd . write ( " # c4ctrl preset (auto generated) \n " . format ( name ) )
2017-03-25 23:45:42 +01:00
else :
2017-03-28 17:47:03 +02:00
fd . write ( " # c4ctrl preset \" {} \" (auto generated) \n " . format ( name ) )
fd . write ( " # \n " )
fd . write ( " # Note: Topics ending with \" /master \" override all other topics in a room. \n " )
fd . write ( " # All spaces will be stripped and lines beginning with \' # \' ignored. \n " )
2017-03-11 14:04:02 +01:00
for room in Wohnzimmer , Plenarsaal , Fnordcenter :
topics = [ ]
2017-03-28 17:47:03 +02:00
max_topic_len = 0
2017-03-11 14:04:02 +01:00
for light in room . lights :
topics . append ( light . topic )
2017-03-28 17:47:03 +02:00
if len ( light . topic ) > max_topic_len :
max_topic_len = len ( light . topic )
2017-03-11 14:04:02 +01:00
2017-03-15 09:53:00 +01:00
responce = c4 . pull ( topics )
2017-03-11 14:04:02 +01:00
fd . write ( " \n # {} \n " . format ( room . name ) )
for light in room . lights :
for r in responce :
if r . topic == light . topic :
2017-03-12 16:24:01 +01:00
light . set_color ( r . payload . hex ( ) )
2017-03-28 17:47:03 +02:00
# Format payload more nicely
color = light . color
if len ( color ) > 6 :
color = color [ : 6 ] + ' ' + color [ 6 : ]
topic = light . topic . ljust ( max_topic_len )
2017-03-17 23:27:54 +01:00
# Out comment master, as it would override everything else
2017-03-11 14:04:02 +01:00
if self . _topic_is_master ( r . topic ) :
2017-03-28 17:47:03 +02:00
fd . write ( " # {} = {} \n " . format ( topic , color ) )
2017-03-11 14:04:02 +01:00
else :
2017-03-28 17:47:03 +02:00
fd . write ( " {} = {} \n " . format ( topic , color ) )
2017-03-11 14:04:02 +01:00
2017-03-25 14:54:54 +01:00
if name != ' - ' :
fd . close ( )
print ( " Wrote preset \" {} \" " . format ( name ) )
2017-03-11 14:04:02 +01:00
2017-03-12 15:13:41 +01:00
class RemotePresets :
2017-04-06 16:33:32 +02:00
""" Remote preset control. """
2017-03-12 15:13:41 +01:00
def __init__ ( self ) :
self . map = {
" global " : {
" name " : " AutoC4 " ,
" list_topic " : " preset/list " ,
" set_topic " : " preset/set " ,
" def_topic " : " preset/def "
} ,
" wohnzimmer " : {
" name " : " Wohnzimmer " ,
" list_topic " : " preset/wohnzimmer/list " ,
" set_topic " : " preset/wohnzimmer/set " ,
" def_topic " : " preset/wohnzimmer/def "
} ,
" plenar " : {
" name " : " Plenarsaal " ,
" list_topic " : " preset/plenar/list " ,
" set_topic " : " preset/plenar/set " ,
" def_topic " : " preset/plenar/def "
} ,
" fnord " : {
" name " : " Fnordcenter " ,
" list_topic " : " preset/fnord/list " ,
" set_topic " : " preset/fnord/set " ,
" def_topic " : " preset/fnord/def "
} ,
" keller " : {
" name " : " Keller " ,
" list_topic " : " preset/keller/list " ,
" set_topic " : " preset/keller/set " ,
" def_topic " : " preset/keller/def "
}
}
def _expand_room_name ( self , name ) :
2017-04-06 16:33:32 +02:00
""" Returns a valid room name expanded from the given name. """
2017-03-12 15:13:41 +01:00
if name in self . map . keys ( ) :
2017-03-12 16:24:01 +01:00
# Return on exact match
2017-03-12 15:13:41 +01:00
return name
for room in self . map . keys ( ) :
if room . find ( name ) == 0 :
return room
# Fallback
return name
def _expand_preset_name ( self , name , rooms , available ) :
2017-04-06 16:33:32 +02:00
""" Returns a valid preset name expanded from the given name.
2017-03-12 15:13:41 +01:00
2017-04-06 16:33:32 +02:00
Takes care to match only presets which are available for all rooms
specified .
rooms is a list of rooms for which the preset should be a valid
option .
available is a dict containing valid presets for rooms as returned
by query_available ( ) . """
2017-03-13 14:09:38 +01:00
# Strip every "global" out of the room list. We take special care of
2017-03-13 14:34:54 +01:00
# "global" later on.
2017-03-13 14:09:38 +01:00
while " global " in rooms :
rooms . remove ( " global " )
2017-03-12 15:13:41 +01:00
matchtable = { }
2017-03-13 14:09:38 +01:00
if " global " not in rooms :
for preset in available [ " global " ] :
# Candidate?
if preset == name or preset . find ( name ) == 0 :
# Presets in "global" are available everywhere
matchtable [ preset ] = len ( rooms )
2017-03-12 15:13:41 +01:00
for room in rooms :
for preset in available [ room ] :
# Candidate?
if preset == name or preset . find ( name ) == 0 :
if preset in matchtable . keys ( ) :
matchtable [ preset ] + = 1
else :
matchtable [ preset ] = 1
2017-03-13 14:09:38 +01:00
# First check if there is an exact match in all rooms
if name in matchtable . keys ( ) and matchtable [ name ] > = len ( rooms ) :
2017-03-12 16:24:01 +01:00
return name
2017-03-12 15:13:41 +01:00
# Return first preset available in all rooms
for match in matchtable . keys ( ) :
2017-03-13 14:09:38 +01:00
if matchtable [ match ] > = len ( rooms ) :
return match
elif match in available [ " global " ] :
2017-03-12 15:13:41 +01:00
return match
# Fallback
return name
def query_available ( self , rooms = [ " global " ] ) :
2017-04-06 16:33:32 +02:00
""" Returns a dict of remotely available presets for [rooms]. """
2017-03-12 15:13:41 +01:00
import json
2017-03-12 16:24:01 +01:00
# Presets in "global" are available everywhere and should always be included
2017-03-12 15:13:41 +01:00
if " global " not in rooms :
2017-03-13 14:52:29 +01:00
rooms . insert ( 0 , " global " )
2017-03-12 15:13:41 +01:00
req = [ ]
for room in rooms :
if room not in self . map . keys ( ) :
print ( " Error: unknown room \" {} \" " . format ( room ) )
return { }
req . append ( self . map [ room ] [ " list_topic " ] )
c4 = C4Interface ( )
2017-03-15 09:53:00 +01:00
responce = c4 . pull ( req )
2017-03-12 15:13:41 +01:00
# Make responce iterable
if type ( responce ) != list : responce = [ responce ]
available = { }
for room in rooms :
for r in responce :
if r . topic == self . map [ room ] [ " list_topic " ] :
available [ room ] = json . decoder . JSONDecoder ( ) . decode ( r . payload . decode ( ) )
return available
def list_available ( self , room = " global " ) :
2017-04-06 16:33:32 +02:00
""" Print a list of available Presets. """
2017-03-12 15:13:41 +01:00
room = self . _expand_room_name ( room )
2017-03-12 16:24:01 +01:00
available = self . query_available ( [ room ] )
2017-03-12 15:13:41 +01:00
if not available :
print ( " No presets available for {} " . format ( self . map [ room ] [ " name " ] ) )
else :
print ( " Available presets for {} : " . format ( self . map [ room ] [ " name " ] ) )
for r in available . keys ( ) :
for preset in available [ r ] :
print ( " " + preset )
def apply_preset ( self , preset , rooms = [ " global " ] ) :
2017-04-06 16:33:32 +02:00
""" Apply preset to given rooms. """
2017-03-12 15:13:41 +01:00
# Strip spaces and expand rooms names
for i in range ( len ( rooms ) ) :
rooms [ i ] = self . _expand_room_name ( rooms [ i ] . strip ( ) )
available = self . query_available ( rooms . copy ( ) )
# Produce some fake data to prevent KeyErrors if in debug mode
if C4Interface . debug :
print ( " [DEBUG] Warning: handing over fake data to allow further execution! " )
available = {
" global " : [ preset ] ,
" wohnzimmer " : [ preset ] ,
" plenar " : [ preset ] ,
" fnord " : [ preset ] ,
" keller " : [ preset ]
}
# Expand preset name (stripping spaces)
preset = self . _expand_preset_name ( preset , rooms . copy ( ) , available . copy ( ) )
for room in rooms :
if preset not in available [ room ] and preset not in available [ " global " ] :
print ( " Error: preset \" {} \" not available for room \" {} \" ! " . format (
preset , self . map [ room ] [ " name " ] ) )
return False
cmd = [ ]
for room in rooms :
2017-03-12 16:24:01 +01:00
cmd . append ( ( self . map [ room ] [ " set_topic " ] , preset ) )
2017-03-12 15:13:41 +01:00
c4 = C4Interface ( )
2017-03-15 09:53:00 +01:00
return c4 . push ( cmd )
2017-03-12 15:13:41 +01:00
2017-03-11 14:04:02 +01:00
if __name__ == " __main__ " :
import argparse
parser = argparse . ArgumentParser (
description = " Command line client for AutoC4. " )
parser . add_argument (
" -d " , " --debug " , action = " store_true " ,
help = " display what would be send to the MQTT broker, but do not actually connect " )
2017-04-06 16:33:32 +02:00
2017-03-11 14:04:02 +01:00
# Various club functions
group_fn = parser . add_argument_group ( title = " various functions " )
group_fn . add_argument (
" -s " , " --status " , action = " store_true " ,
help = " display club status " )
group_fn . add_argument (
" -g " , " --gate " , action = " store_true " ,
help = " open club gate " )
group_fn . add_argument (
" -S " , " --shutdown " , action = " count " ,
help = " shutdown (give twice to force shutdown) " )
2017-04-06 16:33:32 +02:00
2017-03-11 14:04:02 +01:00
# Kitchenlight control
group_kl = parser . add_argument_group ( title = " Kitchenlight control " )
group_kl . add_argument (
" -k " , " --kl-mode " , type = str , metavar = " MODE[,OPTIONS] " ,
help = " set Kitchenlight to MODE " )
group_kl . add_argument (
2017-03-12 15:13:41 +01:00
" -i " , " --list-kl-modes " , action = " store_true " ,
help = " list available Kitchenlight modes and their options " )
2017-04-06 16:33:32 +02:00
2017-03-11 14:04:02 +01:00
# Ambient control
2017-03-12 15:13:41 +01:00
group_cl = parser . add_argument_group ( title = " ambient color control " ,
2017-03-26 00:22:14 +01:00
description = " PRESET may be either a preset name (which may be abbreviated), ' # ' followed by a color value in hex notation (eg. \" #ff0066 \" ) or ' - ' to read from stdin. " )
2017-03-11 14:04:02 +01:00
group_cl . add_argument (
" -w " , " --wohnzimmer " , type = str , dest = " w_color " , metavar = " PRESET " ,
2017-03-12 15:13:41 +01:00
help = " apply local colorscheme PRESET to Wohnzimmer " )
2017-03-11 14:04:02 +01:00
group_cl . add_argument (
" -p " , " --plenarsaal " , type = str , dest = " p_color " , metavar = " PRESET " ,
2017-03-12 15:13:41 +01:00
help = " apply local colorscheme PRESET to Plenarsaal " )
2017-03-11 14:04:02 +01:00
group_cl . add_argument (
" -f " , " --fnordcenter " , type = str , dest = " f_color " , metavar = " PRESET " ,
2017-03-12 15:13:41 +01:00
help = " apply local colorscheme PRESET to Fnordcenter " )
2017-03-17 23:27:54 +01:00
group_cl . add_argument (
2017-04-06 16:33:32 +02:00
" -N " , " --no-magic " , action = " store_true " ,
help = " Do not use fluffyd to change colors. " )
2017-03-11 14:04:02 +01:00
group_cl . add_argument (
2017-03-12 15:13:41 +01:00
" -l " , " --list-presets " , action = " store_true " ,
help = " list locally available presets " )
2017-03-11 14:04:02 +01:00
group_cl . add_argument (
" -o " , " --store-preset " , type = str , dest = " store_as " , metavar = " NAME " ,
2017-03-26 00:22:14 +01:00
help = " store current state as preset NAME ( ' - ' to write to stdout) " )
2017-04-06 16:33:32 +02:00
2017-03-11 14:04:02 +01:00
# Switch control
group_sw = parser . add_argument_group ( title = " light switch control " ,
2017-03-30 18:13:10 +02:00
description = " BINARY_CODE is a string of 0s or 1s for every light in the room. Accepts integers also. Will show some information and ask for input if omitted. " )
2017-03-11 14:04:02 +01:00
group_sw . add_argument (
2017-03-26 00:22:14 +01:00
" -W " , nargs = ' ? ' , dest = " w_switch " , const = " " , metavar = " BINARY_CODE " ,
2017-03-11 14:04:02 +01:00
help = " switch lights in Wohnzimmer on/off " )
group_sw . add_argument (
2017-03-26 00:22:14 +01:00
" -P " , nargs = ' ? ' , dest = " p_switch " , const = " " , metavar = " BINARY_CODE " ,
2017-03-11 14:04:02 +01:00
help = " switch lights in Plenarsaal on/off " )
group_sw . add_argument (
2017-03-26 00:22:14 +01:00
" -F " , nargs = ' ? ' , dest = " f_switch " , const = " " , metavar = " BINARY_CODE " ,
2017-03-11 14:04:02 +01:00
help = " switch lights in Fnordcentter on/off " )
group_sw . add_argument (
2017-03-26 00:22:14 +01:00
" -K " , nargs = ' ? ' , dest = " k_switch " , const = " " , metavar = " BINARY_CODE " ,
2017-03-11 14:04:02 +01:00
help = " switch lights in Keller on/off " )
2017-04-06 16:33:32 +02:00
2017-03-26 00:22:14 +01:00
# Remote presets
group_rp = parser . add_argument_group ( title = " remote preset functions " ,
description = " Available room names are \" wohnzimmer \" , \" plenar \" , \" fnord \" and \" keller \" . Preset and room names may be abbreviated. " )
group_rp . add_argument (
" -r " , " --remote-preset " , type = str , metavar = " PRESET[:ROOM[,ROOM,...]] " ,
help = " activate remote PRESET for ROOM(s). " )
group_rp . add_argument (
" -R " , " --list-remote " , nargs = ' ? ' , const = " global " , metavar = " ROOM " ,
help = " list remote presets for ROOM " )
2017-03-11 14:04:02 +01:00
args = parser . parse_args ( )
# Debug, gate, status and shutdown
if args . debug :
C4Interface . debug = True
if args . status :
2017-04-06 16:33:32 +02:00
status = C4Interface ( ) . status ( )
print ( " Club is " , status )
2017-03-11 14:04:02 +01:00
if args . gate :
C4Interface ( ) . open_gate ( )
if args . shutdown :
if args . shutdown > = 2 :
C4Interface ( ) . shutdown ( force = True )
else :
C4Interface ( ) . shutdown ( )
# Kitchenlight
2017-03-12 15:13:41 +01:00
if args . list_kl_modes :
2017-03-11 14:04:02 +01:00
print ( " Available Kitchenlight modes (options are optional): " )
print ( Kitchenlight . _available_modes )
if args . kl_mode :
kl = Kitchenlight ( )
mode = args . kl_mode . split ( ' , ' )
if len ( mode ) == 1 :
kl . set_mode ( mode [ 0 ] )
else :
kl . set_mode ( mode [ 0 ] , mode [ 1 : ] )
# Colorscheme
if args . store_as :
ColorScheme ( ) . store ( args . store_as )
2017-03-25 23:45:42 +01:00
presets = { } # Store and reuse initialized presets
2017-03-11 14:04:02 +01:00
if args . w_color :
2017-03-25 23:45:42 +01:00
if args . w_color not in presets :
presets [ args . w_color ] = ColorScheme ( autoinit = args . w_color )
2017-04-06 16:33:32 +02:00
if presets [ args . w_color ] : Wohnzimmer ( ) . set_colorscheme ( presets [ args . w_color ] , args . no_magic )
2017-03-11 14:04:02 +01:00
if args . p_color :
2017-03-25 23:45:42 +01:00
if args . p_color not in presets :
presets [ args . p_color ] = ColorScheme ( autoinit = args . p_color )
2017-04-06 16:33:32 +02:00
if presets [ args . p_color ] : Plenarsaal ( ) . set_colorscheme ( presets [ args . p_color ] , args . no_magic )
2017-03-11 14:04:02 +01:00
if args . f_color :
2017-03-25 23:45:42 +01:00
if args . f_color not in presets :
presets [ args . f_color ] = ColorScheme ( autoinit = args . f_color )
2017-04-06 16:33:32 +02:00
if presets [ args . f_color ] : Fnordcenter ( ) . set_colorscheme ( presets [ args . f_color ] , args . no_magic )
2017-03-11 14:04:02 +01:00
if args . list_presets :
ColorScheme ( ) . list_available ( )
2017-03-26 00:22:14 +01:00
# Light switches
if args . w_switch != None :
Wohnzimmer ( ) . light_switch ( args . w_switch )
if args . p_switch != None :
Plenarsaal ( ) . light_switch ( args . p_switch )
if args . f_switch != None :
Fnordcenter ( ) . light_switch ( args . f_switch )
if args . k_switch != None :
Keller ( ) . light_switch ( args . k_switch )
2017-03-12 15:13:41 +01:00
# Remote presets
if args . list_remote :
RemotePresets ( ) . list_available ( args . list_remote . lower ( ) )
if args . remote_preset :
remote_opts = args . remote_preset . split ( ' : ' )
if len ( remote_opts ) == 1 :
2017-03-12 16:24:01 +01:00
RemotePresets ( ) . apply_preset ( remote_opts [ 0 ] . strip ( ) )
2017-03-12 15:13:41 +01:00
else :
2017-03-12 16:24:01 +01:00
RemotePresets ( ) . apply_preset ( remote_opts [ 0 ] . strip ( ) ,
remote_opts [ 1 ] . lower ( ) . split ( ' , ' ) )
2017-03-12 15:13:41 +01:00
2017-03-21 12:08:00 +01:00
# No or no useful command line options?
2017-03-11 14:04:02 +01:00
if len ( sys . argv ) < = 1 or len ( sys . argv ) == 2 and args . debug :
parser . print_help ( )