-
Notifications
You must be signed in to change notification settings - Fork 0
/
jamclient.py
executable file
·294 lines (252 loc) · 9.8 KB
/
jamclient.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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
#!/usr/bin/python
##########################################################################
###Below section is for pi recording code
from Tkinter import * #Note Tkinter for python 2.*, tkinter for python 3+
import subprocess
import signal
import os
##########################################################################
##########################################################################
###Below section is for client code
user_agent = "image uploader"
default_message = "Image $current of $total"
import logging
import os
from os.path import abspath, isabs, isdir, isfile, join
import random
import string
import sys
import mimetypes
import urllib2
import httplib
import time
import re
import time
import socket #may not be necessary
##########################################################################
##########################################################################
###Below is code to give each pi a sort of identity
###Used within upload function
with open("settings.txt") as f:
#reads all the lines of the file
#first line is address of the server
#second line is the name of the pi device
#third line is the link to a url for an icon for that pi
lines = f.readlines()
serverip, username, usericon = lines
##########################################################################
##########################################################################
###Below is code for pi recording
recording = False
#Set display sizes
WINDOW_W = 100
WINDOW_H = 100
def createDisplay():
###creates window with buttons and calls other functions accordingly
global tk, btn1, recordIcon, playIcon, exitIcon, stopIcon
tk = Tk()
stopIcon=PhotoImage(file="stop.gif")
playIcon=PhotoImage(file="play.gif")
recordIcon=PhotoImage(file="record.gif")
exitIcon=PhotoImage(file="exit.gif")
#Add a canvas area ready for drawing on
#canvas = Canvas(tk, width=WINDOW_W, height=WINDOW_H)
#canvas.pack()
#Add an exit button
tk.title('Jam')
f = ("Times", "48", "bold")
btn1 = Button(tk, font=f, image=recordIcon, text="Record", command=record, bg="#00aa00", activebackground="#006600")
btn1.pack()
btn2 = Button(tk, font=f, image=playIcon, text="Play", command=play, bg="#0000cc", activebackground="#000066")
btn2.pack()
btn3 = Button(tk, font=f, image=exitIcon, text="Exit", command=terminate, bg="#cc0000", activebackground="#880000")
btn3.pack()
# Start the tk main-loop (this updates the tk display)
tk.mainloop()
def record():
###function to change green LED light and start and stop recording and changes the icon for recording
global tk, soundfile, p, pid, recording, shell, btn1, recordIcon, stopIcon
if not recording:
recording = True
soundfile = open('Latest.wav', 'w')
p = subprocess.Popen(["arecord", "-D", "hw:1,0", "-f", "s16_le"], stdout=soundfile, shell=False)
os.system("echo none >/sys/class/leds/led0/trigger")
os.system("echo 1 >/sys/class/leds/led0/brightness")
btn1.config(image=stopIcon, text="Stop", bg="#cc0000", activebackground="#660000")
else:
# Get the process id
pid = p.pid
os.kill(pid, signal.SIGINT)
recording = False
os.system("echo none >/sys/class/leds/led0/trigger")
os.system("echo 0 >/sys/class/leds/led0/brightness")
btn1.config(image=recordIcon, text="Record", bg="#00cc00", activebackground="#006600")
if not p.poll():
print "Process correctly halted"
#then have client code sent to http server
upload("Latest.wav")
get_list()
def play():
###function to operate the play button, first op to select output, second op to play that recording
global tk
# 1 at end of next line is for output to headset, 2 is for output to HDMI
os.system("amixer cset numid=3 1")
# this plays the audio file
get_list()
with open("playlist.txt", 'r') as f:
digests = f.readlines()
f.close()
if len(digests) > 0:
print digests
os.system("aplay clips/" + digests[0].strip() + ".wav")
digests = digests[1:]
with open("playlist.txt", 'w') as f:
f.write("".join(digests))
f.close()
def terminate():
###function to kill the system
global tk
tk.destroy()
def main():
createDisplay()
###End of record pi code
##########################################################################
##########################################################################
###Below code is the code for the client
def random_string (length):
return ''.join (random.choice (string.letters) for ii in range (length + 1))
def encode_multipart_data (data, files):
boundary = random_string (30)
def get_content_type (filename):
return mimetypes.guess_type (filename)[0] or 'application/octet-stream'
def encode_field (field_name):
return ('--' + boundary,
'Content-Disposition: form-data; name="%s"' % field_name,
'', str (data [field_name]))
def encode_file (field_name):
filename = files [field_name]
return ('--' + boundary,
'Content-Disposition: form-data; name="%s"; filename="%s"' % (field_name, filename),
'Content-Type: %s' % get_content_type(filename),
'', open (filename, 'rb').read ())
lines = []
for name in data:
lines.extend (encode_field (name))
for name in files:
lines.extend (encode_file (name))
lines.extend (('--%s--' % boundary, ''))
body = '\r\n'.join (lines)
headers = {'content-type': 'multipart/form-data; boundary=' + boundary,
'content-length': str (len (body))}
return body, headers
def send_post (url, data, files):
req = urllib2.Request (url)
connection = httplib.HTTPConnection (req.get_host ())
connection.request ('POST', req.get_selector (),
*encode_multipart_data (data, files))
response = connection.getresponse ()
print ('response = %s', response.read ())
print ('Code: %s %s', response.status, response.reason)
def upload(filename):
#this function uploads the audio files to the server
#it uses the varibles serverip, username, and usericon taken from each pi's settings.txt file
#these variables are taken from the file in the beginning of the code
global serverip, username, usericon
print "made it to before upload"
send_post("http://"+serverip+"/submit", {"date":time.ctime(), "username":username, "usericon":usericon}, {"clip":filename})
print "made it to after upload"
###End of client code
##########################################################################
##########################################################################
###GET LIST OF AUDIO
# this function sends a get request to the server specified in the settings.txt file
# then receives a list of all the audio file names and splits by lines and saves as digest
# then iterates through the digest calling get_audio to retrieve files
def get_list ():
if os.path.exists("last.txt"):
with open("last.txt", 'r') as f:
last = f.readlines()[0]
f.close()
else:
last = "none"
if os.path.exists("playlist.txt"):
with open("playlist.txt", 'r') as f:
digests = f.readlines()
f.close()
else:
digests = []
if last == "none":
#if you haven't downloaded any audio files get them all
url = "http://"+serverip+"/list"
else:
#else request everything after the most recent one you have downloaded
url = "http://"+serverip+"/recent/" + last
req = urllib2.Request (url)
connection = httplib.HTTPConnection (req.get_host ())
connection.request ('GET', req.get_selector ())
response = connection.getresponse ()
data = response.read()
results = data.splitlines()
print "I got some results!"
for digest in results:
print "downloading new" + digest
#CALL GET_AUDIO HERE
get_audio("http://"+serverip+"/download/", digest+".wav")
if len(results) > 0:
with open("playlist.txt", 'a') as f:
print "appending " + str(results)
f.write("\n".join(results) + "\n")
f.close()
last = results[-1]
with open("last.txt", 'w') as f:
f.write(last + "\n")
f.close()
###############
###this code updates the list every s seconds
#s = 0
#while s<=30:
#print s, "seconds"
#s += 1
#time.sleep(1)
#if s == 30:
#get_list()
###############
print ('Code: %s %s', response.status, response.reason)
###end of get list
##########################################################################
##########################################################################
###GET AUDIO
# this function takes a specific line of the digest and checks to see if it already exists on the pi
# if so it returns to the get_list for loop
# else it makes a second get request to download that specific audio file
# then creates a file named after it and stores it within and closes it
# then returns to get_list for loop
def get_audio (urlpart, urlending):
if os.path.exists("clips/"+urlending):
return
else:
req = urllib2.Request (urlpart+urlending)
connection = httplib.HTTPConnection (req.get_host ())
connection.request ('GET', req.get_selector ())
response = connection.getresponse ()
data = response.read()
print len(data)
with open("clips/"+urlending, 'w') as f:
f.write(data)
f.close()
print ('Code: %s %s', response.status, response.reason)
###end of get audio
##########################################################################
get_list()
if __name__ == '__main__':
main()
##########################################
# NEXT CODE:
# should create code to try requesting new files every 3 to 5 minutes
# # i think i got this working??
# should keep track permanently of what you have listened to
# // should only download the ones from the list you havent already listened to
# // for the server: never send the same list to the same pi
# // aka the server keeps track of the last time each pi checks in
# change the urls from /lists to / archive and make a new /new or /recent to hold most recent files
##########################################