import socket import sys import struct import time import xml.etree.ElementTree as ET import threading import optparse from packet import Packet, CMD, itos parser = optparse.OptionParser() parser.add_option('-t', '--test', dest='test', action='store_true', help='Play a test tone (440, 880) on all clients in sequence (the last overlaps with the first of the next)') parser.add_option('-T', '--sync-test', dest='sync_test', action='store_true', help='Don\'t wait for clients to play tones properly--have them all test tone at the same time') parser.add_option('-q', '--quit', dest='quit', action='store_true', help='Instruct all clients to quit') parser.add_option('-f', '--factor', dest='factor', type='float', default=1.0, help='Rescale time by this factor (0'%(self.group, ('U' if self.map is uid_groups else 'T'), self.value) class RouteSet(object): def __init__(self, clis=None): if clis is None: clis = clients self.clients = clis self.routes = [] def Route(self, stream): grp = stream.get('group') if options.verbose: print 'Routing', grp, '...' excl = False for route in self.routes: if route.group == grp: if options.verbose: print 'Matches route', route excl = excl or route.excl matches = filter(lambda x, route=route: route.Apply(x), self.clients) if matches: if options.verbose: print 'Using client', matches[0] self.clients.remove(matches[0]) return matches[0] print 'No matches, moving on...' if excl: if options.verbose: print 'Exclusively routed, no route matched.' return None if not self.clients: if options.verbose: print 'Out of clients, no route matched.' return None cli = self.clients.pop(0) if options.verbose: print 'Default route to', cli return cli routeset = RouteSet() for rspec in options.routes: routeset.routes.extend(Route.Parse(rspec)) if options.verbose: print 'All routes:' for route in routeset.routes: print route class NSThread(threading.Thread): def wait_for(self, t): if t <= 0: return time.sleep(t) def run(self): nsq, cl = self._Thread__args for note in nsq: ttime = float(note.get('time')) pitch = int(note.get('pitch')) vel = int(note.get('vel')) dur = factor*float(note.get('dur')) while time.time() - BASETIME < factor*ttime: self.wait_for(factor*ttime - (time.time() - BASETIME)) s.sendto(str(Packet(CMD.PLAY, int(dur), int((dur*1000000)%1000000), int(440.0 * 2**((pitch-69)/12.0)), vel*2)), cl) if options.verbose: print (time.time() - BASETIME), cl, ': PLAY', pitch, dur, vel self.wait_for(dur - ((time.time() - BASETIME) - factor*ttime)) if options.verbose: print '% 6.5f'%(time.time() - BASETIME,), cl, ': DONE' threads = [] for ns in notestreams: cli = routeset.Route(ns) if cli: nsq = ns.findall('note') threads.append(NSThread(args=(nsq, clients.pop(0)))) BASETIME = time.time() for thr in threads: thr.start() for thr in threads: thr.join()