-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathlibconf_init.py
113 lines (90 loc) · 3.85 KB
/
libconf_init.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
"""
libconf_init Example
====================================================
This example demonstrates a simple method to read the shairport-sync.conf file and create an AirplayListener instance
according to the available configured backends.
"""
import time
import platform
import io
import libconf
import shairportmetadatareader
class MetadataNotEnabledError(Exception):
"""
Thrown if the metadata section inside the shairport-sync.conf is not enabled.
"""
# pylint: disable=W0107
pass
class NoBackendAvailableError(Exception):
"""
Thrown if neither the metadata pipe, nor the udp or mqtt backends are configured.
"""
# pylint: disable=W0107
pass
def get_default_config_path():
"""
:return: default path to the configuration file for the specific platform
"""
# See: https://github.com/mikebrady/shairport-sync
# Readme section: Choose the location of the configuration file
# A final consideration is the location of the configuration file shairport-sync.conf. This will be placed in the
# directory specified by the sysconfdir configuration variable, which defaults to /usr/local/etc. This is normal in
# BSD Unixes, but is unusual in Linux. Hence, for Linux installations, you need to set the sysconfdir variable to
# /etc using the configuration setting --sysconfdir=/etc.
if platform.system() == "Linux":
return "/etc/shairport-sync.conf"
return "/usr/local/etc/shairport-sync/shairport-sync.conf"
def get_listener_for_config_file(path):
"""
Create an AirplayListener instance according to a specific config file.
The UDP backend is the prefered method to use. If this backend isn't available the pipe backend is used. If neither
of those is configured this method falls back to use the MQTT backend.
:param path: path to the shairport-sync.conf file
:return: AirplayListener instance
"""
config_file = io.open(path)
config = libconf.load(config_file)
config_file.close()
# Make sure that the metadata section is configured correctly
metadata = config.metadata
mqtt = config.mqtt
if metadata["enabled"] != "yes":
raise MetadataNotEnabledError("Please set the `enabled` flag inside the `metadata` section to `yes`.")
# Make sure that at least one of the metadata provider is configured
has_pipe = metadata.get("pipe_name", "") != ""
has_udp = metadata.get("socket_address", "") != "" \
and metadata.get("socket_port", 0) > 0
has_mqtt = mqtt.get("enabled", "no") == "yes" \
and mqtt.get("hostname", "") != "" \
and mqtt.get("port", "") != "" \
and mqtt.get("publish_raw", "no") == "yes"
# For some reason the mqtt port is a string and not and integer
mqtt["port"] = int(mqtt["port"])
# Are any of the backends available
if not has_pipe and not has_udp and not has_mqtt:
raise NoBackendAvailableError("Please configure at least one metadata backend.")
# Create a listener instance according to the available backend
if has_udp:
return shairportmetadatareader.AirplayUDPListener(**metadata)
if has_pipe:
return shairportmetadatareader.AirplayPipeListener(**metadata)
if has_mqtt:
return shairportmetadatareader.AirplayMQTTListener(**mqtt)
return None
def on_track_info(lis, info):
"""
Print the current track information.
:param lis: listener instance
:param info: track information
"""
print("Listener: ", lis, "received information: ", info)
if __name__ == "__main__":
# Listen for metadata changes for 60 seconds
# pylint: disable=C0103
conf_path = get_default_config_path()
# pylint: disable=C0103
listener = get_listener_for_config_file(conf_path)
listener.bind(track_info=on_track_info)
listener.start_listening()
time.sleep(60)
listener.stop_listening()