import socket import sys import struct import time import xml.etree.ElementTree as ET import threading import optparse import random 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', '--transpose', dest='transpose', type='int', help='Transpose by a set amount of semitones (positive or negative)') parser.add_option('--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('-R', '--random', dest='random', type='float', help='Generate random notes at approximately this period') parser.add_option('--rand-low', dest='rand_low', type='int', help='Low frequency to randomly sample') parser.add_option('--rand-high', dest='rand_high', type='int', help='High frequency to randomly sample') parser.add_option('-l', '--live', dest='live', help='Enter live mode (play from a controller in real time), specifying the port to connect to as "client,port"; use just "," to manually subscribe later') parser.add_option('-L', '--list-live', dest='list_live', action='store_true', help='List all the clients and ports that can be connected to for live performance') parser.add_option('-q', '--quit', dest='quit', action='store_true', help='Instruct all clients to quit') parser.add_option('-p', '--play', dest='play', action='append', help='Play a single tone or chord (specified multiple times) on all listening clients (either "midi pitch" or "@frequency")') parser.add_option('-P', '--play-async', dest='play_async', action='store_true', help='Don\'t wait for the tone to finish using the local clock') parser.add_option('-D', '--duration', dest='duration', type='float', help='How long to play this note for') parser.add_option('-V', '--volume', dest='volume', type='int', help='How loud to play this note (0-255)') parser.add_option('-s', '--silence', dest='silence', action='store_true', help='Instruct all clients to stop playing any active tones') parser.add_option('-S', '--seek', dest='seek', type='float', help='Start time in seconds (scaled by --factor)') parser.add_option('-f', '--factor', dest='factor', type='float', help='Rescale time by this factor (0 0: while True: for cl in clients: s.sendto(str(Packet(CMD.PLAY, int(options.random), int(1000000*(options.random-int(options.random))), random.randint(options.rand_low, options.rand_high), 255)), cl) time.sleep(options.random) if options.live or options.list_live: import midi from midi import sequencer S = sequencer.S if options.list_live: print sequencer.SequencerHardware() exit() seq = sequencer.SequencerRead(sequencer_resolution=120) client_set = set(clients) active_set = {} # note (pitch) -> [client] deferred_set = set() # pitches held due to sustain sustain_status = False client, _, port = options.live.partition(',') if client or port: seq.subscribe_port(client, port) seq.start_sequencer() while True: ev = S.event_input(seq.client) event = None if ev: if options.verbose: print 'SEQ:', ev if ev < 0: seq._error(ev) if ev.type == S.SND_SEQ_EVENT_NOTEON: event = midi.NoteOnEvent(channel = ev.data.note.channel, pitch = ev.data.note.note, velocity = ev.data.note.velocity) elif ev.type == S.SND_SEQ_EVENT_NOTEOFF: event = midi.NoteOffEvent(channel = ev.data.note.channel, pitch = ev.data.note.note, velocity = ev.data.note.velocity) elif ev.type == S.SND_SEQ_EVENT_CONTROLLER: event = midi.ControlChangeEvent(channel = ev.data.control.channel, control = ev.data.control.param, value = ev.data.control.value) elif ev.type == S.SND_SEQ_EVENT_PGMCHANGE: event = midi.ProgramChangeEvent(channel = ev.data.control.channel, value = ev.data.control.value) elif ev.type == S.SND_SEQ_EVENT_PITCHBEND: event = midi.PitchWheelEvent(channel = ev.data.control.channel, pitch = ev.data.control.value) elif options.verbose: print 'WARNING: Unparsed event, type %r'%(ev.type,) continue if event is not None: if isinstance(event, midi.NoteOnEvent) and event.velocity == 0: event.__class__ = midi.NoteOffEvent if options.verbose: print 'EVENT:', event if isinstance(event, midi.NoteOnEvent): if event.pitch in active_set: if sustain_status: deferred_set.discard(event.pitch) inactive_set = client_set - set(sum(active_set.values(), [])) if not inactive_set: print 'WARNING: Out of clients to do note %r; dropped'%(event.pitch,) continue cli = sorted(inactive_set)[0] s.sendto(str(Packet(CMD.PLAY, 65535, 0, int(440.0 * 2**((event.pitch-69)/12.0)), 2*event.velocity)), cli) active_set.setdefault(event.pitch, []).append(cli) if options.verbose: print 'LIVE:', event.pitch, '+ =>', active_set[event.pitch] elif isinstance(event, midi.NoteOffEvent): if event.pitch not in active_set or not active_set[event.pitch]: print 'WARNING: Deactivating inactive note %r'%(event.pitch,) continue if sustain_status: deferred_set.add(event.pitch) continue cli = active_set[event.pitch].pop() s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), cli) if options.verbose: print 'LIVE:', event.pitch, '- =>', active_set[event.pitch] elif isinstance(event, midi.ControlChangeEvent): if event.control == 64: sustain_status = (event.value >= 64) if not sustain_status: for pitch in deferred_set: if pitch not in active_set or not active_set[pitch]: print 'WARNING: Attempted deferred removal of inactive note %r'%(pitch,) continue for cli in active_set[pitch]: s.sendto(str(Packet(CMD.PLAY, 0, 1, 1, 0)), cli) del active_set[pitch] deferred_set.clear() for fname in args: try: iv = ET.parse(fname).getroot() except IOError: import traceback traceback.print_exc() print fname, ': Bad file' continue notestreams = iv.findall("./streams/stream[@type='ns']") groups = set([ns.get('group') for ns in notestreams if 'group' in ns.keys()]) print len(notestreams), 'notestreams' print len(clients), 'clients' print len(groups), 'groups' class Route(object): def __init__(self, fattr, fvalue, group, excl=False): if fattr == 'U': self.map = uid_groups elif fattr == 'T': self.map = type_groups else: raise ValueError('Not a valid attribute specifier: %r'%(fattr,)) self.value = fvalue if group is not None and group not in groups: raise ValueError('Not a present group: %r'%(group,)) self.group = group self.excl = excl @classmethod def Parse(cls, s): fspecs, _, grpspecs = map(lambda x: x.strip(), s.partition('=')) fpairs = [] ret = [] for fspec in [i.strip() for i in fspecs.split(',')]: fattr, _, fvalue = map(lambda x: x.strip(), fspec.partition(':')) fpairs.append((fattr, fvalue)) for part in [i.strip() for i in grpspecs.split(',')]: for fattr, fvalue in fpairs: if part[0] == '+': ret.append(Route(fattr, fvalue, part[1:], False)) elif part[0] == '-': ret.append(Route(fattr, fvalue, part[1:], True)) elif part[0] == '0': ret.append(Route(fattr, fvalue, None, True)) else: raise ValueError('Not an exclusivity: %r'%(part[0],)) return ret def Apply(self, cli): return cli in self.map.get(self.value, []) def __repr__(self): return ''%(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): testset = self.clients[:] grp = stream.get('group', 'ALL') if options.verbose: print 'Routing', grp, '...' excl = False for route in self.routes: if route.group == grp: if options.verbose: print '\tMatches route', route excl = excl or route.excl matches = filter(lambda x, route=route: route.Apply(x), testset) if matches: if options.verbose: print '\tUsing client', matches[0] self.clients.remove(matches[0]) return matches[0] if options.verbose: print '\tNo matches, moving on...' if route.group is None: if options.verbose: print 'Encountered NULL route, removing from search space...' toremove = [] for cli in testset: if route.Apply(cli): toremove.append(cli) for cli in toremove: if options.verbose: print '\tRemoving', cli, '...' testset.remove(cli) if excl: if options.verbose: print '\tExclusively routed, no route matched.' return None if not testset: if options.verbose: print '\tOut of clients, no route matched.' return None cli = testset[0] self.clients.remove(cli) if options.verbose: print '\tDefault route to', cli return cli routeset = RouteSet() for rspec in options.routes: try: routeset.routes.extend(Route.Parse(rspec)) except Exception: import traceback traceback.print_exc() if options.verbose: print 'All routes:' for route in routeset.routes: print route class NSThread(threading.Thread): def drop_missed(self): nsq, cl = self._Thread__args cnt = 0 while nsq and float(nsq[0].get('time'))*factor < time.time() - BASETIME: nsq.pop(0) cnt += 1 if options.verbose: print self, 'dropped', cnt, 'notes due to miss' self._Thread__args = (nsq, cl) 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')) + options.transpose 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 * (options.volume / 255.0))), 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, cli))) if options.verbose: print 'Playback threads:' for thr in threads: print thr._Thread__args[1] BASETIME = time.time() - (options.seek*factor) if options.seek > 0: for thr in threads: thr.drop_missed() for thr in threads: thr.start() for thr in threads: thr.join() print fname, ': Done!'