forked from psung/zeya
-
Notifications
You must be signed in to change notification settings - Fork 1
/
zeyaclient.py
executable file
·98 lines (89 loc) · 3.31 KB
/
zeyaclient.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
#!/usr/bin/python
# -*- coding: utf-8 -*-
#
# Copyright (C) 2009 Phil Sung
#
# This file is part of Zeya.
#
# Zeya is free software: you can redistribute it and/or modify it under the
# terms of the GNU Affero General Public License as published by the Free
# Software Foundation, either version 3 of the License, or (at your option) any
# later version.
#
# Zeya is distributed in the hope that it will be useful, but WITHOUT ANY
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
# A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
# details.
#
# You should have received a copy of the GNU Affero General Public License
# along with Zeya. If not, see <http://www.gnu.org/licenses/>.
# Console frontend to Zeya server.
#
# Usage:
# zeyaclient.py http://server.local:8080
#
# The user is prompted for a query, and all songs matching it are played. The
# query may be matched against the title, artist, or album of the song.
import os
import readline # Modifies the behavior of raw_input.
import signal
import subprocess
import sys
import time
import urllib2
try:
import json
json.dumps
except (ImportError, AttributeError):
import simplejson as json
def song_matches(query, song):
"""
Return True if the query matches the given song.
A query might look like "help, the beatles"
A song is considered to match if each comma-separated component of the
query appears somewhere in one of the song's metadata fields. Matching is
case-insensitive.
"""
parts = [part.strip() for part in query.lower().split(",")]
return all(part in song['album'].lower() or part in song['title'].lower() \
or part in song['artist'].lower() for part in parts)
# TODO: refactor the parts that directly interact with the server into a
# separate module.
def run(server_path):
library_file = urllib2.urlopen(server_path + "/getlibrary")
library_data = json.loads(library_file.read())
print "Loaded %d songs from library." % (len(library_data),)
print 'You can issue queries like: "Beatles" or "help, the beatles"'
while True:
# Prompt user for a query...
try:
query = raw_input("\rQuery? ")
except:
# User pressed C-d or C-c.
print
break
if not query:
break
# ...then play all the songs we can find that match the query.
matching_songs = \
[song for song in library_data if song_matches(query, song)]
for song in matching_songs:
print "\r%s - %s" % (song['title'], song['artist'])
song_url = "%s/getcontent?key=%d" % (server_path, song['key'])
p = subprocess.Popen(["/usr/bin/ogg123", "-q", song_url])
try:
p.communicate()
except KeyboardInterrupt:
# After a single ^C, skip to the next song.
os.kill(p.pid, signal.SIGTERM)
try:
time.sleep(0.5)
except KeyboardInterrupt:
# If ^C^C is typed (within 0.5 sec) then break out back to
# the prompt.
break
if __name__ == "__main__":
if len(sys.argv) < 2:
print "Usage: %s http://server:8080" % (os.path.basename(sys.argv[0]),)
sys.exit(1)
run(sys.argv[1])